diff --git a/.noir-sync-commit b/.noir-sync-commit index a6780a1e9ef..95a26c1ef5b 100644 --- a/.noir-sync-commit +++ b/.noir-sync-commit @@ -1 +1 @@ -f337992de96ef656681ebfc96a30c2c9c9b82a70 \ No newline at end of file +dd7084545dfd93a07599fc10676b6c8ec1e3d458 diff --git a/noir/noir-repo/.github/critical_libraries_status/.gitignore b/noir/noir-repo/.github/critical_libraries_status/.gitignore new file mode 100644 index 00000000000..38a3cf9caf1 --- /dev/null +++ b/noir/noir-repo/.github/critical_libraries_status/.gitignore @@ -0,0 +1,3 @@ +.actual.jsonl +.actual.jsonl.jq +.failures.jsonl.jq \ No newline at end of file diff --git a/noir/noir-repo/test_programs/execution_success/cast_and_shift_global/Prover.toml b/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/aztec-nr.failures.jsonl similarity index 100% rename from noir/noir-repo/test_programs/execution_success/cast_and_shift_global/Prover.toml rename to noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/aztec-nr.failures.jsonl diff --git a/noir/noir-repo/test_programs/execution_success/trait_inheritance/Prover.toml b/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-contracts.failures.jsonl similarity index 100% rename from noir/noir-repo/test_programs/execution_success/trait_inheritance/Prover.toml rename to noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-contracts.failures.jsonl diff --git a/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/blob.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/blob.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/parity-lib.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/parity-lib.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/private-kernel-lib.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/private-kernel-lib.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/reset-kernel-lib.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/reset-kernel-lib.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/rollup-lib.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/rollup-lib.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/types.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/AztecProtocol/aztec-packages/noir-projects/noir-protocol-circuits/crates/types.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/README.md b/noir/noir-repo/.github/critical_libraries_status/README.md new file mode 100644 index 00000000000..47e9d7ad6ed --- /dev/null +++ b/noir/noir-repo/.github/critical_libraries_status/README.md @@ -0,0 +1,20 @@ +# Critical Libraries Status + +This directory contains one `.failures.jsonl` file per external directory that is checked by CI. +CI will run the external repository tests and compare the test failures against those recorded +in these files. If there's a difference, CI will fail. + +This allows us to mark some tests as expected to fail if we introduce breaking changes. +When tests are fixed on the external repository, CI will let us know that we need to remove +the `.failures.jsonl` failures on our side. + +The format of the `.failures.jsonl` files is one JSON per line with a failure: + +```json +{"suite":"one","name":"foo"} +``` + +If it's expected that an external repository doesn't compile (because a PR introduces breaking changes +to, say, the type system) you can remove the `.failures.jsonl` file for that repository and CI +will pass again. Once the repository compiles again, CI will let us know and require us to put +back the `.failures.jsonl` file. \ No newline at end of file diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/ec/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/ec/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/eddsa/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/eddsa/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/mimc/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/mimc/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir-bignum/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir-bignum/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir-edwards/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir-edwards/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_base64/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_base64/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_bigcurve/.failures.jsonl.does_not_compile b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_bigcurve/.failures.jsonl.does_not_compile new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_check_shuffle/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_check_shuffle/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_json_parser/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_json_parser/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_rsa/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_rsa/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_sort/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_sort/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_string_search/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/noir_string_search/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/schnorr/.failures.jsonl b/noir/noir-repo/.github/critical_libraries_status/noir-lang/schnorr/.failures.jsonl new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/critical_libraries_status/noir-lang/sparse_array/.failures.jsonl.does_not_compile b/noir/noir-repo/.github/critical_libraries_status/noir-lang/sparse_array/.failures.jsonl.does_not_compile new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/.github/scripts/check_test_results.sh b/noir/noir-repo/.github/scripts/check_test_results.sh new file mode 100755 index 00000000000..25833387990 --- /dev/null +++ b/noir/noir-repo/.github/scripts/check_test_results.sh @@ -0,0 +1,39 @@ +#!/bin/bash +set -eu + +# Usage: ./check_test_results.sh +# Compares the output of two test results of the same repository. +# If any of the files doesn't exist or is empty, the script will consider that the test suite +# couldn't be compiled. + +function process_json_lines() { + cat $1 | jq -c 'select(.type == "test" and .event == "failed") | {suite: .suite, name: .name}' | jq -s -c 'sort_by(.suite, .name) | .[]' > $1.jq +} + +if [ -f $1 ] && [ -f $2 ]; then + # Both files exist, let's compare them + $(process_json_lines $2) + if ! diff $1 $2.jq; then + echo "Error: test failures don't match expected failures" + echo "Lines prefixed with '>' are new test failures (you could add them to '$1')" + echo "Lines prefixed with '<' are tests that were expected to fail but passed (you could remove them from '$1')" + exit -1 + fi +elif [ -f $1 ]; then + # Only the expected file exists, which means the actual test couldn't be compiled. + echo "Error: external library tests couldn't be compiled." + echo "You could rename '$1' to '$1.does_not_compile' if it's expected that the external library can't be compiled." + exit -1 +elif [ -f $2 ]; then + # Only the actual file exists, which means we are expecting the external library + # not to compile but it did. + echo "Error: expected external library not to compile, but it did." + echo "You could create '$1' with these contents:" + $(process_json_lines $2) + cat $2.jq + exit -1 +else + # Both files don't exists, which means we are expecting the external library not + # to compile, and it didn't, so all is good. + exit 0 +fi diff --git a/noir/noir-repo/.github/workflows/docs-pr.yml b/noir/noir-repo/.github/workflows/docs-pr.yml index fdd4d25f5ae..0d47176cc00 100644 --- a/noir/noir-repo/.github/workflows/docs-pr.yml +++ b/noir/noir-repo/.github/workflows/docs-pr.yml @@ -55,7 +55,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: diff --git a/noir/noir-repo/.github/workflows/formatting.yml b/noir/noir-repo/.github/workflows/formatting.yml index f8ebd53dc70..4e836ef2493 100644 --- a/noir/noir-repo/.github/workflows/formatting.yml +++ b/noir/noir-repo/.github/workflows/formatting.yml @@ -25,7 +25,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 with: targets: x86_64-unknown-linux-gnu components: clippy, rustfmt @@ -51,7 +51,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 with: targets: x86_64-unknown-linux-gnu components: clippy, rustfmt @@ -89,7 +89,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: @@ -143,3 +143,29 @@ jobs: - name: Format test suite working-directory: ./test_programs run: ./format.sh check + + # This is a job which depends on all test jobs and reports the overall status. + # This allows us to add/remove test jobs without having to update the required workflows. + formatting-end: + name: Formatting End + runs-on: ubuntu-22.04 + # We want this job to always run (even if the dependant jobs fail) as we want this job to fail rather than skipping. + if: ${{ always() }} + needs: + - clippy + - rustfmt + - eslint + - nargo_fmt + + steps: + - name: Report overall success + run: | + if [[ $FAIL == true ]]; then + exit 1 + else + exit 0 + fi + env: + # We treat any skipped or failing jobs as a failure for the workflow as a whole. + FAIL: ${{ contains(needs.*.result, 'failure') || contains(needs.*.result, 'cancelled') || contains(needs.*.result, 'skipped') }} + diff --git a/noir/noir-repo/.github/workflows/publish-acvm.yml b/noir/noir-repo/.github/workflows/publish-acvm.yml index fb2e2001e40..27d927a67d1 100644 --- a/noir/noir-repo/.github/workflows/publish-acvm.yml +++ b/noir/noir-repo/.github/workflows/publish-acvm.yml @@ -18,7 +18,7 @@ jobs: ref: ${{ inputs.noir-ref }} - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 # These steps are in a specific order so crate dependencies are updated first - name: Publish acir_field @@ -74,4 +74,4 @@ jobs: WORKFLOW_URL: ${{ github.server_url }}/${{ github.repository }}/actions/runs/${{ github.run_id }} with: update_existing: true - filename: .github/ACVM_PUBLISH_FAILED.md \ No newline at end of file + filename: .github/ACVM_PUBLISH_FAILED.md diff --git a/noir/noir-repo/.github/workflows/publish-es-packages.yml b/noir/noir-repo/.github/workflows/publish-es-packages.yml index e629ae1f133..76c6fce6d5e 100644 --- a/noir/noir-repo/.github/workflows/publish-es-packages.yml +++ b/noir/noir-repo/.github/workflows/publish-es-packages.yml @@ -24,7 +24,7 @@ jobs: ref: ${{ inputs.noir-ref }} - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: @@ -58,7 +58,7 @@ jobs: ref: ${{ inputs.noir-ref }} - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: @@ -95,7 +95,7 @@ jobs: ref: ${{ inputs.noir-ref }} - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: diff --git a/noir/noir-repo/.github/workflows/publish-nargo.yml b/noir/noir-repo/.github/workflows/publish-nargo.yml index fa0b6f2d9fb..d7d9c1ea03e 100644 --- a/noir/noir-repo/.github/workflows/publish-nargo.yml +++ b/noir/noir-repo/.github/workflows/publish-nargo.yml @@ -46,7 +46,7 @@ jobs: echo "MACOSX_DEPLOYMENT_TARGET=$(xcrun -sdk macosx$(sw_vers -productVersion) --show-sdk-platform-version)" >> $GITHUB_ENV - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 with: targets: ${{ matrix.target }} @@ -120,7 +120,7 @@ jobs: ref: ${{ inputs.tag || env.GITHUB_REF }} - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 with: targets: ${{ matrix.target }} diff --git a/noir/noir-repo/.github/workflows/reports.yml b/noir/noir-repo/.github/workflows/reports.yml index 1e355dc9e6b..b1a74520a66 100644 --- a/noir/noir-repo/.github/workflows/reports.yml +++ b/noir/noir-repo/.github/workflows/reports.yml @@ -15,7 +15,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: @@ -76,7 +76,7 @@ jobs: - name: Compare gates reports id: gates_diff - uses: noir-lang/noir-gates-diff@1931aaaa848a1a009363d6115293f7b7fc72bb87 + uses: noir-lang/noir-gates-diff@7e4ddaa91c69380f15ccba514eac17bc7432a8cc with: report: gates_report.json summaryQuantile: 0.9 # only display the 10% most significant circuit size diffs in the summary (defaults to 20%) @@ -116,12 +116,23 @@ jobs: - name: Generate Brillig bytecode size report working-directory: ./test_programs run: | - ./gates_report_brillig.sh - mv gates_report_brillig.json ../gates_report_brillig.json + mkdir ./reports + + ./gates_report_brillig.sh 9223372036854775807 + jq '.programs |= map(.package_name |= (. + "_inliner_max"))' gates_report_brillig.json > ./reports/gates_report_brillig_inliner_max.json + + ./gates_report_brillig.sh 0 + jq '.programs |= map(.package_name |= (. + "_inliner_zero"))' gates_report_brillig.json > ./reports/gates_report_brillig_inliner_zero.json + + ./gates_report_brillig.sh -9223372036854775808 + jq '.programs |= map(.package_name |= (. + "_inliner_min"))' gates_report_brillig.json > ./reports/gates_report_brillig_inliner_min.json + + # Merge all reports + jq -s '{ programs: map(.programs) | add }' ./reports/* > ../gates_report_brillig.json - name: Compare Brillig bytecode size reports id: brillig_bytecode_diff - uses: noir-lang/noir-gates-diff@d88f7523b013b9edd3f31c5cfddaef87a3fe1b48 + uses: noir-lang/noir-gates-diff@7e4ddaa91c69380f15ccba514eac17bc7432a8cc with: report: gates_report_brillig.json header: | @@ -165,12 +176,23 @@ jobs: - name: Generate Brillig execution report working-directory: ./test_programs run: | - ./gates_report_brillig_execution.sh - mv gates_report_brillig_execution.json ../gates_report_brillig_execution.json + mkdir ./reports + + ./gates_report_brillig_execution.sh 9223372036854775807 + jq '.programs |= map(.package_name |= (. + "_inliner_max"))' gates_report_brillig_execution.json > ./reports/gates_report_brillig_execution_inliner_max.json + + ./gates_report_brillig_execution.sh 0 + jq '.programs |= map(.package_name |= (. + "_inliner_zero"))' gates_report_brillig_execution.json > ./reports/gates_report_brillig_execution_inliner_zero.json + + ./gates_report_brillig_execution.sh -9223372036854775808 + jq '.programs |= map(.package_name |= (. + "_inliner_min"))' gates_report_brillig_execution.json > ./reports/gates_report_brillig_execution_inliner_min.json + + # Merge all reports + jq -s '{ programs: map(.programs) | add }' ./reports/* > ../gates_report_brillig_execution.json - name: Compare Brillig execution reports id: brillig_execution_diff - uses: noir-lang/noir-gates-diff@d88f7523b013b9edd3f31c5cfddaef87a3fe1b48 + uses: noir-lang/noir-gates-diff@c1503343c3e264925ef67c68a2a5eeadd245a77b with: report: gates_report_brillig_execution.json header: | @@ -217,16 +239,30 @@ jobs: ./memory_report.sh mv memory_report.json ../memory_report.json - - name: Upload memory report + - name: Upload compilation memory report uses: actions/upload-artifact@v4 with: - name: in_progress_memory_report + name: in_progress_compilation_mem_report path: memory_report.json retention-days: 3 overwrite: true - generate_compilation_report: - name: Compilation time + - name: Generate execution memory report + working-directory: ./test_programs + run: | + ./memory_report.sh 0 1 + mv memory_report.json ../memory_report.json + + - name: Upload execution memory report + uses: actions/upload-artifact@v4 + with: + name: in_progress_execution_mem_report + path: memory_report.json + retention-days: 3 + overwrite: true + + generate_compilation_and_execution_report: + name: Compilation and execution time needs: [build-nargo] runs-on: ubuntu-22.04 permissions: @@ -252,10 +288,15 @@ jobs: - name: Generate Compilation report working-directory: ./test_programs run: | - ./compilation_report.sh - cat compilation_report.json + ./compilation_report.sh 0 1 mv compilation_report.json ../compilation_report.json + - name: Generate Execution report + working-directory: ./test_programs + run: | + ./execution_report.sh 0 1 + mv execution_report.json ../execution_report.json + - name: Upload compilation report uses: actions/upload-artifact@v4 with: @@ -263,24 +304,37 @@ jobs: path: compilation_report.json retention-days: 3 overwrite: true + + - name: Upload execution report + uses: actions/upload-artifact@v4 + with: + name: in_progress_execution_report + path: execution_report.json + retention-days: 3 + overwrite: true - external_repo_compilation_report: + external_repo_compilation_and_execution_report: needs: [build-nargo] - runs-on: ubuntu-latest + runs-on: ubuntu-22.04 timeout-minutes: 15 strategy: fail-fast: false matrix: include: - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-contracts } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/parity-root } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-inner } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-tail } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-reset } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-base-private } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-base-public } - - name: External repo compilation report - ${{ matrix.project.repo }}/${{ matrix.project.path }} + # - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-contracts, cannot_execute: true } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-inner, num_runs: 5 } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-tail, num_runs: 5 } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-reset, num_runs: 5 } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-base-private, num_runs: 5 } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-base-public, num_runs: 5 } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-block-merge, num_runs: 5 } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-block-root-empty, num_runs: 5, cannot_execute: true } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-block-root-single-tx, num_runs: 1, flags: "--skip-brillig-constraints-check --skip-underconstrained-check", cannot_execute: true } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-block-root, num_runs: 1, flags: "--skip-brillig-constraints-check --skip-underconstrained-check" } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-merge, num_runs: 5 } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-root, num_runs: 5 } + + name: External repo compilation and execution reports - ${{ matrix.project.repo }}/${{ matrix.project.path }} steps: - name: Download nargo binary uses: actions/download-artifact@v4 @@ -301,6 +355,8 @@ jobs: path: scripts sparse-checkout: | test_programs/compilation_report.sh + test_programs/execution_report.sh + test_programs/parse_time.sh sparse-checkout-cone-mode: false - name: Checkout @@ -314,11 +370,23 @@ jobs: working-directory: ./test-repo/${{ matrix.project.path }} run: | mv /home/runner/work/noir/noir/scripts/test_programs/compilation_report.sh ./compilation_report.sh - chmod +x ./compilation_report.sh - ./compilation_report.sh 1 + touch parse_time.sh + chmod +x parse_time.sh + cp /home/runner/work/noir/noir/scripts/test_programs/parse_time.sh ./parse_time.sh + ./compilation_report.sh 1 ${{ matrix.project.num_runs }} + env: + FLAGS: ${{ matrix.project.flags }} + + - name: Generate execution report + working-directory: ./test-repo/${{ matrix.project.path }} + if: ${{ !matrix.project.cannot_execute }} + run: | + mv /home/runner/work/noir/noir/scripts/test_programs/execution_report.sh ./execution_report.sh + mv /home/runner/work/noir/noir/scripts/test_programs/parse_time.sh ./parse_time.sh + ./execution_report.sh 1 ${{ matrix.project.num_runs }} - name: Move compilation report - id: report + id: compilation_report shell: bash run: | PACKAGE_NAME=${{ matrix.project.path }} @@ -326,22 +394,44 @@ jobs: mv ./test-repo/${{ matrix.project.path }}/compilation_report.json ./compilation_report_$PACKAGE_NAME.json echo "compilation_report_name=$PACKAGE_NAME" >> $GITHUB_OUTPUT + - name: Move execution report + id: execution_report + shell: bash + if: ${{ !matrix.project.cannot_execute }} + run: | + PACKAGE_NAME=${{ matrix.project.path }} + PACKAGE_NAME=$(basename $PACKAGE_NAME) + mv ./test-repo/${{ matrix.project.path }}/execution_report.json ./execution_report_$PACKAGE_NAME.json + echo "execution_report_name=$PACKAGE_NAME" >> $GITHUB_OUTPUT + - name: Upload compilation report uses: actions/upload-artifact@v4 with: - name: compilation_report_${{ steps.report.outputs.compilation_report_name }} - path: compilation_report_${{ steps.report.outputs.compilation_report_name }}.json + name: compilation_report_${{ steps.compilation_report.outputs.compilation_report_name }} + path: compilation_report_${{ steps.compilation_report.outputs.compilation_report_name }}.json + retention-days: 3 + overwrite: true + + - name: Upload execution report + uses: actions/upload-artifact@v4 + with: + name: execution_report_${{ steps.execution_report.outputs.execution_report_name }} + path: execution_report_${{ steps.execution_report.outputs.execution_report_name }}.json retention-days: 3 overwrite: true upload_compilation_report: name: Upload compilation report - needs: [generate_compilation_report, external_repo_compilation_report] - # We want this job to run even if one variation of the matrix in `external_repo_compilation_report` fails + needs: [generate_compilation_and_execution_report, external_repo_compilation_and_execution_report] + # We want this job to run even if one variation of the matrix in `external_repo_compilation_and_execution_report` fails if: always() - runs-on: ubuntu-latest + runs-on: ubuntu-22.04 permissions: pull-requests: write + # deployments permission to deploy GitHub pages website + deployments: write + # contents permission to update benchmark contents in gh-pages branch + contents: write steps: - uses: actions/checkout@v4 @@ -364,11 +454,11 @@ jobs: - name: Parse compilation report id: compilation_report - uses: noir-lang/noir-bench-report@0d7464a8c39170523932d7846b6e6b458a294aea + uses: noir-lang/noir-bench-report@6ba151d7795042c4ff51864fbeb13c0a6a79246c with: report: compilation_report.json header: | - # Compilation Report + Compilation Report memory_report: false - name: Add memory report to sticky comment @@ -378,9 +468,29 @@ jobs: header: compilation message: ${{ steps.compilation_report.outputs.markdown }} + - name: Convert to `benchmark-action` format + run: | + jq ".compilation_reports | map({name: .artifact_name, value: (.time[:-1] | tonumber), unit: \"s\"}) " ./compilation_report.json > time_bench.json + + - name: Store benchmark result + continue-on-error: true + uses: benchmark-action/github-action-benchmark@4de1bed97a47495fc4c5404952da0499e31f5c29 + with: + name: "Compilation Time" + tool: "customSmallerIsBetter" + output-file-path: ./time_bench.json + github-token: ${{ secrets.GITHUB_TOKEN }} + # We want this to only run on master to avoid garbage data from PRs being added. + auto-push: ${{ github.ref == 'refs/heads/master' }} + alert-threshold: "110%" + comment-on-alert: true + fail-on-alert: false + alert-comment-cc-users: "@TomAFrench" + max-items-in-chart: 50 + external_repo_memory_report: needs: [build-nargo] - runs-on: ubuntu-latest + runs-on: ubuntu-22.04 timeout-minutes: 30 strategy: fail-fast: false @@ -388,10 +498,17 @@ jobs: include: # TODO: Bring this report back under a flag. The `noir-contracts` report takes just under 30 minutes. # - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-contracts } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/parity-root } - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-inner } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-tail } - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-reset } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-tail } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-base-private } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-base-public } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-block-merge } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-block-root-empty, cannot_execute: true } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-block-root-single-tx, flags: "--skip-brillig-constraints-check --skip-underconstrained-check", cannot_execute: true } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-block-root, flags: "--skip-brillig-constraints-check --skip-underconstrained-check" } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-merge } + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-root } name: External repo memory report - ${{ matrix.project.repo }}/${{ matrix.project.path }} steps: @@ -414,6 +531,7 @@ jobs: path: scripts sparse-checkout: | test_programs/memory_report.sh + test_programs/parse_memory.sh sparse-checkout-cone-mode: false - name: Checkout @@ -423,15 +541,43 @@ jobs: path: test-repo ref: ${{ matrix.project.ref }} - - name: Generate compilation report + - name: Generate compilation memory report working-directory: ./test-repo/${{ matrix.project.path }} run: | mv /home/runner/work/noir/noir/scripts/test_programs/memory_report.sh ./memory_report.sh - chmod +x ./memory_report.sh + mv /home/runner/work/noir/noir/scripts/test_programs/parse_memory.sh ./parse_memory.sh ./memory_report.sh 1 + # Rename the memory report as the execution report is about to write to the same file + cp memory_report.json compilation_memory_report.json + env: + FLAGS: ${{ matrix.project.flags }} + + - name: Generate execution memory report + working-directory: ./test-repo/${{ matrix.project.path }} + if: ${{ !matrix.project.cannot_execute }} + run: | + ./memory_report.sh 1 1 - name: Move compilation report - id: report + id: compilation_mem_report + shell: bash + run: | + PACKAGE_NAME=${{ matrix.project.path }} + PACKAGE_NAME=$(basename $PACKAGE_NAME) + mv ./test-repo/${{ matrix.project.path }}/compilation_memory_report.json ./memory_report_$PACKAGE_NAME.json + echo "memory_report_name=$PACKAGE_NAME" >> $GITHUB_OUTPUT + + - name: Upload compilation memory report + uses: actions/upload-artifact@v4 + with: + name: compilation_mem_report_${{ steps.compilation_mem_report.outputs.memory_report_name }} + path: memory_report_${{ steps.compilation_mem_report.outputs.memory_report_name }}.json + retention-days: 3 + overwrite: true + + - name: Move execution report + id: execution_mem_report + if: ${{ !matrix.project.cannot_execute }} shell: bash run: | PACKAGE_NAME=${{ matrix.project.path }} @@ -439,22 +585,26 @@ jobs: mv ./test-repo/${{ matrix.project.path }}/memory_report.json ./memory_report_$PACKAGE_NAME.json echo "memory_report_name=$PACKAGE_NAME" >> $GITHUB_OUTPUT - - name: Upload memory report + - name: Upload execution memory report uses: actions/upload-artifact@v4 with: - name: memory_report_${{ steps.report.outputs.memory_report_name }} - path: memory_report_${{ steps.report.outputs.memory_report_name }}.json + name: execution_mem_report_${{ steps.execution_mem_report.outputs.memory_report_name }} + path: memory_report_${{ steps.execution_mem_report.outputs.memory_report_name }}.json retention-days: 3 overwrite: true - upload_memory_report: - name: Upload memory report + upload_compilation_memory_report: + name: Upload compilation memory report needs: [generate_memory_report, external_repo_memory_report] # We want this job to run even if one variation of the matrix in `external_repo_memory_report` fails if: always() - runs-on: ubuntu-latest + runs-on: ubuntu-22.04 permissions: pull-requests: write + # deployments permission to deploy GitHub pages website + deployments: write + # contents permission to update benchmark contents in gh-pages branch + contents: write steps: - uses: actions/checkout@v4 @@ -462,12 +612,12 @@ jobs: - name: Download initial memory report uses: actions/download-artifact@v4 with: - name: in_progress_memory_report + name: in_progress_compilation_mem_report - name: Download matrix memory reports uses: actions/download-artifact@v4 with: - pattern: memory_report_* + pattern: compilation_mem_report_* path: ./reports - name: Merge memory reports using jq @@ -475,64 +625,177 @@ jobs: mv ./.github/scripts/merge-bench-reports.sh merge-bench-reports.sh ./merge-bench-reports.sh memory_report - - name: Parse memory report - id: memory_report - uses: noir-lang/noir-bench-report@0d7464a8c39170523932d7846b6e6b458a294aea + - name: Parse compilation memory report + id: compilation_mem_report + uses: noir-lang/noir-bench-report@6ba151d7795042c4ff51864fbeb13c0a6a79246c with: report: memory_report.json header: | - # Memory Report + Compilation Memory Report memory_report: true - name: Add memory report to sticky comment if: github.event_name == 'pull_request' || github.event_name == 'pull_request_target' uses: marocchino/sticky-pull-request-comment@v2 with: - header: memory - message: ${{ steps.memory_report.outputs.markdown }} + header: compilation_memory + message: ${{ steps.compilation_mem_report.outputs.markdown }} - generate_compilation_report: - name: Compilation time - needs: [build-nargo] + - name: Convert to `benchmark-action` format + run: | + jq ".memory_reports | map({name: .artifact_name, value: (.peak_memory | tonumber), unit: \"MB\"}) " ./memory_report.json > memory_bench.json + + - name: Store benchmark result + continue-on-error: true + uses: benchmark-action/github-action-benchmark@4de1bed97a47495fc4c5404952da0499e31f5c29 + with: + name: "Compilation Memory" + tool: "customSmallerIsBetter" + output-file-path: ./memory_bench.json + github-token: ${{ secrets.GITHUB_TOKEN }} + # We want this to only run on master to avoid garbage data from PRs being added. + auto-push: ${{ github.ref == 'refs/heads/master' }} + alert-threshold: "110%" + comment-on-alert: true + fail-on-alert: false + alert-comment-cc-users: "@TomAFrench" + max-items-in-chart: 50 + + upload_execution_memory_report: + name: Upload execution memory report + needs: [generate_memory_report, external_repo_memory_report] + # We want this job to run even if one variation of the matrix in `external_repo_memory_report` fails + if: always() runs-on: ubuntu-22.04 permissions: pull-requests: write + # deployments permission to deploy GitHub pages website + deployments: write + # contents permission to update benchmark contents in gh-pages branch + contents: write steps: - uses: actions/checkout@v4 - - name: Download nargo binary + - name: Download initial memory report uses: actions/download-artifact@v4 with: - name: nargo - path: ./nargo + name: in_progress_execution_mem_report - - name: Set nargo on PATH + - name: Download matrix memory reports + uses: actions/download-artifact@v4 + with: + pattern: execution_mem_report_* + path: ./reports + + - name: Merge memory reports using jq run: | - nargo_binary="${{ github.workspace }}/nargo/nargo" - chmod +x $nargo_binary - echo "$(dirname $nargo_binary)" >> $GITHUB_PATH - export PATH="$PATH:$(dirname $nargo_binary)" - nargo -V + mv ./.github/scripts/merge-bench-reports.sh merge-bench-reports.sh + ./merge-bench-reports.sh memory_report + # Rename the memory report as to not clash with the compilation memory report file name + cp memory_report.json execution_memory_report.json - - name: Generate Compilation report - working-directory: ./test_programs + - name: Parse execution memory report + id: execution_mem_report + uses: noir-lang/noir-bench-report@6ba151d7795042c4ff51864fbeb13c0a6a79246c + with: + report: execution_memory_report.json + header: | + Execution Memory Report + memory_report: true + + - name: Add execution memory report to sticky comment + if: github.event_name == 'pull_request' || github.event_name == 'pull_request_target' + uses: marocchino/sticky-pull-request-comment@v2 + with: + header: execution_memory + message: ${{ steps.execution_mem_report.outputs.markdown }} + + - name: Convert to `benchmark-action` format run: | - ./compilation_report.sh - mv compilation_report.json ../compilation_report.json + jq ".memory_reports | map({name: .artifact_name, value: (.peak_memory | tonumber), unit: \"MB\"}) " ./execution_memory_report.json > memory_bench.json + + - name: Store benchmark result + continue-on-error: true + uses: benchmark-action/github-action-benchmark@4de1bed97a47495fc4c5404952da0499e31f5c29 + with: + name: "Execution Memory" + tool: "customSmallerIsBetter" + output-file-path: ./memory_bench.json + github-token: ${{ secrets.GITHUB_TOKEN }} + # We want this to only run on master to avoid garbage data from PRs being added. + auto-push: ${{ github.ref == 'refs/heads/master' }} + alert-threshold: "110%" + comment-on-alert: true + fail-on-alert: false + alert-comment-cc-users: "@TomAFrench" + max-items-in-chart: 50 + + + upload_execution_report: + name: Upload execution report + needs: [generate_compilation_and_execution_report, external_repo_compilation_and_execution_report] + # We want this job to run even if one variation of the matrix in `external_repo_compilation_and_execution_report` fails + if: always() + runs-on: ubuntu-22.04 + permissions: + pull-requests: write + # deployments permission to deploy GitHub pages website + deployments: write + # contents permission to update benchmark contents in gh-pages branch + contents: write - - name: Parse compilation report - id: compilation_report - uses: noir-lang/noir-bench-report@0d7464a8c39170523932d7846b6e6b458a294aea + steps: + - uses: actions/checkout@v4 + + - name: Download initial execution report + uses: actions/download-artifact@v4 with: - report: compilation_report.json + name: in_progress_execution_report + + - name: Download matrix execution reports + uses: actions/download-artifact@v4 + with: + pattern: execution_report_* + path: ./reports + + - name: Merge execution reports using jq + run: | + mv ./.github/scripts/merge-bench-reports.sh merge-bench-reports.sh + ./merge-bench-reports.sh execution_report + + - name: Parse execution report + id: execution_report + uses: noir-lang/noir-bench-report@6ba151d7795042c4ff51864fbeb13c0a6a79246c + with: + report: execution_report.json header: | - # Compilation Report - memory_report: false + Execution Report + execution_report: true - name: Add memory report to sticky comment if: github.event_name == 'pull_request' || github.event_name == 'pull_request_target' uses: marocchino/sticky-pull-request-comment@v2 with: - header: compilation - message: ${{ steps.compilation_report.outputs.markdown }} + header: execution_time + message: ${{ steps.execution_report.outputs.markdown }} + + - name: Convert to `benchmark-action` format + run: | + jq ".execution_reports | map({name: .artifact_name, value: (.time[:-1] | tonumber), unit: \"s\"}) " ./execution_report.json > time_bench.json + + - name: Store benchmark result + continue-on-error: true + uses: benchmark-action/github-action-benchmark@4de1bed97a47495fc4c5404952da0499e31f5c29 + with: + name: "Execution Time" + tool: "customSmallerIsBetter" + output-file-path: ./time_bench.json + github-token: ${{ secrets.GITHUB_TOKEN }} + # We want this to only run on master to avoid garbage data from PRs being added. + auto-push: ${{ github.ref == 'refs/heads/master' }} + alert-threshold: "110%" + comment-on-alert: true + fail-on-alert: false + alert-comment-cc-users: "@TomAFrench" + max-items-in-chart: 50 diff --git a/noir/noir-repo/.github/workflows/test-js-packages.yml b/noir/noir-repo/.github/workflows/test-js-packages.yml index d227b4eb00b..3fabf8ce39a 100644 --- a/noir/noir-repo/.github/workflows/test-js-packages.yml +++ b/noir/noir-repo/.github/workflows/test-js-packages.yml @@ -35,7 +35,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: @@ -68,7 +68,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: @@ -100,7 +100,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: @@ -135,7 +135,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 - uses: Swatinem/rust-cache@v2 with: @@ -543,8 +543,6 @@ jobs: external-repo-checks: needs: [build-nargo, critical-library-list] runs-on: ubuntu-22.04 - # Only run when 'run-external-checks' label is present - if: contains(github.event.pull_request.labels.*.name, 'run-external-checks') timeout-minutes: 30 strategy: fail-fast: false @@ -557,11 +555,17 @@ jobs: - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/parity-lib } - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/private-kernel-lib } - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/reset-kernel-lib } - - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-lib } - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/types } + # Use 1 test threads for rollup-lib because each test requires a lot of memory, and multiple ones in parallel exceed the maximum memory limit. + - project: { repo: AztecProtocol/aztec-packages, path: noir-projects/noir-protocol-circuits/crates/rollup-lib, nargo_args: "--test-threads 1" } name: Check external repo - ${{ matrix.project.repo }}/${{ matrix.project.path }} steps: + - name: Checkout + uses: actions/checkout@v4 + with: + path: noir-repo + - name: Checkout uses: actions/checkout@v4 with: @@ -592,9 +596,19 @@ jobs: - name: Run nargo test working-directory: ./test-repo/${{ matrix.project.path }} - run: nargo test -q --silence-warnings + run: | + output_file=${{ github.workspace }}/noir-repo/.github/critical_libraries_status/${{ matrix.project.repo }}/${{ matrix.project.path }}.actual.jsonl + nargo test --silence-warnings --skip-brillig-constraints-check --format json ${{ matrix.project.nargo_args }} | tee $output_file + if [ ! -s $output_file ]; then + # The file is empty so we delete it to signal that `nargo test` failed before it could run any tests + rm -f $output_file + fi env: NARGO_IGNORE_TEST_FAILURES_FROM_FOREIGN_CALLS: true + + - name: Compare test results + working-directory: ./noir-repo + run: .github/scripts/check_test_results.sh .github/critical_libraries_status/${{ matrix.project.repo }}/${{ matrix.project.path }}.failures.jsonl .github/critical_libraries_status/${{ matrix.project.repo }}/${{ matrix.project.path }}.actual.jsonl # This is a job which depends on all test jobs and reports the overall status. # This allows us to add/remove test jobs without having to update the required workflows. diff --git a/noir/noir-repo/.github/workflows/test-rust-workspace-msrv.yml b/noir/noir-repo/.github/workflows/test-rust-workspace-msrv.yml index 6fd71eb56a2..f4fbbf79d89 100644 --- a/noir/noir-repo/.github/workflows/test-rust-workspace-msrv.yml +++ b/noir/noir-repo/.github/workflows/test-rust-workspace-msrv.yml @@ -29,7 +29,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 with: targets: x86_64-unknown-linux-gnu @@ -72,7 +72,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 with: targets: x86_64-unknown-linux-gnu diff --git a/noir/noir-repo/.github/workflows/test-rust-workspace.yml b/noir/noir-repo/.github/workflows/test-rust-workspace.yml index 1514270ff56..5d8abbc3e55 100644 --- a/noir/noir-repo/.github/workflows/test-rust-workspace.yml +++ b/noir/noir-repo/.github/workflows/test-rust-workspace.yml @@ -23,7 +23,7 @@ jobs: uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 with: targets: x86_64-unknown-linux-gnu @@ -59,7 +59,7 @@ jobs: - uses: actions/checkout@v4 - name: Setup toolchain - uses: dtolnay/rust-toolchain@1.74.1 + uses: dtolnay/rust-toolchain@1.75.0 with: targets: x86_64-unknown-linux-gnu diff --git a/noir/noir-repo/.gitignore b/noir/noir-repo/.gitignore index 8442d688fbf..9bfb8a89331 100644 --- a/noir/noir-repo/.gitignore +++ b/noir/noir-repo/.gitignore @@ -36,6 +36,7 @@ gates_report.json gates_report_brillig.json gates_report_brillig_execution.json compilation_report.json +execution_report.json # Github Actions scratch space # This gives a location to download artifacts into the repository in CI without making git dirty. diff --git a/noir/noir-repo/.release-please-manifest.json b/noir/noir-repo/.release-please-manifest.json index b5c8da729e3..fd6cef7047a 100644 --- a/noir/noir-repo/.release-please-manifest.json +++ b/noir/noir-repo/.release-please-manifest.json @@ -1,3 +1,3 @@ { - ".": "1.0.0-beta.0" + ".": "1.0.0-beta.1" } diff --git a/noir/noir-repo/.vscode/settings.json b/noir/noir-repo/.vscode/settings.json index fb8ea527881..cd1c5f886df 100644 --- a/noir/noir-repo/.vscode/settings.json +++ b/noir/noir-repo/.vscode/settings.json @@ -7,5 +7,6 @@ }, "[javascript]": { "editor.defaultFormatter": "esbenp.prettier-vscode" - } + }, + "rust-analyzer.server.extraEnv": { "RUSTUP_TOOLCHAIN": "stable" } } diff --git a/noir/noir-repo/CHANGELOG.md b/noir/noir-repo/CHANGELOG.md index 19d36b885ed..da72291a3b0 100644 --- a/noir/noir-repo/CHANGELOG.md +++ b/noir/noir-repo/CHANGELOG.md @@ -1,5 +1,106 @@ # Changelog +## [1.0.0-beta.1](https://github.com/noir-lang/noir/compare/v1.0.0-beta.0...v1.0.0-beta.1) (2024-12-20) + + +### ⚠ BREAKING CHANGES + +* **stdlib:** Remove Schnorr ([#6749](https://github.com/noir-lang/noir/issues/6749)) +* remove SchnorrVerify opcode (https://github.com/AztecProtocol/aztec-packages/pull/9897) +* several format string fixes and improvements ([#6703](https://github.com/noir-lang/noir/issues/6703)) +* remove `ec` module from stdlib ([#6612](https://github.com/noir-lang/noir/issues/6612)) +* Disallow `#[export]` on associated methods ([#6626](https://github.com/noir-lang/noir/issues/6626)) + +### Features + +* `nargo test -q` (or `nargo test --format terse`) ([#6776](https://github.com/noir-lang/noir/issues/6776)) ([919149d](https://github.com/noir-lang/noir/commit/919149d3413be5232b33611094687fdb5fd86086)) +* `std::hint::black_box` function. ([#6529](https://github.com/noir-lang/noir/issues/6529)) ([237e8fa](https://github.com/noir-lang/noir/commit/237e8fa9ef8621e3da472e64c649849b4281004c)) +* Add `(x | 1)` optimization for booleans ([#6795](https://github.com/noir-lang/noir/issues/6795)) ([feec2a1](https://github.com/noir-lang/noir/commit/feec2a1a9851306d7cf682c796ab084dea10b2ec)) +* Add `BoundedVec::from_parts` and `BoundedVec::from_parts_unchecked` ([#6691](https://github.com/noir-lang/noir/issues/6691)) ([768aa7c](https://github.com/noir-lang/noir/commit/768aa7ce9ed809fa2c9d368f2b2f625d9689a63f)) +* Add `nargo test --format json` ([#6796](https://github.com/noir-lang/noir/issues/6796)) ([eb975ab](https://github.com/noir-lang/noir/commit/eb975ab28fb056cf92859377c02f2bb1a608eda3)) +* Add a warning when using unsafe blocks without safety comments ([#6860](https://github.com/noir-lang/noir/issues/6860)) ([5c00a79](https://github.com/noir-lang/noir/commit/5c00a79d2c93056d07330c350bf7b6efbf81d477)) +* Add memory report into the CI ([#6630](https://github.com/noir-lang/noir/issues/6630)) ([6acef6d](https://github.com/noir-lang/noir/commit/6acef6d795cd74dee4a21e82c5a912d58b40b06c)) +* Allow filtering which SSA passes are printed ([#6636](https://github.com/noir-lang/noir/issues/6636)) ([50f4aa7](https://github.com/noir-lang/noir/commit/50f4aa72e409e7205724f90046d394bb83584e9c)) +* Allow ignoring test failures from foreign calls ([#6660](https://github.com/noir-lang/noir/issues/6660)) ([e3a0914](https://github.com/noir-lang/noir/commit/e3a0914c6dede6f54f426ed7d790a0c98a7e0908)) +* Better error message when trying to invoke struct function field ([#6661](https://github.com/noir-lang/noir/issues/6661)) ([ea7c04a](https://github.com/noir-lang/noir/commit/ea7c04a8410ed8d2ce8e5a27e3c0784ba3195638)) +* **ci:** Initial compilation report on test_programs ([#6731](https://github.com/noir-lang/noir/issues/6731)) ([b3c04f0](https://github.com/noir-lang/noir/commit/b3c04f02d467c71ac9cb5eb6eca20b5bc0a2e47d)) +* **cli:** Run command on the package closest to the current directory ([#6752](https://github.com/noir-lang/noir/issues/6752)) ([f45b354](https://github.com/noir-lang/noir/commit/f45b3546bf82fd35eb446f7cbf00b739f287b92a)) +* **cli:** Verify `return` against ABI and `Prover.toml` ([#6765](https://github.com/noir-lang/noir/issues/6765)) ([5795a09](https://github.com/noir-lang/noir/commit/5795a099657a268b735a539298dfeefa445db3ff)) +* **comptime:** Implement blackbox functions in comptime interpreter ([#6551](https://github.com/noir-lang/noir/issues/6551)) ([10a9f81](https://github.com/noir-lang/noir/commit/10a9f8104e1ebb8fad044927ff130a0e2ce9131b)) +* Configurable external check failures ([#6810](https://github.com/noir-lang/noir/issues/6810)) ([73ccd45](https://github.com/noir-lang/noir/commit/73ccd45590222fc82642a6a9aa657c2915fc2c58)) +* Flatten nested if-else statements with equivalent conditions ([#6875](https://github.com/noir-lang/noir/issues/6875)) ([1a0a5f6](https://github.com/noir-lang/noir/commit/1a0a5f61231c3b65fa6397ec0769d3e6c5c238a3)) +* Improve parser recovery of constructor field with '::' instead of ':' ([#6701](https://github.com/noir-lang/noir/issues/6701)) ([c400543](https://github.com/noir-lang/noir/commit/c400543a9d9747798fd3b27b8508ac0a0668a09c)) +* Order attribute execution by their source ordering ([#6326](https://github.com/noir-lang/noir/issues/6326)) ([852155d](https://github.com/noir-lang/noir/commit/852155dc1c4a910bf9cd4e7af334f3856c1c4643)) +* **perf:** Track last loads per block in mem2reg and remove them if possible ([#6088](https://github.com/noir-lang/noir/issues/6088)) ([624ae6c](https://github.com/noir-lang/noir/commit/624ae6c6f0fdb077533abf93f8ff94814a7394f4)) +* Reduce memory consumption by storing array length as `u32` during SSA ([#6606](https://github.com/noir-lang/noir/issues/6606)) ([6196d05](https://github.com/noir-lang/noir/commit/6196d05bcb7ecd0b84fcc5ccc20d8dab99bc8052)) +* Replace `eval_global_as_array_length` with type/interpreter evaluation ([#6469](https://github.com/noir-lang/noir/issues/6469)) ([ddb4673](https://github.com/noir-lang/noir/commit/ddb46733fcf596b5c8508a208b2690df52aa16e3)) +* Replace quadratic removal of `rc` instructions ([#6705](https://github.com/noir-lang/noir/issues/6705)) ([7619da5](https://github.com/noir-lang/noir/commit/7619da59fc34cdd6e3b2581ad1668b5131ba4dde)) +* Replace quadratic removal of rc instructions (https://github.com/AztecProtocol/aztec-packages/pull/10416) ([66d3275](https://github.com/noir-lang/noir/commit/66d32751311378701b075ee7b2106d61e531ae4f)) +* Revert changes to `ValueMerger` and `Instruction::IfElse` ([#6673](https://github.com/noir-lang/noir/issues/6673)) ([f81244c](https://github.com/noir-lang/noir/commit/f81244c6bb29e8869f489d536141eebf6f68f00a)) +* Several `nargo test` improvements ([#6728](https://github.com/noir-lang/noir/issues/6728)) ([1b0dd41](https://github.com/noir-lang/noir/commit/1b0dd4149d9249f0ea4fb5e2228c688e0135618f)) +* Show printable byte arrays as byte strings in SSA ([#6709](https://github.com/noir-lang/noir/issues/6709)) ([fc11b63](https://github.com/noir-lang/noir/commit/fc11b631a2a1c0054b3b2a9e9fd2b7fa3a285076)) +* Simplify `jmpif`s by reversing branches if condition is negated ([#5891](https://github.com/noir-lang/noir/issues/5891)) ([ba7a568](https://github.com/noir-lang/noir/commit/ba7a568430c3477cc39e0ec147b11bdfc95093de)) +* **ssa:** Bring back tracking of RC instructions during DIE ([#6783](https://github.com/noir-lang/noir/issues/6783)) ([bc03152](https://github.com/noir-lang/noir/commit/bc03152366f242a6976f6e006e12520989e5e112)) +* **ssa:** Deduplicate intrinsics with predicates ([#6615](https://github.com/noir-lang/noir/issues/6615)) ([53f16c7](https://github.com/noir-lang/noir/commit/53f16c7fe75da04c54517b3d3199094b15195ce4)) +* **ssa:** Hoist MakeArray instructions during loop invariant code motion ([#6782](https://github.com/noir-lang/noir/issues/6782)) ([b88db67](https://github.com/noir-lang/noir/commit/b88db67a4fa92f861329105fb732a7b1309620fe)) +* **ssa:** Hoisting of array get using known induction variable maximum ([#6639](https://github.com/noir-lang/noir/issues/6639)) ([26d2351](https://github.com/noir-lang/noir/commit/26d235198f9a2fedbe438b3f7b39184554c5e1c1)) +* **ssa:** Implement missing brillig constraints SSA check ([#6658](https://github.com/noir-lang/noir/issues/6658)) ([c5a4caf](https://github.com/noir-lang/noir/commit/c5a4caf4e3971b8e9cb73681dcb21db0ba5550fc)) +* **ssa:** Option to set the maximum acceptable Brillig bytecode increase in unrolling ([#6641](https://github.com/noir-lang/noir/issues/6641)) ([4ff3081](https://github.com/noir-lang/noir/commit/4ff308128755c95b4d461bbcb7e3a49f16145585)) +* **ssa:** Simplify array get from set that writes to the same dynamic index ([#6684](https://github.com/noir-lang/noir/issues/6684)) ([304403f](https://github.com/noir-lang/noir/commit/304403f24e2a15b57bb054c4402a8d7f8d275668)) +* Sync from aztec-packages ([#6634](https://github.com/noir-lang/noir/issues/6634)) ([aa143a7](https://github.com/noir-lang/noir/commit/aa143a75d3460785ed88ea7ab3337c880c1153fd)) +* Sync from aztec-packages ([#6656](https://github.com/noir-lang/noir/issues/6656)) ([594aad2](https://github.com/noir-lang/noir/commit/594aad21f30614b1733a3ba2b8a2a5f5d7b7e119)) +* Sync from aztec-packages ([#6824](https://github.com/noir-lang/noir/issues/6824)) ([b3bca76](https://github.com/noir-lang/noir/commit/b3bca76620229e32c531417c6fa92e4a2c044fa0)) +* Sync from noir (https://github.com/AztecProtocol/aztec-packages/pull/10307) ([66d3275](https://github.com/noir-lang/noir/commit/66d32751311378701b075ee7b2106d61e531ae4f)) +* **test:** Check that `nargo::ops::transform_program` is idempotent ([#6694](https://github.com/noir-lang/noir/issues/6694)) ([9f3e0e6](https://github.com/noir-lang/noir/commit/9f3e0e68f8bc179fbceb606598a9749d3dc1c9e3)) +* **tooling:** Skip program transformation when loaded from cache ([#6689](https://github.com/noir-lang/noir/issues/6689)) ([7feb658](https://github.com/noir-lang/noir/commit/7feb6589d98246400497dfa492f39d67bae85977)) +* Warn on unnecessary unsafe blocks ([#6867](https://github.com/noir-lang/noir/issues/6867)) ([a97402a](https://github.com/noir-lang/noir/commit/a97402aa3b3a1a7d40f61242e02d3772175846f2)) + + +### Bug Fixes + +* Allow empty loop headers ([#6736](https://github.com/noir-lang/noir/issues/6736)) ([332ba79](https://github.com/noir-lang/noir/commit/332ba790287b152f14af8c88d0349323287e59bf)) +* Allow multiple `_` parameters, and disallow `_` as an expression you can read from ([#6657](https://github.com/noir-lang/noir/issues/6657)) ([d80a9d7](https://github.com/noir-lang/noir/commit/d80a9d71a8e4081a3aeacf34b2d5c2b0faf87484)) +* Always return an array of `u8`s when simplifying `Intrinsic::ToRadix` calls ([#6663](https://github.com/noir-lang/noir/issues/6663)) ([59c0c35](https://github.com/noir-lang/noir/commit/59c0c3562ad75d6c10fa7c6a2f74e3fbf68ec3e6)) +* Correct signed integer handling in `noirc_abi` ([#6638](https://github.com/noir-lang/noir/issues/6638)) ([ecaf63d](https://github.com/noir-lang/noir/commit/ecaf63da19a3a76e7c2940721d9044b1980a588f)) +* Correct types returned by constant EC operations simplified within SSA ([#6652](https://github.com/noir-lang/noir/issues/6652)) ([eec5970](https://github.com/noir-lang/noir/commit/eec5970658157704dac5c41e6d61b2aa652ce996)) +* Detect cycles in globals ([#6859](https://github.com/noir-lang/noir/issues/6859)) ([0d7642c](https://github.com/noir-lang/noir/commit/0d7642cb2071fbee148978a89a0922bfffe5be6a)) +* Disable failure persistance in nargo test fuzzing ([#6777](https://github.com/noir-lang/noir/issues/6777)) ([68ff7bd](https://github.com/noir-lang/noir/commit/68ff7bd85cb17f57afd481ec55318ec282a93aa6)) +* Disallow `#[export]` on associated methods ([#6626](https://github.com/noir-lang/noir/issues/6626)) ([7b56904](https://github.com/noir-lang/noir/commit/7b56904e56d95b88cefcbf3862e822fd3b1c8730)) +* Do not merge expressions that contain output witnesses ([#6757](https://github.com/noir-lang/noir/issues/6757)) ([f9abf72](https://github.com/noir-lang/noir/commit/f9abf724abd674ea4ccb342a770d237c70864ee1)) +* Do not warn on unused functions marked with #[export] ([#6625](https://github.com/noir-lang/noir/issues/6625)) ([30f8378](https://github.com/noir-lang/noir/commit/30f8378525b6f8ee305d356388c32761e12ee61c)) +* Don't deduplicate binary math of unsigned types ([#6848](https://github.com/noir-lang/noir/issues/6848)) ([ee0754b](https://github.com/noir-lang/noir/commit/ee0754b1c6b36961c180901db59dd593c183de77)) +* Don't remove necessary RC instructions in DIE pass ([#6585](https://github.com/noir-lang/noir/issues/6585)) ([440d94d](https://github.com/noir-lang/noir/commit/440d94d8149ede5f211437e9405f65b460cfcbf8)) +* Double alias in path ([#6855](https://github.com/noir-lang/noir/issues/6855)) ([82f595b](https://github.com/noir-lang/noir/commit/82f595b960a8fd54bcb5d2a76ea304af5782509b)) +* Git dependency trailing slash ([#6725](https://github.com/noir-lang/noir/issues/6725)) ([df71df7](https://github.com/noir-lang/noir/commit/df71df7e875b40529aa9404d45fd391a8857568a)) +* Implement `as_field` and `from_field` in the interpreter ([#6829](https://github.com/noir-lang/noir/issues/6829)) ([f037c36](https://github.com/noir-lang/noir/commit/f037c36f6bfcb8efb1950e444b50bc3eff28ffc4)) +* Improve type error when indexing a variable of unknown type ([#6744](https://github.com/noir-lang/noir/issues/6744)) ([909b22b](https://github.com/noir-lang/noir/commit/909b22bb20645761aabf8b40622aa79c6a96ed6a)) +* LSP auto-import text indent ([#6699](https://github.com/noir-lang/noir/issues/6699)) ([bbe7564](https://github.com/noir-lang/noir/commit/bbe756414612a37371812ace300e77c309791729)) +* LSP code action wasn't triggering on beginning or end of identifier ([#6616](https://github.com/noir-lang/noir/issues/6616)) ([1b910bc](https://github.com/noir-lang/noir/commit/1b910bc424d0435479b4104d2ed50557fdaf2bea)) +* **LSP:** Use generic self type to narrow down methods to complete ([#6617](https://github.com/noir-lang/noir/issues/6617)) ([454b77b](https://github.com/noir-lang/noir/commit/454b77b3ac4e99b1a272fdc5c36f8babb5781cec)) +* Map entry point indexes after all ssa passes ([#6740](https://github.com/noir-lang/noir/issues/6740)) ([1b6e26b](https://github.com/noir-lang/noir/commit/1b6e26b06ceb12abf92fdd49b70e6e2d10852d3b)) +* Minimal change to avoid reverting entire PR [#6685](https://github.com/noir-lang/noir/issues/6685) ([#6778](https://github.com/noir-lang/noir/issues/6778)) ([0925a33](https://github.com/noir-lang/noir/commit/0925a332dbaa561aad195c143079588158498dad)) +* Optimize array ref counts to copy arrays much less often ([#6685](https://github.com/noir-lang/noir/issues/6685)) ([24cc19e](https://github.com/noir-lang/noir/commit/24cc19ed9f29792c7b056124b2adf87fc6c18e42)) +* Optimizer to keep track of changing opcode locations ([#6781](https://github.com/noir-lang/noir/issues/6781)) ([13c41d2](https://github.com/noir-lang/noir/commit/13c41d21f81fb40cdf2a970be119a83d11da9e03)) +* Parser would hand on function type with colon in it ([#6764](https://github.com/noir-lang/noir/issues/6764)) ([9d7aadc](https://github.com/noir-lang/noir/commit/9d7aadc63c28b2c61d3524902c8b1038a46ba6f0)) +* Prevent hoisting binary instructions which can overflow ([#6672](https://github.com/noir-lang/noir/issues/6672)) ([b4750d8](https://github.com/noir-lang/noir/commit/b4750d8dc13245bad81cbf7bef7010e3794e040a)) +* Print ssa blocks without recursion ([#6715](https://github.com/noir-lang/noir/issues/6715)) ([5ccde81](https://github.com/noir-lang/noir/commit/5ccde8196400a9b99e5c3c4d9af0e3136e72b4cb)) +* Println("{{}}") was printing "{{}}" instead of "{}" ([#6745](https://github.com/noir-lang/noir/issues/6745)) ([36bca82](https://github.com/noir-lang/noir/commit/36bca82c9a25fdb5f1cd657dace11e46e59491c8)) +* Several format string fixes and improvements ([#6703](https://github.com/noir-lang/noir/issues/6703)) ([b70daf4](https://github.com/noir-lang/noir/commit/b70daf423890aa0f885ccf32531fa1583770c23c)) +* **ssa:** Don't deduplicate constraints in blocks that are not dominated ([#6627](https://github.com/noir-lang/noir/issues/6627)) ([b024581](https://github.com/noir-lang/noir/commit/b0245811bfd84e0bf3559aa1e2f37ec52d08691e)) +* **ssa:** Remove RC tracker in DIE ([#6700](https://github.com/noir-lang/noir/issues/6700)) ([f2607fd](https://github.com/noir-lang/noir/commit/f2607fd0eafac0018c157e101c7ebb1fe4223f73)) +* **ssa:** Track all local allocations during flattening ([#6619](https://github.com/noir-lang/noir/issues/6619)) ([6491175](https://github.com/noir-lang/noir/commit/649117570b95b26776150e337c458d478eb48c2e)) +* Typo in u128 docs ([#6711](https://github.com/noir-lang/noir/issues/6711)) ([37a4996](https://github.com/noir-lang/noir/commit/37a4996a7e33b9afe78dcb494f6c3e796d852607)) +* Use correct type for attribute arguments ([#6640](https://github.com/noir-lang/noir/issues/6640)) ([de3e77a](https://github.com/noir-lang/noir/commit/de3e77a4acaab4bd2edc9a9e5226e54f468fd620)) +* Use extension in docs link so it also works on GitHub ([#6787](https://github.com/noir-lang/noir/issues/6787)) ([655a3d3](https://github.com/noir-lang/noir/commit/655a3d3fdcf4f4cdb4381e3ff47d2f822c4b2276)) +* Used signed division for signed modulo ([#6635](https://github.com/noir-lang/noir/issues/6635)) ([dace078](https://github.com/noir-lang/noir/commit/dace07849aa28795abb30b3f9d979ffc6b6487e6)) + + +### Miscellaneous Chores + +* Remove `ec` module from stdlib ([#6612](https://github.com/noir-lang/noir/issues/6612)) ([1e965bc](https://github.com/noir-lang/noir/commit/1e965bc8b9c4222c7b2ad7502df415781308de7f)) +* Remove SchnorrVerify opcode (https://github.com/AztecProtocol/aztec-packages/pull/9897) ([66d3275](https://github.com/noir-lang/noir/commit/66d32751311378701b075ee7b2106d61e531ae4f)) +* **stdlib:** Remove Schnorr ([#6749](https://github.com/noir-lang/noir/issues/6749)) ([57ebee0](https://github.com/noir-lang/noir/commit/57ebee03e00b69d1cc8f541b083b001cd517ec35)) + ## [1.0.0-beta.0](https://github.com/noir-lang/noir/compare/v0.39.0...v1.0.0-beta.0) (2024-11-22) diff --git a/noir/noir-repo/CRITICAL_NOIR_LIBRARIES b/noir/noir-repo/CRITICAL_NOIR_LIBRARIES index c753b76a4fc..7637d9ac6df 100644 --- a/noir/noir-repo/CRITICAL_NOIR_LIBRARIES +++ b/noir/noir-repo/CRITICAL_NOIR_LIBRARIES @@ -1,3 +1,4 @@ +https://github.com/noir-lang/noir_check_shuffle https://github.com/noir-lang/ec https://github.com/noir-lang/eddsa https://github.com/noir-lang/mimc diff --git a/noir/noir-repo/Cargo.lock b/noir/noir-repo/Cargo.lock index 4907de7ae62..e82d47d690a 100644 --- a/noir/noir-repo/Cargo.lock +++ b/noir/noir-repo/Cargo.lock @@ -4,7 +4,7 @@ version = 3 [[package]] name = "acir" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acir_field", "base64 0.21.7", @@ -26,12 +26,12 @@ dependencies = [ [[package]] name = "acir_field" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "ark-bls12-381", - "ark-bn254", - "ark-ff", - "cfg-if 1.0.0", + "ark-bn254 0.5.0", + "ark-ff 0.5.0", + "cfg-if", "hex", "num-bigint", "proptest", @@ -40,12 +40,15 @@ dependencies = [ [[package]] name = "acvm" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acir", "acvm_blackbox_solver", "ark-bls12-381", - "ark-bn254", + "ark-bn254 0.4.0", + "ark-bn254 0.5.0", + "ark-ff 0.4.2", + "ark-ff 0.5.0", "bn254_blackbox_solver", "brillig_vm", "fxhash", @@ -60,7 +63,7 @@ dependencies = [ [[package]] name = "acvm_blackbox_solver" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acir", "blake2", @@ -69,6 +72,7 @@ dependencies = [ "keccak", "libaes", "num-bigint", + "num-prime", "p256", "proptest", "sha2", @@ -88,23 +92,23 @@ dependencies = [ "nargo", "paste", "proptest", - "rand 0.8.5", + "rand", "thiserror", - "toml 0.7.8", + "toml", "tracing-appender", "tracing-subscriber", ] [[package]] name = "acvm_js" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "bn254_blackbox_solver", "build-data", "console_error_panic_hook", "const-str", - "getrandom 0.2.15", + "getrandom", "gloo-utils", "js-sys", "pkg-config", @@ -143,8 +147,8 @@ version = "0.8.11" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e89da841a80418a9b391ebaea17f5c112ffaaa96f621d2c285b5174da76b9011" dependencies = [ - "cfg-if 1.0.0", - "getrandom 0.2.15", + "cfg-if", + "getrandom", "once_cell", "version_check", "zerocopy", @@ -168,6 +172,12 @@ dependencies = [ "equator", ] +[[package]] +name = "allocator-api2" +version = "0.2.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "683d7910e743518b0e34f1186f92494becacb047c7b6bf616c96772180fef923" + [[package]] name = "android-tzdata" version = "0.1.1" @@ -246,14 +256,14 @@ checksum = "4c95c10ba0b00a02636238b814946408b1322d5ac4760326e6fb8ec956d85775" [[package]] name = "ark-bls12-381" -version = "0.4.0" +version = "0.5.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c775f0d12169cba7aae4caeb547bb6a50781c7449a8aa53793827c9ec4abf488" +checksum = "3df4dcc01ff89867cd86b0da835f23c3f02738353aaee7dde7495af71363b8d5" dependencies = [ - "ark-ec", - "ark-ff", - "ark-serialize", - "ark-std", + "ark-ec 0.5.0", + "ark-ff 0.5.0", + "ark-serialize 0.5.0", + "ark-std 0.5.0", ] [[package]] @@ -262,9 +272,20 @@ version = "0.4.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a22f4561524cd949590d78d7d4c5df8f592430d221f7f3c9497bbafd8972120f" dependencies = [ - "ark-ec", - "ark-ff", - "ark-std", + "ark-ec 0.4.2", + "ark-ff 0.4.2", + "ark-std 0.4.0", +] + +[[package]] +name = "ark-bn254" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d69eab57e8d2663efa5c63135b2af4f396d66424f88954c21104125ab6b3e6bc" +dependencies = [ + "ark-ec 0.5.0", + "ark-ff 0.5.0", + "ark-std 0.5.0", ] [[package]] @@ -273,13 +294,34 @@ version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "defd9a439d56ac24968cca0571f598a61bc8c55f71d50a89cda591cb750670ba" dependencies = [ - "ark-ff", - "ark-poly", - "ark-serialize", - "ark-std", + "ark-ff 0.4.2", + "ark-poly 0.4.2", + "ark-serialize 0.4.2", + "ark-std 0.4.0", "derivative", "hashbrown 0.13.2", - "itertools", + "itertools 0.10.5", + "num-traits", + "zeroize", +] + +[[package]] +name = "ark-ec" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "43d68f2d516162846c1238e755a7c4d131b892b70cc70c471a8e3ca3ed818fce" +dependencies = [ + "ahash", + "ark-ff 0.5.0", + "ark-poly 0.5.0", + "ark-serialize 0.5.0", + "ark-std 0.5.0", + "educe", + "fnv", + "hashbrown 0.15.1", + "itertools 0.13.0", + "num-bigint", + "num-integer", "num-traits", "zeroize", ] @@ -290,13 +332,13 @@ version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ec847af850f44ad29048935519032c33da8aa03340876d351dfab5660d2966ba" dependencies = [ - "ark-ff-asm", - "ark-ff-macros", - "ark-serialize", - "ark-std", + "ark-ff-asm 0.4.2", + "ark-ff-macros 0.4.2", + "ark-serialize 0.4.2", + "ark-std 0.4.0", "derivative", "digest", - "itertools", + "itertools 0.10.5", "num-bigint", "num-traits", "paste", @@ -304,6 +346,26 @@ dependencies = [ "zeroize", ] +[[package]] +name = "ark-ff" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a177aba0ed1e0fbb62aa9f6d0502e9b46dad8c2eab04c14258a1212d2557ea70" +dependencies = [ + "ark-ff-asm 0.5.0", + "ark-ff-macros 0.5.0", + "ark-serialize 0.5.0", + "ark-std 0.5.0", + "arrayvec", + "digest", + "educe", + "itertools 0.13.0", + "num-bigint", + "num-traits", + "paste", + "zeroize", +] + [[package]] name = "ark-ff-asm" version = "0.4.2" @@ -314,6 +376,16 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "ark-ff-asm" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "62945a2f7e6de02a31fe400aa489f0e0f5b2502e69f95f853adb82a96c7a6b60" +dependencies = [ + "quote", + "syn 2.0.87", +] + [[package]] name = "ark-ff-macros" version = "0.4.2" @@ -327,27 +399,80 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "ark-ff-macros" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09be120733ee33f7693ceaa202ca41accd5653b779563608f1234f78ae07c4b3" +dependencies = [ + "num-bigint", + "num-traits", + "proc-macro2", + "quote", + "syn 2.0.87", +] + +[[package]] +name = "ark-grumpkin" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ef677b59f5aff4123207c4dceb1c0ec8fdde2d4af7886f48be42ad864bfa0352" +dependencies = [ + "ark-bn254 0.5.0", + "ark-ec 0.5.0", + "ark-ff 0.5.0", + "ark-std 0.5.0", +] + [[package]] name = "ark-poly" version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d320bfc44ee185d899ccbadfa8bc31aab923ce1558716e1997a1e74057fe86bf" dependencies = [ - "ark-ff", - "ark-serialize", - "ark-std", + "ark-ff 0.4.2", + "ark-serialize 0.4.2", + "ark-std 0.4.0", "derivative", "hashbrown 0.13.2", ] +[[package]] +name = "ark-poly" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "579305839da207f02b89cd1679e50e67b4331e2f9294a57693e5051b7703fe27" +dependencies = [ + "ahash", + "ark-ff 0.5.0", + "ark-serialize 0.5.0", + "ark-std 0.5.0", + "educe", + "fnv", + "hashbrown 0.15.1", +] + [[package]] name = "ark-serialize" version = "0.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "adb7b85a02b83d2f22f89bd5cac66c9c89474240cb6207cb1efc16d098e822a5" dependencies = [ - "ark-serialize-derive", - "ark-std", + "ark-serialize-derive 0.4.2", + "ark-std 0.4.0", + "digest", + "num-bigint", +] + +[[package]] +name = "ark-serialize" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3f4d068aaf107ebcd7dfb52bc748f8030e0fc930ac8e360146ca54c1203088f7" +dependencies = [ + "ark-serialize-derive 0.5.0", + "ark-std 0.5.0", + "arrayvec", "digest", "num-bigint", ] @@ -363,6 +488,17 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "ark-serialize-derive" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "213888f660fddcca0d257e88e54ac05bca01885f258ccdf695bafd77031bb69d" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", +] + [[package]] name = "ark-std" version = "0.4.0" @@ -370,7 +506,17 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "94893f1e0c6eeab764ade8dc4c0db24caf4fe7cbbaafc0eba0a9030f447b5185" dependencies = [ "num-traits", - "rand 0.8.5", + "rand", +] + +[[package]] +name = "ark-std" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "246a225cc6131e9ee4f24619af0f19d67761fff15d7ccc22e42b80846e69449a" +dependencies = [ + "num-traits", + "rand", ] [[package]] @@ -422,7 +568,7 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "138985dd8aefbefeaa66b01b7f5b2b6b4c333fcef1cc5f32c63a2aabe37d6de3" dependencies = [ - "futures 0.3.31", + "futures", "lsp-types 0.94.1", "pin-project-lite", "rustix", @@ -436,6 +582,23 @@ dependencies = [ "waitpid-any", ] +[[package]] +name = "async-trait" +version = "0.1.83" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "721cae7de5c34fbb2acd27e21e6d2cf7b886dce0c27388d46c4e6c47ea4318dd" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", +] + +[[package]] +name = "atomic-waker" +version = "1.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0" + [[package]] name = "autocfg" version = "1.4.0" @@ -450,7 +613,7 @@ checksum = "26b05800d2e817c8b3b4b54abd461726265fa9789ae34330622f2db9ee696f9d" dependencies = [ "addr2line", "cc", - "cfg-if 1.0.0", + "cfg-if", "libc", "miniz_oxide 0.7.4", "object", @@ -463,12 +626,6 @@ version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "349a06037c7bf932dd7e7d1f653678b2038b9ad46a74102f1fc7bd7872678cce" -[[package]] -name = "base64" -version = "0.13.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9e1b586273c5702936fe7b7d6896644d8be71e6314cfe09d3167c95f712589e8" - [[package]] name = "base64" version = "0.21.7" @@ -487,6 +644,12 @@ version = "1.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8c3c1a368f70d6cf7302d78f8f7093da241fb8e8807c05cc9e51a125895a6d5b" +[[package]] +name = "binary-merge" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "597bb81c80a54b6a4381b23faba8d7774b144c94cbd1d6fe3f1329bd776554ab" + [[package]] name = "bincode" version = "1.3.3" @@ -498,18 +661,18 @@ dependencies = [ [[package]] name = "bit-set" -version = "0.5.3" +version = "0.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "0700ddab506f33b20a03b13996eccd309a48e5ff77d0d95926aa0210fb4e95f1" +checksum = "08807e080ed7f9d5433fa9b275196cfc35414f66a0c79d864dc51a0d825231a3" dependencies = [ "bit-vec", ] [[package]] name = "bit-vec" -version = "0.6.3" +version = "0.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "349f9b6a179ed607305526ca489b34ad0a41aed5f7980fa90eb03160b69598fb" +checksum = "5e764a1d40d510daf35e07be9eb06e75770908c27d411ee6c92109c9840eaaf7" [[package]] name = "bitflags" @@ -573,7 +736,7 @@ dependencies = [ "arrayref", "arrayvec", "cc", - "cfg-if 1.0.0", + "cfg-if", "constant_time_eq", ] @@ -595,31 +758,31 @@ dependencies = [ "ff 0.12.1", "group 0.12.1", "pairing", - "rand_core 0.6.4", + "rand_core", "subtle", ] [[package]] name = "bn254_blackbox_solver" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acir", "acvm_blackbox_solver", - "ark-bn254", - "ark-ec", - "ark-ff", - "ark-std", + "ark-bn254 0.5.0", + "ark-ec 0.5.0", + "ark-ff 0.5.0", + "ark-grumpkin", + "ark-std 0.5.0", "criterion", "hex", "lazy_static", - "noir_grumpkin", "num-bigint", "pprof", ] [[package]] name = "brillig" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acir_field", "serde", @@ -627,7 +790,7 @@ dependencies = [ [[package]] name = "brillig_vm" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acir", "acvm_blackbox_solver", @@ -682,38 +845,6 @@ version = "1.8.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9ac0150caa2ae65ca5bd83f25c7de183dea78d4d366469f148435e2acfbad0da" -[[package]] -name = "camino" -version = "1.1.9" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8b96ec4966b5813e2c0507c1f86115c8c5abaadc3980879c3424042a02fd1ad3" -dependencies = [ - "serde", -] - -[[package]] -name = "cargo-platform" -version = "0.1.8" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "24b1f0365a6c6bb4020cd05806fd0d33c44d38046b8bd7f0e40814b9763cabfc" -dependencies = [ - "serde", -] - -[[package]] -name = "cargo_metadata" -version = "0.15.4" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eee4243f1f26fc7a42710e7439c149e2b10b05472f88090acce52632f231a73a" -dependencies = [ - "camino", - "cargo-platform", - "semver", - "serde", - "serde_json", - "thiserror", -] - [[package]] name = "cast" version = "0.3.0" @@ -730,10 +861,10 @@ dependencies = [ ] [[package]] -name = "cfg-if" -version = "0.1.10" +name = "cesu8" +version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4785bdd1c96b2a846b2bd7cc02e86b6b3dbf14e7e53446c4f54c92a361040822" +checksum = "6d43a04d8753f35258c91f8ec639f792891f748a1edbd759cf1dcea3382ad83c" [[package]] name = "cfg-if" @@ -869,7 +1000,7 @@ checksum = "fc4159b76af02757139baf42c0c971c6dc155330999fbfd8eddb29b97fb2db68" dependencies = [ "codespan-reporting", "lsp-types 0.88.0", - "url 2.5.4", + "url", ] [[package]] @@ -916,6 +1047,16 @@ version = "1.0.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5b63caa9aa9397e2d9480a9b13673856c78d8ac123288526c37d7839f2a86990" +[[package]] +name = "combine" +version = "4.6.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ba5a308b75df32fe02788e748662718f03fde005016435c444eea572398219fd" +dependencies = [ + "bytes", + "memchr", +] + [[package]] name = "comma" version = "1.0.0" @@ -940,7 +1081,7 @@ version = "0.1.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a06aeb73f470f66dcdbf7223caeebb85984942f22f1adb2a088cf9668146bbbc" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "wasm-bindgen", ] @@ -982,12 +1123,6 @@ version = "0.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7c74b8349d32d297c9134b8c88677813a227df8f779daa29bfc29c183fe3dca6" -[[package]] -name = "convert_case" -version = "0.4.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e" - [[package]] name = "convert_case" version = "0.6.0" @@ -997,6 +1132,16 @@ dependencies = [ "unicode-segmentation", ] +[[package]] +name = "core-foundation" +version = "0.9.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91e195e091a93c46f7102ec7818a2aa394e1e1771c3ab4825963fa03e45afb8f" +dependencies = [ + "core-foundation-sys", + "libc", +] + [[package]] name = "core-foundation-sys" version = "0.8.7" @@ -1009,7 +1154,7 @@ version = "0.4.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "96e58d342ad113c2b878f16d5d034c03be492ae460cdbc02b7f0f2284d310c7d" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", ] [[package]] @@ -1027,7 +1172,7 @@ version = "1.4.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "a97769d94ddab943e4510d138150169a2758b5ef3eb191a9ee688de3e23ef7b3" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", ] [[package]] @@ -1042,7 +1187,7 @@ dependencies = [ "clap", "criterion-plot", "is-terminal", - "itertools", + "itertools 0.10.5", "num-traits", "once_cell", "oorandom", @@ -1063,7 +1208,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6b50826342786a51a89e2da3a28f1c32b06e387201bc2d19791f622c673706b1" dependencies = [ "cast", - "itertools", + "itertools 0.10.5", ] [[package]] @@ -1113,7 +1258,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ef2b4b23cddf68b89b8f8069890e8c270d54e2d5fe1b143820234805e4cb17ef" dependencies = [ "generic-array", - "rand_core 0.6.4", + "rand_core", "subtle", "zeroize", ] @@ -1201,12 +1346,12 @@ version = "6.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5041cc499144891f3790297212f32a74fb938e5136a14943f338ef9e0ae276cf" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "crossbeam-utils", "hashbrown 0.14.5", "lock_api", "once_cell", - "parking_lot_core 0.9.10", + "parking_lot_core", ] [[package]] @@ -1249,19 +1394,6 @@ dependencies = [ "syn 1.0.109", ] -[[package]] -name = "derive_more" -version = "0.99.18" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5f33878137e4dafd7fa914ad4e259e18a4e8e532b9617a2d0150262bf53abfce" -dependencies = [ - "convert_case 0.4.0", - "proc-macro2", - "quote", - "rustc_version", - "syn 2.0.87", -] - [[package]] name = "difflib" version = "0.4.0" @@ -1294,7 +1426,7 @@ version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b98cf8ebf19c3d1b223e151f99a4f9f0690dca41414773390fc824184ac833e1" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "dirs-sys-next", ] @@ -1364,6 +1496,18 @@ dependencies = [ "signature", ] +[[package]] +name = "educe" +version = "0.6.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1d7bc049e1bd8cdeb31b68bbd586a9464ecf9f3944af3958a7a9d0f8b9799417" +dependencies = [ + "enum-ordinalize", + "proc-macro2", + "quote", + "syn 2.0.87", +] + [[package]] name = "either" version = "1.13.0" @@ -1384,7 +1528,7 @@ dependencies = [ "generic-array", "group 0.12.1", "pkcs8", - "rand_core 0.6.4", + "rand_core", "sec1", "subtle", "zeroize", @@ -1408,6 +1552,26 @@ version = "0.1.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c34f04666d835ff5d62e058c3995147c06f42fe86ff053337632bca83e42702d" +[[package]] +name = "enum-ordinalize" +version = "4.3.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fea0dcfa4e54eeb516fe454635a95753ddd39acda650ce703031c6973e315dd5" +dependencies = [ + "enum-ordinalize-derive", +] + +[[package]] +name = "enum-ordinalize-derive" +version = "4.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0d28318a75d4aead5c4db25382e8ef717932d0346600cacae6357eb5941bc5ff" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", +] + [[package]] name = "env_filter" version = "0.1.2" @@ -1495,7 +1659,7 @@ version = "3.0.13" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ef033ed5e9bad94e55838ca0ca906db0e043f517adda0c8b79c7a8c66c93c1b5" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "rustix", "windows-sys 0.48.0", ] @@ -1507,7 +1671,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d013fc25338cc558c5c2cfbad646908fb23591e2404481826742b651c9af7160" dependencies = [ "bitvec", - "rand_core 0.6.4", + "rand_core", "subtle", ] @@ -1518,7 +1682,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ded41244b729663b1e574f1b4fb731469f69f79c17667b5d776b16cda0479449" dependencies = [ "bitvec", - "rand_core 0.6.4", + "rand_core", "subtle", ] @@ -1540,23 +1704,13 @@ dependencies = [ "windows-sys 0.52.0", ] -[[package]] -name = "file-lock" -version = "2.1.11" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "040b48f80a749da50292d0f47a1e2d5bf1d772f52836c07f64bfccc62ba6e664" -dependencies = [ - "cc", - "libc", -] - [[package]] name = "filetime" version = "0.2.25" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "35c0522e981e68cbfa8c3f978441a5f34b30b96e146b33cd3359176b50fe8586" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "libc", "libredox 0.1.3", "windows-sys 0.59.0", @@ -1601,7 +1755,7 @@ dependencies = [ [[package]] name = "fm" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "codespan-reporting", "iter-extended", @@ -1614,13 +1768,29 @@ version = "1.0.7" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" +[[package]] +name = "foldhash" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a0d2fde1f7b3d48b8395d5f2de76c18a528bd6a9cdde438df747bfcba3e05d6f" + [[package]] name = "form_urlencoded" version = "1.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e13624c2627564efccf4934284bdd98cbaa14e79b0b5a141218e507b3a823456" dependencies = [ - "percent-encoding 2.3.1", + "percent-encoding", +] + +[[package]] +name = "fs2" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9564fc758e15025b46aa6643b1b77d047d1a56a1aea6e01002ac0c7026876213" +dependencies = [ + "libc", + "winapi", ] [[package]] @@ -1638,12 +1808,6 @@ version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e6d5a32815ae3f33302d95fdcb2ce17862f8c65363dcfd29360480ba1001fc9c" -[[package]] -name = "futures" -version = "0.1.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3a471a38ef8ed83cd6e40aa59c1ffe17db6855c18e3604d9c4ed8c08ebc28678" - [[package]] name = "futures" version = "0.3.31" @@ -1652,7 +1816,6 @@ checksum = "65bc07b1a8bc7c85c5f2e110c476c7389b4554ba72af57d8445ea63a576b0876" dependencies = [ "futures-channel", "futures-core", - "futures-executor", "futures-io", "futures-sink", "futures-task", @@ -1675,18 +1838,6 @@ version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "05f29059c0c2090612e8d742178b0580d2dc940c837851ad723096f87af6663e" -[[package]] -name = "futures-executor" -version = "0.3.31" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1e28d1d997f585e54aebc3f97d39e72338912123a67330d723fdbb564d646c9f" -dependencies = [ - "futures-core", - "futures-task", - "futures-util", - "num_cpus", -] - [[package]] name = "futures-io" version = "0.3.31" @@ -1722,7 +1873,6 @@ version = "0.3.31" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9fa08315bb612088cc391249efdc3bc77536f16c91f6cf495e6fbe85b20a4a81" dependencies = [ - "futures 0.1.31", "futures-channel", "futures-core", "futures-io", @@ -1754,27 +1904,16 @@ dependencies = [ "version_check", ] -[[package]] -name = "getrandom" -version = "0.1.16" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8fc3cb4d91f53b50155bdcfd23f6a4c39ae1969c2ae85982b135750cccaf5fce" -dependencies = [ - "cfg-if 1.0.0", - "libc", - "wasi 0.9.0+wasi-snapshot-preview1", -] - [[package]] name = "getrandom" version = "0.2.15" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "c4567c8db10ae91089c99af84c68c38da3ec2f087c3f82960bcdbf3656b6f4d7" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "js-sys", "libc", - "wasi 0.11.0+wasi-snapshot-preview1", + "wasi", "wasm-bindgen", ] @@ -1835,7 +1974,7 @@ checksum = "5dfbfb3a6cfbd390d5c9564ab283a0349b9b9fcd46a706c1eb10e0db70bfbac7" dependencies = [ "ff 0.12.1", "memuse", - "rand_core 0.6.4", + "rand_core", "subtle", ] @@ -1846,17 +1985,36 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f0f9ef7462f7c099f518d754361858f86d8a07af53ba9af0fe635bbccb151a63" dependencies = [ "ff 0.13.0", - "rand_core 0.6.4", + "rand_core", "subtle", ] +[[package]] +name = "h2" +version = "0.4.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ccae279728d634d083c00f6099cb58f01cc99c145b84b8be2f6c74618d79922e" +dependencies = [ + "atomic-waker", + "bytes", + "fnv", + "futures-core", + "futures-sink", + "http", + "indexmap 2.6.0", + "slab", + "tokio", + "tokio-util", + "tracing", +] + [[package]] name = "half" version = "2.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6dd08c532ae367adf81c312a4580bc67f1d0fe8bc9c460520283f4c0ff277888" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "crunchy", ] @@ -1879,7 +2037,7 @@ dependencies = [ "ff 0.12.1", "group 0.12.1", "pasta_curves 0.4.1", - "rand_core 0.6.4", + "rand_core", "rayon", ] @@ -1909,6 +2067,11 @@ name = "hashbrown" version = "0.15.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3a9bfc1af68b1726ea47d3d5109de126281def866b33970e10fbab11b5dafab3" +dependencies = [ + "allocator-api2", + "equivalent", + "foldhash", +] [[package]] name = "heck" @@ -1960,9 +2123,9 @@ dependencies = [ [[package]] name = "http" -version = "0.2.12" +version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "601cbb57e577e2f5ef5be8e7b83f0f63994f25aa94d673e54a92d5c516d101f1" +checksum = "f16ca2af56261c99fba8bac40a10251ce8188205a4c448fbb745a2e4daa76fea" dependencies = [ "bytes", "fnv", @@ -1971,12 +2134,24 @@ dependencies = [ [[package]] name = "http-body" -version = "0.4.6" +version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7ceab25649e9960c0311ea418d17bee82c0dcec1bd053b5f9a66e265a693bed2" +checksum = "1efedce1fb8e6913f23e0c92de8e62cd5b772a67e7b3946df930a62566c93184" dependencies = [ "bytes", "http", +] + +[[package]] +name = "http-body-util" +version = "0.1.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "793429d76616a256bcb62c2a2ec2bed781c8307e797e2598c50010f2bee2544f" +dependencies = [ + "bytes", + "futures-util", + "http", + "http-body", "pin-project-lite", ] @@ -1994,25 +2169,60 @@ checksum = "df3b46402a9d5adb4c86a0cf463f42e19994e3ee891101b1841f30a545cb49a9" [[package]] name = "hyper" -version = "0.14.31" +version = "1.5.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8c08302e8fa335b151b788c775ff56e7a03ae64ff85c548ee820fecb70356e85" +checksum = "256fb8d4bd6413123cc9d91832d78325c48ff41677595be797d90f42969beae0" dependencies = [ "bytes", "futures-channel", - "futures-core", "futures-util", + "h2", "http", "http-body", "httparse", "httpdate", "itoa", "pin-project-lite", + "smallvec", + "tokio", + "want", +] + +[[package]] +name = "hyper-rustls" +version = "0.27.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "08afdbb5c31130e3034af566421053ab03787c640246a446327f550d11bcb333" +dependencies = [ + "futures-util", + "http", + "hyper", + "hyper-util", + "log", + "rustls", + "rustls-pki-types", + "tokio", + "tokio-rustls", + "tower-service", +] + +[[package]] +name = "hyper-util" +version = "0.1.10" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "df2dcfbe0677734ab2f3ffa7fa7bfd4706bfdc1ef393f2ee30184aed67e631b4" +dependencies = [ + "bytes", + "futures-channel", + "futures-util", + "http", + "http-body", + "hyper", + "pin-project-lite", "socket2", "tokio", "tower-service", "tracing", - "want", ] [[package]] @@ -2168,17 +2378,6 @@ version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" -[[package]] -name = "idna" -version = "0.1.5" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "38f09e0f0b1fb55fdee1f17470ad800da77af5186a1a76c026b679358b7e844e" -dependencies = [ - "matches", - "unicode-bidi", - "unicode-normalization", -] - [[package]] name = "idna" version = "1.0.3" @@ -2223,7 +2422,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d0acd33ff0285af998aaf9b57342af478078f53492322fafc47450e09397e0e9" dependencies = [ "bitmaps", - "rand_core 0.6.4", + "rand_core", "rand_xoshiro", "serde", "sized-chunks", @@ -2327,12 +2526,12 @@ dependencies = [ ] [[package]] -name = "instant" -version = "0.1.13" +name = "inplace-vec-builder" +version = "0.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e0242819d153cba4b4b05a5a8f2a7e9bbf97b6055b2a002b395c96b5ff3c0222" +checksum = "cf64c2edc8226891a71f127587a2861b132d2b942310843814d5001d99a1d307" dependencies = [ - "cfg-if 1.0.0", + "smallvec", ] [[package]] @@ -2354,7 +2553,7 @@ checksum = "7943c866cc5cd64cbc25b2e01621d07fa8eb2a1a23160ee81ce38704e97b8ecf" [[package]] name = "iter-extended" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" [[package]] name = "itertools" @@ -2365,6 +2564,15 @@ dependencies = [ "either", ] +[[package]] +name = "itertools" +version = "0.13.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "413ee7dfc52ee1a4949ceeb7dbc8a33f2d6c088194d9f922fb8318faf1f01186" +dependencies = [ + "either", +] + [[package]] name = "itoa" version = "1.0.11" @@ -2372,126 +2580,147 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "49f1f14873335454500d59611f1cf4a4b0f786f9ac11f4312a78e4cf2566695b" [[package]] -name = "js-sys" -version = "0.3.63" +name = "jni" +version = "0.19.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2f37a4a5928311ac501dee68b3c7613a1037d0edb30c8e5427bd832d55d1b790" +checksum = "c6df18c2e3db7e453d3c6ac5b3e9d5182664d28788126d39b91f2d1e22b017ec" dependencies = [ - "wasm-bindgen", + "cesu8", + "combine", + "jni-sys", + "log", + "thiserror", + "walkdir", ] [[package]] -name = "jsonrpc" -version = "0.16.0" +name = "jni-sys" +version = "0.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "34efde8d2422fb79ed56db1d3aea8fa5b583351d15a26770cdee2f88813dd702" +checksum = "8eaf4bc02d17cbdd7ff4c7438cafcdf7fb9a4613313ad11b4f8fefe7d3fa0130" + +[[package]] +name = "js-sys" +version = "0.3.63" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2f37a4a5928311ac501dee68b3c7613a1037d0edb30c8e5427bd832d55d1b790" dependencies = [ - "base64 0.13.1", - "minreq", - "serde", - "serde_json", + "wasm-bindgen", ] [[package]] -name = "jsonrpc-client-transports" -version = "18.0.0" +name = "jsonrpsee" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d2b99d4207e2a04fb4581746903c2bb7eb376f88de9c699d0f3e10feeac0cd3a" +checksum = "c5c71d8c1a731cc4227c2f698d377e7848ca12c8a48866fc5e6951c43a4db843" dependencies = [ - "derive_more", - "futures 0.3.31", - "jsonrpc-core", - "jsonrpc-pubsub", - "log", - "serde", - "serde_json", - "url 1.7.2", + "jsonrpsee-core", + "jsonrpsee-http-client", + "jsonrpsee-proc-macros", + "jsonrpsee-server", + "jsonrpsee-types", + "tokio", + "tracing", ] [[package]] -name = "jsonrpc-core" -version = "18.0.0" +name = "jsonrpsee-core" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "14f7f76aef2d054868398427f6c54943cf3d1caa9a7ec7d0c38d69df97a965eb" +checksum = "f2882f6f8acb9fdaec7cefc4fd607119a9bd709831df7d7672a1d3b644628280" dependencies = [ - "futures 0.3.31", - "futures-executor", + "async-trait", + "bytes", "futures-util", - "log", + "http", + "http-body", + "http-body-util", + "jsonrpsee-types", + "parking_lot", + "rand", + "rustc-hash 2.1.0", "serde", - "serde_derive", "serde_json", + "thiserror", + "tokio", + "tracing", ] [[package]] -name = "jsonrpc-core-client" -version = "18.0.0" +name = "jsonrpsee-http-client" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b51da17abecbdab3e3d4f26b01c5ec075e88d3abe3ab3b05dc9aa69392764ec0" +checksum = "b3638bc4617f96675973253b3a45006933bde93c2fd8a6170b33c777cc389e5b" dependencies = [ - "futures 0.3.31", - "jsonrpc-client-transports", + "async-trait", + "base64 0.22.1", + "http-body", + "hyper", + "hyper-rustls", + "hyper-util", + "jsonrpsee-core", + "jsonrpsee-types", + "rustls", + "rustls-platform-verifier", + "serde", + "serde_json", + "thiserror", + "tokio", + "tower", + "tracing", + "url", ] [[package]] -name = "jsonrpc-derive" -version = "18.0.0" +name = "jsonrpsee-proc-macros" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5b939a78fa820cdfcb7ee7484466746a7377760970f6f9c6fe19f9edcc8a38d2" +checksum = "c06c01ae0007548e73412c08e2285ffe5d723195bf268bce67b1b77c3bb2a14d" dependencies = [ + "heck 0.5.0", "proc-macro-crate", "proc-macro2", "quote", - "syn 1.0.109", + "syn 2.0.87", ] [[package]] -name = "jsonrpc-http-server" -version = "18.0.0" +name = "jsonrpsee-server" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e1dea6e07251d9ce6a552abfb5d7ad6bc290a4596c8dcc3d795fae2bbdc1f3ff" +checksum = "82ad8ddc14be1d4290cd68046e7d1d37acd408efed6d3ca08aefcc3ad6da069c" dependencies = [ - "futures 0.3.31", + "futures-util", + "http", + "http-body", + "http-body-util", "hyper", - "jsonrpc-core", - "jsonrpc-server-utils", - "log", - "net2", - "parking_lot 0.11.2", - "unicase", -] - -[[package]] -name = "jsonrpc-pubsub" -version = "18.0.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "240f87695e6c6f62fb37f05c02c04953cf68d6408b8c1c89de85c7a0125b1011" -dependencies = [ - "futures 0.3.31", - "jsonrpc-core", - "lazy_static", - "log", - "parking_lot 0.11.2", - "rand 0.7.3", + "hyper-util", + "jsonrpsee-core", + "jsonrpsee-types", + "pin-project", + "route-recognizer", "serde", + "serde_json", + "soketto", + "thiserror", + "tokio", + "tokio-stream", + "tokio-util", + "tower", + "tracing", ] [[package]] -name = "jsonrpc-server-utils" -version = "18.0.0" +name = "jsonrpsee-types" +version = "0.24.7" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fa4fdea130485b572c39a460d50888beb00afb3e35de23ccd7fad8ff19f0e0d4" +checksum = "a178c60086f24cc35bb82f57c651d0d25d99c4742b4d335de04e97fa1f08a8a1" dependencies = [ - "bytes", - "futures 0.3.31", - "globset", - "jsonrpc-core", - "lazy_static", - "log", - "tokio", - "tokio-stream", - "tokio-util 0.6.10", - "unicase", + "http", + "serde", + "serde_json", + "thiserror", ] [[package]] @@ -2504,7 +2733,7 @@ dependencies = [ "bls12_381", "ff 0.12.1", "group 0.12.1", - "rand_core 0.6.4", + "rand_core", "subtle", ] @@ -2514,7 +2743,7 @@ version = "0.11.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "72c1e0b51e7ec0a97369623508396067a486bd0cbed95a2659a4b863d28cfc8b" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "ecdsa", "elliptic-curve", "sha2", @@ -2570,12 +2799,6 @@ version = "0.2.162" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "18d287de67fe55fd7e1581fe933d965a5a9477b38e949cfa9f8574ef01506398" -[[package]] -name = "libm" -version = "0.2.11" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8355be11b20d696c8f18f6cc018c4e372165b1fa8126cef092399c9951984ffa" - [[package]] name = "libredox" version = "0.0.2" @@ -2604,8 +2827,8 @@ version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3c9a85a9752c549ceb7578064b4ed891179d20acd85f27318573b64d2d7ee7ee" dependencies = [ - "ark-bn254", - "ark-ff", + "ark-bn254 0.4.0", + "ark-ff 0.4.2", "num-bigint", "thiserror", ] @@ -2647,6 +2870,15 @@ dependencies = [ "fid-rs", ] +[[package]] +name = "lru" +version = "0.12.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "234cf4f4a04dc1f57e24b96cc0cd600cf2af460d4161ac5ecdd0af8e1f3b2a38" +dependencies = [ + "hashbrown 0.15.1", +] + [[package]] name = "lsp-types" version = "0.88.0" @@ -2657,7 +2889,7 @@ dependencies = [ "serde", "serde_json", "serde_repr", - "url 2.5.4", + "url", ] [[package]] @@ -2670,7 +2902,7 @@ dependencies = [ "serde", "serde_json", "serde_repr", - "url 2.5.4", + "url", ] [[package]] @@ -2682,12 +2914,6 @@ dependencies = [ "regex-automata 0.1.10", ] -[[package]] -name = "matches" -version = "0.1.10" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2532096657941c2fea9c289d370a250971c689d4f143798ff67113ec042024a5" - [[package]] name = "memchr" version = "2.7.4" @@ -2714,9 +2940,9 @@ dependencies = [ [[package]] name = "memuse" -version = "0.2.1" +version = "0.2.2" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2145869435ace5ea6ea3d35f59be559317ec9a0d04e1812d5f185a87b6d36f1a" +checksum = "3d97bbf43eb4f088f8ca469930cde17fa036207c9a5e02ccc5107c4e8b17c964" [[package]] name = "miniz_oxide" @@ -2736,17 +2962,6 @@ dependencies = [ "adler2", ] -[[package]] -name = "minreq" -version = "2.12.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "763d142cdff44aaadd9268bebddb156ef6c65a0e13486bb81673cf2d8739f9b0" -dependencies = [ - "log", - "serde", - "serde_json", -] - [[package]] name = "mio" version = "0.8.11" @@ -2755,7 +2970,7 @@ checksum = "a4a650543ca06a924e8b371db273b2756685faae30f8487da1b56505a8f78b0c" dependencies = [ "libc", "log", - "wasi 0.11.0+wasi-snapshot-preview1", + "wasi", "windows-sys 0.48.0", ] @@ -2767,22 +2982,18 @@ checksum = "80e04d1dcff3aae0704555fe5fee3bcfaf3d1fdf8a7e521d5b9d2b42acb52cec" dependencies = [ "hermit-abi 0.3.9", "libc", - "wasi 0.11.0+wasi-snapshot-preview1", + "wasi", "windows-sys 0.52.0", ] [[package]] name = "nargo" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "fm", "iter-extended", - "jsonrpc", - "jsonrpc-core", - "jsonrpc-core-client", - "jsonrpc-derive", - "jsonrpc-http-server", + "jsonrpsee", "noir_fuzzer", "noirc_abi", "noirc_driver", @@ -2790,20 +3001,24 @@ dependencies = [ "noirc_frontend", "noirc_printable_type", "proptest", - "rand 0.8.5", + "rand", "rayon", "serde", + "serde_json", "thiserror", + "tokio", "tracing", "walkdir", ] [[package]] name = "nargo_cli" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", - "ark-bn254", + "ark-bn254 0.4.0", + "ark-bn254 0.5.0", + "ark-ff 0.4.2", "assert_cmd", "assert_fs", "async-lsp", @@ -2817,8 +3032,8 @@ dependencies = [ "criterion", "dap", "dirs", - "file-lock", "fm", + "fs2", "fxhash", "iai", "iter-extended", @@ -2850,31 +3065,31 @@ dependencies = [ "tempfile", "termcolor", "termion", - "test-binary", "test-case", "thiserror", "tokio", - "tokio-util 0.7.12", - "toml 0.7.8", + "tokio-util", + "toml", "tower", + "tracing", "tracing-appender", "tracing-subscriber", ] [[package]] name = "nargo_fmt" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "noirc_frontend", "serde", "similar-asserts", "thiserror", - "toml 0.7.8", + "toml", ] [[package]] name = "nargo_toml" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "dirs", "fm", @@ -2886,19 +3101,8 @@ dependencies = [ "tempfile", "test-case", "thiserror", - "toml 0.7.8", - "url 2.5.4", -] - -[[package]] -name = "net2" -version = "0.2.39" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b13b648036a2339d06de780866fbdfda0dde886de7b3af2ddeba8b14f4ee34ac" -dependencies = [ - "cfg-if 0.1.10", - "libc", - "winapi", + "toml", + "url", ] [[package]] @@ -2918,7 +3122,7 @@ checksum = "8f3790c00a0150112de0f4cd161e3d7fc4b2d8a5542ffc35f099a2562aecb35c" dependencies = [ "bitflags 1.3.2", "cc", - "cfg-if 1.0.0", + "cfg-if", "libc", "memoffset", ] @@ -2931,7 +3135,7 @@ checksum = "f346ff70e7dbfd675fe90590b92d59ef2de15a8779ae305ebcbfd3f0caf59be4" dependencies = [ "autocfg", "bitflags 1.3.2", - "cfg-if 1.0.0", + "cfg-if", "libc", "memoffset", "pin-utils", @@ -2944,13 +3148,13 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "598beaf3cc6fdd9a5dfb1630c2800c7acd31df7aaf0f565796fba2b53ca1af1b" dependencies = [ "bitflags 1.3.2", - "cfg-if 1.0.0", + "cfg-if", "libc", ] [[package]] name = "noir_debugger" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "assert_cmd", @@ -2974,35 +3178,23 @@ dependencies = [ [[package]] name = "noir_fuzzer" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "noirc_abi", "noirc_artifacts", "proptest", - "rand 0.8.5", -] - -[[package]] -name = "noir_grumpkin" -version = "0.1.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4e7d49a4b14b13c0dc730b05780b385828ab88f4148daaad7db080ecdce07350" -dependencies = [ - "ark-bn254", - "ark-ec", - "ark-ff", - "ark-std", + "rand", ] [[package]] name = "noir_lsp" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "async-lsp", "codespan-lsp", - "convert_case 0.6.0", + "convert_case", "fm", "fxhash", "lsp-types 0.94.1", @@ -3024,7 +3216,7 @@ dependencies = [ [[package]] name = "noir_profiler" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acir", "bn254_blackbox_solver", @@ -3050,13 +3242,13 @@ dependencies = [ [[package]] name = "noir_wasm" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "build-data", "console_error_panic_hook", "fm", - "getrandom 0.2.15", + "getrandom", "gloo-utils", "js-sys", "nargo", @@ -3074,7 +3266,7 @@ dependencies = [ [[package]] name = "noirc_abi" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "iter-extended", @@ -3082,23 +3274,23 @@ dependencies = [ "num-bigint", "num-traits", "proptest", - "proptest-derive 0.4.0", + "proptest-derive", "serde", "serde_json", "strum", "strum_macros", "thiserror", - "toml 0.7.8", + "toml", ] [[package]] name = "noirc_abi_wasm" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "build-data", "console_error_panic_hook", - "getrandom 0.2.15", + "getrandom", "gloo-utils", "iter-extended", "js-sys", @@ -3110,11 +3302,11 @@ dependencies = [ [[package]] name = "noirc_arena" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" [[package]] name = "noirc_artifacts" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "codespan-reporting", @@ -3129,7 +3321,7 @@ dependencies = [ [[package]] name = "noirc_driver" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "build-data", @@ -3148,7 +3340,7 @@ dependencies = [ [[package]] name = "noirc_errors" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "base64 0.21.7", @@ -3165,11 +3357,11 @@ dependencies = [ [[package]] name = "noirc_evaluator" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "bn254_blackbox_solver", - "cfg-if 1.0.0", + "cfg-if", "chrono", "fxhash", "im", @@ -3184,21 +3376,24 @@ dependencies = [ "serde_json", "serde_with", "similar-asserts", + "smallvec", "test-case", "thiserror", "tracing", "tracing-test", + "vec-collections", ] [[package]] name = "noirc_frontend" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", "base64 0.21.7", "bn254_blackbox_solver", - "cfg-if 1.0.0", + "cfg-if", "fm", + "fxhash", "im", "iter-extended", "noirc_arena", @@ -3208,9 +3403,9 @@ dependencies = [ "num-traits", "petgraph", "proptest", - "proptest-derive 0.5.0", + "proptest-derive", "rangemap", - "rustc-hash", + "rustc-hash 1.1.0", "serde", "serde_json", "small-ord-set", @@ -3223,14 +3418,10 @@ dependencies = [ [[package]] name = "noirc_printable_type" -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" dependencies = [ "acvm", - "iter-extended", - "jsonrpc", "serde", - "serde_json", - "thiserror", ] [[package]] @@ -3268,7 +3459,7 @@ dependencies = [ "file-id", "log", "notify", - "parking_lot 0.12.3", + "parking_lot", "walkdir", ] @@ -3290,6 +3481,7 @@ checksum = "a5e44f723f1133c9deac646763579fdb3ac745e418f2a7af9cd0c431da1f20b9" dependencies = [ "num-integer", "num-traits", + "rand", ] [[package]] @@ -3318,23 +3510,39 @@ dependencies = [ ] [[package]] -name = "num-traits" -version = "0.2.19" +name = "num-modular" +version = "0.5.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841" +checksum = "64a5fe11d4135c3bcdf3a95b18b194afa9608a5f6ff034f5d857bc9a27fb0119" dependencies = [ - "autocfg", - "libm", + "num-bigint", + "num-integer", + "num-traits", ] [[package]] -name = "num_cpus" -version = "1.16.0" +name = "num-prime" +version = "0.4.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4161fcb6d602d4d2081af7c3a45852d875a03dd337a6bfdd6e06407b61342a43" +checksum = "e238432a7881ec7164503ccc516c014bf009be7984cde1ba56837862543bdec3" dependencies = [ - "hermit-abi 0.3.9", - "libc", + "bitvec", + "either", + "lru", + "num-bigint", + "num-integer", + "num-modular", + "num-traits", + "rand", +] + +[[package]] +name = "num-traits" +version = "0.2.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "071dfc062690e90b734c0b2273ce72ad0ffa95f0c74596bc250dcfd960262841" +dependencies = [ + "autocfg", ] [[package]] @@ -3364,6 +3572,12 @@ version = "11.1.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b410bbe7e14ab526a0e86877eb47c6996a2bd7746f027ba551028c925390e4e9" +[[package]] +name = "openssl-probe" +version = "0.1.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" + [[package]] name = "overload" version = "0.1.1" @@ -3396,17 +3610,6 @@ dependencies = [ "group 0.12.1", ] -[[package]] -name = "parking_lot" -version = "0.11.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7d17b78036a60663b797adeaee46f5c9dfebb86948d1255007a1d6be0271ff99" -dependencies = [ - "instant", - "lock_api", - "parking_lot_core 0.8.6", -] - [[package]] name = "parking_lot" version = "0.12.3" @@ -3414,21 +3617,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "f1bf18183cf54e8d6059647fc3063646a1801cf30896933ec2311622cc4b9a27" dependencies = [ "lock_api", - "parking_lot_core 0.9.10", -] - -[[package]] -name = "parking_lot_core" -version = "0.8.6" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "60a2cfe6f0ad2bfc16aefa463b497d5c7a5ecd44a23efa72aa342d90177356dc" -dependencies = [ - "cfg-if 1.0.0", - "instant", - "libc", - "redox_syscall 0.2.16", - "smallvec", - "winapi", + "parking_lot_core", ] [[package]] @@ -3437,7 +3626,7 @@ version = "0.9.10" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1e401f977ab385c9e4e3ab30627d6f26d00e2c73eef317493c4ec6d468726cf8" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "libc", "redox_syscall 0.5.7", "smallvec", @@ -3454,7 +3643,7 @@ dependencies = [ "ff 0.12.1", "group 0.12.1", "lazy_static", - "rand 0.8.5", + "rand", "static_assertions", "subtle", ] @@ -3469,7 +3658,7 @@ dependencies = [ "ff 0.13.0", "group 0.13.0", "lazy_static", - "rand 0.8.5", + "rand", "static_assertions", "subtle", ] @@ -3480,12 +3669,6 @@ version = "1.0.15" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a" -[[package]] -name = "percent-encoding" -version = "1.0.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "31010dd2e1ac33d5b46a5b413495239882813e0369f8ed8a5e266f173602f831" - [[package]] name = "percent-encoding" version = "2.3.1" @@ -3520,7 +3703,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5d5285893bb5eb82e6aaf5d59ee909a06a16737a8970984dd7746ba9283498d6" dependencies = [ "phf_shared", - "rand 0.8.5", + "rand", ] [[package]] @@ -3546,6 +3729,26 @@ dependencies = [ "siphasher", ] +[[package]] +name = "pin-project" +version = "1.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "be57f64e946e500c8ee36ef6331845d40a93055567ec57e8fae13efd33759b95" +dependencies = [ + "pin-project-internal", +] + +[[package]] +name = "pin-project-internal" +version = "1.1.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c0f5fad0874fc7abcd4d750e76917eaebbecaa2c20bde22e1dbeeba8beb758c" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", +] + [[package]] name = "pin-project-lite" version = "0.2.15" @@ -3616,7 +3819,7 @@ checksum = "ebbe2f8898beba44815fdc9e5a4ae9c929e21c5dc29b0c774a15555f7f58d6d0" dependencies = [ "aligned-vec", "backtrace", - "cfg-if 1.0.0", + "cfg-if", "criterion", "findshlibs", "inferno", @@ -3624,7 +3827,7 @@ dependencies = [ "log", "nix 0.26.4", "once_cell", - "parking_lot 0.12.3", + "parking_lot", "smallvec", "symbolic-demangle", "tempfile", @@ -3648,7 +3851,7 @@ checksum = "59230a63c37f3e18569bdb90e4a89cbf5bf8b06fea0b84e65ea10cc4df47addd" dependencies = [ "difflib", "float-cmp", - "itertools", + "itertools 0.10.5", "normalize-line-endings", "predicates-core", "regex", @@ -3697,11 +3900,11 @@ dependencies = [ [[package]] name = "proc-macro-crate" -version = "0.1.5" +version = "3.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1d6ea3c4595b96363c13943497db34af4460fb474a95c43f4446ad341b8c9785" +checksum = "8ecf48c7ca261d60b74ab1a7b20da18bede46776b2e55535cb958eb595c5fa7b" dependencies = [ - "toml 0.5.11", + "toml_edit 0.22.22", ] [[package]] @@ -3721,17 +3924,17 @@ dependencies = [ [[package]] name = "proptest" -version = "1.5.0" +version = "1.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b4c2511913b88df1637da85cc8d96ec8e43a3f8bb8ccb71ee1ac240d6f3df58d" +checksum = "14cae93065090804185d3b75f0bf93b8eeda30c7a9b4a33d3bdb3988d6229e50" dependencies = [ "bit-set", "bit-vec", "bitflags 2.6.0", "lazy_static", "num-traits", - "rand 0.8.5", - "rand_chacha 0.3.1", + "rand", + "rand_chacha", "rand_xorshift", "regex-syntax 0.8.5", "rusty-fork", @@ -3739,17 +3942,6 @@ dependencies = [ "unarray", ] -[[package]] -name = "proptest-derive" -version = "0.4.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9cf16337405ca084e9c78985114633b6827711d22b9e6ef6c6c0d665eb3f0b6e" -dependencies = [ - "proc-macro2", - "quote", - "syn 1.0.109", -] - [[package]] name = "proptest-derive" version = "0.5.0" @@ -3801,19 +3993,6 @@ dependencies = [ "nibble_vec", ] -[[package]] -name = "rand" -version = "0.7.3" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6a6b1679d49b24bbfe0c803429aa1874472f50d9b363131f0e89fc356b544d03" -dependencies = [ - "getrandom 0.1.16", - "libc", - "rand_chacha 0.2.2", - "rand_core 0.5.1", - "rand_hc", -] - [[package]] name = "rand" version = "0.8.5" @@ -3821,18 +4000,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "34af8d1a0e25924bc5b7c43c079c942339d8f0a8b57c39049bef581b46327404" dependencies = [ "libc", - "rand_chacha 0.3.1", - "rand_core 0.6.4", -] - -[[package]] -name = "rand_chacha" -version = "0.2.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f4c8ed856279c9737206bf725bf36935d8666ead7aa69b52be55af369d193402" -dependencies = [ - "ppv-lite86", - "rand_core 0.5.1", + "rand_chacha", + "rand_core", ] [[package]] @@ -3842,16 +4011,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e6c10a63a0fa32252be49d21e7709d4d4baf8d231c2dbce1eaa8141b9b127d88" dependencies = [ "ppv-lite86", - "rand_core 0.6.4", -] - -[[package]] -name = "rand_core" -version = "0.5.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "90bde5296fc891b0cef12a6d03ddccc162ce7b2aff54160af9338f8d40df6d19" -dependencies = [ - "getrandom 0.1.16", + "rand_core", ] [[package]] @@ -3860,16 +4020,7 @@ version = "0.6.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ec0be4795e2f6a28069bec0b5ff3e2ac9bafc99e6a9a7dc3547996c5c816922c" dependencies = [ - "getrandom 0.2.15", -] - -[[package]] -name = "rand_hc" -version = "0.2.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ca3129af7b92a17112d59ad498c6f81eaf463253766b90396d39ea7a39d6613c" -dependencies = [ - "rand_core 0.5.1", + "getrandom", ] [[package]] @@ -3878,7 +4029,7 @@ version = "0.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "d25bf25ec5ae4a3f1b92f929810509a2f53d7dca2f50b794ff57e3face536c8f" dependencies = [ - "rand_core 0.6.4", + "rand_core", ] [[package]] @@ -3887,7 +4038,7 @@ version = "0.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6f97cdb2a36ed4183de61b2f824cc45c9f1037f28afe0a322e9fff4c108b5aaa" dependencies = [ - "rand_core 0.6.4", + "rand_core", ] [[package]] @@ -3916,15 +4067,6 @@ dependencies = [ "crossbeam-utils", ] -[[package]] -name = "redox_syscall" -version = "0.2.16" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fb5a58c1855b4b6819d59012155603f0b22ad30cad752600aadfcb695265519a" -dependencies = [ - "bitflags 1.3.2", -] - [[package]] name = "redox_syscall" version = "0.4.1" @@ -3955,7 +4097,7 @@ version = "0.4.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ba009ff324d1fc1b900bd1fdb31564febe58a8ccc8a6fdbb93b543d33b13ca43" dependencies = [ - "getrandom 0.2.15", + "getrandom", "libredox 0.1.3", "thiserror", ] @@ -4037,6 +4179,27 @@ dependencies = [ "bytemuck", ] +[[package]] +name = "ring" +version = "0.17.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c17fa4cb658e3583423e915b9f3acc01cceaee1860e33d59ebae66adc3a2dc0d" +dependencies = [ + "cc", + "cfg-if", + "getrandom", + "libc", + "spin", + "untrusted", + "windows-sys 0.52.0", +] + +[[package]] +name = "route-recognizer" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "afab94fb28594581f62d981211a9a4d53cc8130bbcbbb89a0440d9b8e81a7746" + [[package]] name = "rust-embed" version = "6.8.1" @@ -4083,6 +4246,12 @@ version = "1.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "08d43f7aa6b08d49f382cde6a7982047c3426db949b1424bc4b7ec9ae12c6ce2" +[[package]] +name = "rustc-hash" +version = "2.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c7fb8039b3032c191086b10f11f319a6e99e1e82889c5cc6046f515c9db1d497" + [[package]] name = "rustc_version" version = "0.4.1" @@ -4105,6 +4274,87 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "rustls" +version = "0.23.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5065c3f250cbd332cd894be57c40fa52387247659b14a2d6041d121547903b1b" +dependencies = [ + "log", + "once_cell", + "ring", + "rustls-pki-types", + "rustls-webpki", + "subtle", + "zeroize", +] + +[[package]] +name = "rustls-native-certs" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e5bfb394eeed242e909609f56089eecfe5fda225042e8b171791b9c95f5931e5" +dependencies = [ + "openssl-probe", + "rustls-pemfile", + "rustls-pki-types", + "schannel", + "security-framework", +] + +[[package]] +name = "rustls-pemfile" +version = "2.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dce314e5fee3f39953d46bb63bb8a46d40c2f8fb7cc5a3b6cab2bde9721d6e50" +dependencies = [ + "rustls-pki-types", +] + +[[package]] +name = "rustls-pki-types" +version = "1.10.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d2bf47e6ff922db3825eb750c4e2ff784c6ff8fb9e13046ef6a1d1c5401b0b37" + +[[package]] +name = "rustls-platform-verifier" +version = "0.3.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "afbb878bdfdf63a336a5e63561b1835e7a8c91524f51621db870169eac84b490" +dependencies = [ + "core-foundation", + "core-foundation-sys", + "jni", + "log", + "once_cell", + "rustls", + "rustls-native-certs", + "rustls-platform-verifier-android", + "rustls-webpki", + "security-framework", + "security-framework-sys", + "webpki-roots", + "winapi", +] + +[[package]] +name = "rustls-platform-verifier-android" +version = "0.1.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f87165f0995f63a9fbeea62b64d10b4d9d8e78ec6d7d51fb2125fda7bb36788f" + +[[package]] +name = "rustls-webpki" +version = "0.102.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "64ca1bc8749bd4cf37b5ce386cc146580777b4e8572c7b97baf22c83f444bee9" +dependencies = [ + "ring", + "rustls-pki-types", + "untrusted", +] + [[package]] name = "rustversion" version = "1.0.18" @@ -4130,7 +4380,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "db7826789c0e25614b03e5a54a0717a86f9ff6e6e5247f92b369472869320039" dependencies = [ "bitflags 1.3.2", - "cfg-if 1.0.0", + "cfg-if", "clipboard-win", "dirs-next", "fd-lock", @@ -4225,6 +4475,15 @@ dependencies = [ "winapi-util", ] +[[package]] +name = "schannel" +version = "0.1.27" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1f29ebaa345f945cec9fbbc532eb307f0fdad8161f281b6369539c8d84876b3d" +dependencies = [ + "windows-sys 0.59.0", +] + [[package]] name = "scoped-tls" version = "1.0.1" @@ -4251,14 +4510,35 @@ dependencies = [ "zeroize", ] +[[package]] +name = "security-framework" +version = "2.11.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "897b2245f0b511c87893af39b033e5ca9cce68824c4d7e7630b5a1d339658d02" +dependencies = [ + "bitflags 2.6.0", + "core-foundation", + "core-foundation-sys", + "libc", + "num-bigint", + "security-framework-sys", +] + +[[package]] +name = "security-framework-sys" +version = "2.12.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fa39c7303dc58b5543c94d22c1766b0d31f2ee58306363ea622b10bbc075eaa2" +dependencies = [ + "core-foundation-sys", + "libc", +] + [[package]] name = "semver" version = "1.0.23" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "61697e0a1c7e512e84a621326239844a24d8207b4669b41bc18b32ea5cbf988b" -dependencies = [ - "serde", -] [[package]] name = "serde" @@ -4376,13 +4656,24 @@ dependencies = [ "syn 2.0.87", ] +[[package]] +name = "sha1" +version = "0.10.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e3bf829a2d51ab4a5ddf1352d8470c140cadc8301b2ae1789db023f01cedd6ba" +dependencies = [ + "cfg-if", + "cpufeatures", + "digest", +] + [[package]] name = "sha2" version = "0.10.8" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "793db75ad2bcafc3ffa7c68b215fee268f537982cd901d132f89c6343f3a3dc8" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "cpufeatures", "digest", ] @@ -4425,7 +4716,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "74233d3b3b2f6d4b006dc19dee745e73e2a6bfb6f93607cd3b02bd5b00797d7c" dependencies = [ "digest", - "rand_core 0.6.4", + "rand_core", ] [[package]] @@ -4487,6 +4778,9 @@ name = "smallvec" version = "1.13.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3c5e1a9a646d36c3599cd173a41282daf47c44583ad367b8e6837255952e5c67" +dependencies = [ + "serde", +] [[package]] name = "smawk" @@ -4513,6 +4807,28 @@ dependencies = [ "windows-sys 0.52.0", ] +[[package]] +name = "soketto" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2e859df029d160cb88608f5d7df7fb4753fd20fdfb4de5644f3d8b8440841721" +dependencies = [ + "base64 0.22.1", + "bytes", + "futures", + "http", + "httparse", + "log", + "rand", + "sha1", +] + +[[package]] +name = "sorted-iter" +version = "0.1.11" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bceb57dc07c92cdae60f5b27b3fa92ecaaa42fe36c55e22dbfb0b44893e0b1f7" + [[package]] name = "spin" version = "0.9.8" @@ -4652,7 +4968,7 @@ version = "3.14.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "28cce251fcbc87fac86a866eeb0d6c2d536fc16d06f184bb61aeae11aa4cee0c" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "fastrand", "once_cell", "rustix", @@ -4697,19 +5013,6 @@ version = "0.4.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3369f5ac52d5eb6ab48c6b4ffdc8efbcad6b89c765749064ba298f2c68a16a76" -[[package]] -name = "test-binary" -version = "3.0.2" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6c7cb854285c40b61c0fade358bf63a2bb1226688a1ea11432ea65349209e6e3" -dependencies = [ - "camino", - "cargo_metadata", - "once_cell", - "paste", - "thiserror", -] - [[package]] name = "test-case" version = "3.3.1" @@ -4725,7 +5028,7 @@ version = "3.3.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "adcb7fd841cd518e279be3d5a3eb0636409487998a4aff22f3de87b81e88384f" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "proc-macro2", "quote", "syn 2.0.87", @@ -4790,7 +5093,7 @@ version = "1.1.8" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "8b9ef9bad013ada3808854ceac7b46812a6465ba368859a37e2100283d2d719c" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "once_cell", ] @@ -4845,26 +5148,11 @@ dependencies = [ "serde_json", ] -[[package]] -name = "tinyvec" -version = "1.8.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "445e881f4f6d382d5f27c034e25eb92edd7c784ceab92a0937db7f2e9471b938" -dependencies = [ - "tinyvec_macros", -] - -[[package]] -name = "tinyvec_macros" -version = "0.1.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" - [[package]] name = "tokio" -version = "1.41.1" +version = "1.42.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "22cfb5bee7a6a52939ca9224d6ac897bb669134078daa8735560897f69de4d33" +checksum = "5cec9b21b0450273377fc97bd4c33a8acffc8c996c987a7c5b319a0083707551" dependencies = [ "backtrace", "bytes", @@ -4888,28 +5176,25 @@ dependencies = [ ] [[package]] -name = "tokio-stream" -version = "0.1.16" +name = "tokio-rustls" +version = "0.26.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "4f4e6ce100d0eb49a2734f8c0812bcd324cf357d21810932c5df6b96ef2b86f1" +checksum = "5f6d0975eaace0cf0fcadee4e4aaa5da15b5c079146f2cffb67c113be122bf37" dependencies = [ - "futures-core", - "pin-project-lite", + "rustls", "tokio", ] [[package]] -name = "tokio-util" -version = "0.6.10" +name = "tokio-stream" +version = "0.1.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "36943ee01a6d67977dd3f84a5a1d2efeb4ada3a1ae771cadfaa535d9d9fc6507" +checksum = "eca58d7bba4a75707817a2c44174253f9236b2d5fbd055602e9d5c07c139a047" dependencies = [ - "bytes", "futures-core", - "futures-sink", - "log", "pin-project-lite", "tokio", + "tokio-util", ] [[package]] @@ -4926,15 +5211,6 @@ dependencies = [ "tokio", ] -[[package]] -name = "toml" -version = "0.5.11" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f4f7f0dd8d50a853a531c426359045b1998f04219d88799810762cd4ad314234" -dependencies = [ - "serde", -] - [[package]] name = "toml" version = "0.7.8" @@ -4944,7 +5220,7 @@ dependencies = [ "serde", "serde_spanned", "toml_datetime", - "toml_edit", + "toml_edit 0.19.15", ] [[package]] @@ -4966,7 +5242,18 @@ dependencies = [ "serde", "serde_spanned", "toml_datetime", - "winnow", + "winnow 0.5.40", +] + +[[package]] +name = "toml_edit" +version = "0.22.22" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4ae48d6208a266e853d946088ed816055e556cc6028c5e8e2b84d9fa5dd7c7f5" +dependencies = [ + "indexmap 2.6.0", + "toml_datetime", + "winnow 0.6.20", ] [[package]] @@ -4975,6 +5262,10 @@ version = "0.4.13" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "b8fa9be0de6cf49e536ce1851f987bd21a43b771b09473c3549a6c853db37c1c" dependencies = [ + "futures-core", + "futures-util", + "pin-project", + "pin-project-lite", "tower-layer", "tower-service", "tracing", @@ -5058,6 +5349,16 @@ dependencies = [ "tracing-core", ] +[[package]] +name = "tracing-serde" +version = "0.1.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "bc6b213177105856957181934e4920de57730fc69bf42c37ee5bb664d406d9e1" +dependencies = [ + "serde", + "tracing-core", +] + [[package]] name = "tracing-subscriber" version = "0.3.18" @@ -5068,12 +5369,15 @@ dependencies = [ "nu-ansi-term", "once_cell", "regex", + "serde", + "serde_json", "sharded-slab", "smallvec", "thread_local", "tracing", "tracing-core", "tracing-log", + "tracing-serde", ] [[package]] @@ -5137,18 +5441,6 @@ version = "0.1.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "eaea85b334db583fe3274d12b4cd1880032beab409c0d774be044d4480ab9a94" -[[package]] -name = "unicase" -version = "2.8.0" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7e51b68083f157f853b6379db119d1c1be0e6e4dec98101079dec41f6f5cf6df" - -[[package]] -name = "unicode-bidi" -version = "0.3.17" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5ab17db44d7388991a428b2ee655ce0c212e862eff1768a455c58f9aad6e7893" - [[package]] name = "unicode-ident" version = "1.0.13" @@ -5161,15 +5453,6 @@ version = "0.1.5" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3b09c83c3c29d37506a3e260c08c03743a6bb66a9cd432c6934ab501a190571f" -[[package]] -name = "unicode-normalization" -version = "0.1.24" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5033c97c4262335cded6d6fc3e5c18ab755e1a3dc96376350f3d8e9f009ad956" -dependencies = [ - "tinyvec", -] - [[package]] name = "unicode-segmentation" version = "1.12.0" @@ -5189,15 +5472,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ebc1c04c71510c7f702b52b7c350734c9ff1295c464a03335b00bb84fc54f853" [[package]] -name = "url" -version = "1.7.2" +name = "untrusted" +version = "0.9.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "dd4e7c0d531266369519a4aa4f399d748bd37043b00bde1e4ff1f60a120b355a" -dependencies = [ - "idna 0.1.5", - "matches", - "percent-encoding 1.0.1", -] +checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" [[package]] name = "url" @@ -5206,8 +5484,8 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "32f8b686cadd1473f4bd0117a5d28d36b1ade384ea9b5069a1c40aefed7fda60" dependencies = [ "form_urlencoded", - "idna 1.0.3", - "percent-encoding 2.3.1", + "idna", + "percent-encoding", "serde", ] @@ -5241,6 +5519,21 @@ version = "0.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "830b7e5d4d90034032940e4ace0d9a9a057e7a45cd94e6c007832e39edb82f6d" +[[package]] +name = "vec-collections" +version = "0.4.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3c9965c8f2ffed1dbcd16cafe18a009642f540fa22661c6cfd6309ddb02e4982" +dependencies = [ + "binary-merge", + "inplace-vec-builder", + "lazy_static", + "num-traits", + "serde", + "smallvec", + "sorted-iter", +] + [[package]] name = "version_check" version = "0.9.5" @@ -5285,12 +5578,6 @@ dependencies = [ "try-lock", ] -[[package]] -name = "wasi" -version = "0.9.0+wasi-snapshot-preview1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cccddf32554fecc6acb585f82a32a72e28b48f8c4c1883ddfeeeaa96f7d8e519" - [[package]] name = "wasi" version = "0.11.0+wasi-snapshot-preview1" @@ -5303,7 +5590,7 @@ version = "0.2.86" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "5bba0e8cb82ba49ff4e229459ff22a191bbe9a1cb3a341610c9c33efc27ddf73" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "serde", "serde_json", "wasm-bindgen-macro", @@ -5330,7 +5617,7 @@ version = "0.4.36" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "2d1985d03709c53167ce907ff394f5316aa22cb4e12761295c5dc57dacb6297e" dependencies = [ - "cfg-if 1.0.0", + "cfg-if", "js-sys", "wasm-bindgen", "web-sys", @@ -5399,6 +5686,15 @@ dependencies = [ "wasm-bindgen", ] +[[package]] +name = "webpki-roots" +version = "0.26.7" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "5d642ff16b7e79272ae451b7322067cdc17cadf68c23264be9d94a32319efe7e" +dependencies = [ + "rustls-pki-types", +] + [[package]] name = "winapi" version = "0.3.9" @@ -5421,7 +5717,7 @@ version = "0.1.9" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cf221c93e13a30d793f7645a0e7762c55d169dbb0a49671918a2319d289b10bb" dependencies = [ - "windows-sys 0.48.0", + "windows-sys 0.59.0", ] [[package]] @@ -5596,6 +5892,15 @@ dependencies = [ "memchr", ] +[[package]] +name = "winnow" +version = "0.6.20" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "36c1fec1a2bb5866f07c25f68c26e565c4c200aebb96d7e55710c19d3e8ac49b" +dependencies = [ + "memchr", +] + [[package]] name = "write16" version = "1.0.0" @@ -5731,13 +6036,13 @@ version = "0.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4352d1081da6922701401cdd4cbf29a2723feb4cfabb5771f6fee8e9276da1c7" dependencies = [ - "ark-ff", - "ark-std", + "ark-ff 0.4.2", + "ark-std 0.4.0", "bitvec", "blake2", "bls12_381", "byteorder", - "cfg-if 1.0.0", + "cfg-if", "group 0.12.1", "group 0.13.0", "halo2", @@ -5745,7 +6050,7 @@ dependencies = [ "jubjub", "lazy_static", "pasta_curves 0.5.1", - "rand 0.8.5", + "rand", "serde", "sha2", "sha3", diff --git a/noir/noir-repo/Cargo.toml b/noir/noir-repo/Cargo.toml index 0acee2a040b..567f1db085b 100644 --- a/noir/noir-repo/Cargo.toml +++ b/noir/noir-repo/Cargo.toml @@ -40,11 +40,11 @@ resolver = "2" [workspace.package] # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors = ["The Noir Team "] edition = "2021" -rust-version = "1.74.1" +rust-version = "1.75.0" license = "MIT OR Apache-2.0" repository = "https://github.com/noir-lang/noir/" @@ -57,13 +57,13 @@ unused_qualifications = "warn" [workspace.dependencies] # ACVM workspace dependencies -acir_field = { version = "1.0.0-beta.0", path = "acvm-repo/acir_field", default-features = false } -acir = { version = "1.0.0-beta.0", path = "acvm-repo/acir", default-features = false } -acvm = { version = "1.0.0-beta.0", path = "acvm-repo/acvm" } -brillig = { version = "1.0.0-beta.0", path = "acvm-repo/brillig", default-features = false } -brillig_vm = { version = "1.0.0-beta.0", path = "acvm-repo/brillig_vm", default-features = false } -acvm_blackbox_solver = { version = "1.0.0-beta.0", path = "acvm-repo/blackbox_solver", default-features = false } -bn254_blackbox_solver = { version = "1.0.0-beta.0", path = "acvm-repo/bn254_blackbox_solver", default-features = false } +acir_field = { version = "1.0.0-beta.1", path = "acvm-repo/acir_field", default-features = false } +acir = { version = "1.0.0-beta.1", path = "acvm-repo/acir", default-features = false } +acvm = { version = "1.0.0-beta.1", path = "acvm-repo/acvm" } +brillig = { version = "1.0.0-beta.1", path = "acvm-repo/brillig", default-features = false } +brillig_vm = { version = "1.0.0-beta.1", path = "acvm-repo/brillig_vm", default-features = false } +acvm_blackbox_solver = { version = "1.0.0-beta.1", path = "acvm-repo/blackbox_solver", default-features = false } +bn254_blackbox_solver = { version = "1.0.0-beta.1", path = "acvm-repo/bn254_blackbox_solver", default-features = false } # Noir compiler workspace dependencies fm = { path = "compiler/fm" } @@ -83,20 +83,18 @@ noir_lsp = { path = "tooling/lsp" } noir_debugger = { path = "tooling/debugger" } noirc_abi = { path = "tooling/noirc_abi" } noirc_artifacts = { path = "tooling/noirc_artifacts" } -bb_abstraction_leaks = { path = "tooling/bb_abstraction_leaks" } -acvm_cli = { path = "tooling/acvm_cli" } # Arkworks -ark-bn254 = { version = "^0.4.0", default-features = false, features = [ +ark-bn254 = { version = "^0.5.0", default-features = false, features = [ "curve", ] } -ark-bls12-381 = { version = "^0.4.0", default-features = false, features = [ +ark-bls12-381 = { version = "^0.5.0", default-features = false, features = [ "curve", ] } -grumpkin = { version = "0.1.0", package = "noir_grumpkin", features = ["std"] } -ark-ec = { version = "^0.4.0", default-features = false } -ark-ff = { version = "^0.4.0", default-features = false } -ark-std = { version = "^0.4.0", default-features = false } +ark-grumpkin = { version = "^0.5.0", default-features = false } +ark-ec = { version = "^0.5.0", default-features = false } +ark-ff = { version = "^0.5.0", default-features = false } +ark-std = { version = "^0.5.0", default-features = false } # Misc utils crates iter-extended = { path = "utils/iter-extended" } @@ -148,22 +146,26 @@ num-traits = "0.2" similar-asserts = "1.5.0" tempfile = "3.6.0" test-case = "3.3.1" -jsonrpc = { version = "0.16.0", features = ["minreq_http"] } +jsonrpsee = { version = "0.24.7", features = ["client-core"] } flate2 = "1.0.24" color-eyre = "0.6.2" rand = "0.8.5" -proptest = "1.2.0" -proptest-derive = "0.4.0" +# The `fork` and `timeout` feature doesn't compile with Wasm (wait-timeout doesn't find the `imp` module). +proptest = { version = "1.6.0", default-features = false, features = [ + "std", + "bit-set", +] } +proptest-derive = "0.5.0" rayon = "1.8.0" sha2 = { version = "0.10.6", features = ["compress"] } sha3 = "0.10.6" strum = "0.24" strum_macros = "0.24" - +tokio = "1.42" im = { version = "15.1", features = ["serde"] } tracing = "0.1.40" tracing-web = "0.1.3" -tracing-subscriber = { version = "0.3.18", features = ["env-filter"] } +tracing-subscriber = { version = "0.3.18", features = ["env-filter", "json"] } rust-embed = "6.6.0" [profile.dev] diff --git a/noir/noir-repo/README.md b/noir/noir-repo/README.md index 7f5cd5ce522..c2e41435b66 100644 --- a/noir/noir-repo/README.md +++ b/noir/noir-repo/README.md @@ -34,7 +34,7 @@ The current focus is to gather as much feedback as possible while in the alpha p ## Minimum Rust version -This workspace's minimum supported rustc version is 1.74.1. +This workspace's minimum supported rustc version is 1.75.0. ## License @@ -47,4 +47,4 @@ Unless you explicitly state otherwise, any contribution intentionally submitted [Forum]: https://forum.aztec.network/c/noir [Discord]: https://discord.gg/JtqzkdeQ6G [Documentation]: https://noir-lang.org/docs -[Contributing]: CONTRIBUTING.md \ No newline at end of file +[Contributing]: CONTRIBUTING.md diff --git a/noir/noir-repo/acvm-repo/acir/Cargo.toml b/noir/noir-repo/acvm-repo/acir/Cargo.toml index 8139a58eefc..d7ff0a085f0 100644 --- a/noir/noir-repo/acvm-repo/acir/Cargo.toml +++ b/noir/noir-repo/acvm-repo/acir/Cargo.toml @@ -2,7 +2,7 @@ name = "acir" description = "ACIR is the IR that the VM processes, it is analogous to LLVM IR" # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors.workspace = true edition.workspace = true diff --git a/noir/noir-repo/acvm-repo/acir/src/circuit/black_box_functions.rs b/noir/noir-repo/acvm-repo/acir/src/circuit/black_box_functions.rs index 700589d2040..d0ec7d02201 100644 --- a/noir/noir-repo/acvm-repo/acir/src/circuit/black_box_functions.rs +++ b/noir/noir-repo/acvm-repo/acir/src/circuit/black_box_functions.rs @@ -42,13 +42,13 @@ pub enum BlackBoxFunc { /// https://tools.ietf.org/html/rfc7693 /// - inputs are a byte array, i.e a vector of (witness, 8) /// - output is a byte array of length 32, i.e. an array of 32 - /// (witness, 8), constrained to be the blake2s of the inputs. + /// (witness, 8), constrained to be the blake2s of the inputs. Blake2s, /// Computes the Blake3 hash of the inputs /// - inputs are a byte array, i.e a vector of (witness, 8) /// - output is a byte array of length 32, i.e an array of 32 - /// (witness, 8), constrained to be the blake3 of the inputs. + /// (witness, 8), constrained to be the blake3 of the inputs. Blake3, /// Verifies a ECDSA signature over the secp256k1 curve. diff --git a/noir/noir-repo/acvm-repo/acir/src/circuit/mod.rs b/noir/noir-repo/acvm-repo/acir/src/circuit/mod.rs index 4ff581bf17a..e651e6998a4 100644 --- a/noir/noir-repo/acvm-repo/acir/src/circuit/mod.rs +++ b/noir/noir-repo/acvm-repo/acir/src/circuit/mod.rs @@ -281,7 +281,7 @@ impl Deserialize<'a>> Program { where D: Deserializer<'de>, { - let bytecode_b64: String = serde::Deserialize::deserialize(deserializer)?; + let bytecode_b64: String = Deserialize::deserialize(deserializer)?; let program_bytes = base64::engine::general_purpose::STANDARD .decode(bytecode_b64) .map_err(D::Error::custom)?; diff --git a/noir/noir-repo/acvm-repo/acir/src/circuit/opcodes.rs b/noir/noir-repo/acvm-repo/acir/src/circuit/opcodes.rs index f47c40b0dd7..dec58b5f90b 100644 --- a/noir/noir-repo/acvm-repo/acir/src/circuit/opcodes.rs +++ b/noir/noir-repo/acvm-repo/acir/src/circuit/opcodes.rs @@ -47,6 +47,7 @@ pub enum Opcode { /// - **express a constraint** on witnesses; for instance to express that a /// witness `w` is a boolean, you can add the opcode: `w*w-w=0` /// - or, to **compute the value** of an arithmetic operation of some inputs. + /// /// For instance, to multiply two witnesses `x` and `y`, you would use the /// opcode `z-x*y=0`, which would constrain `z` to be `x*y`. /// diff --git a/noir/noir-repo/acvm-repo/acir/src/native_types/expression/mod.rs b/noir/noir-repo/acvm-repo/acir/src/native_types/expression/mod.rs index 2bbbc39d0ca..cdb8974526f 100644 --- a/noir/noir-repo/acvm-repo/acir/src/native_types/expression/mod.rs +++ b/noir/noir-repo/acvm-repo/acir/src/native_types/expression/mod.rs @@ -85,6 +85,7 @@ impl Expression { /// /// - `mul_term` in an expression contains degree-2 terms /// - `linear_combinations` contains degree-1 terms + /// /// Hence, it is sufficient to check that there are no `mul_terms` /// /// Examples: @@ -98,11 +99,11 @@ impl Expression { /// Returns `true` if the expression can be seen as a degree-1 univariate polynomial /// /// - `mul_terms` in an expression can be univariate, however unless the coefficient - /// is zero, it is always degree-2. + /// is zero, it is always degree-2. /// - `linear_combinations` contains the sum of degree-1 terms, these terms do not - /// need to contain the same variable and so it can be multivariate. However, we - /// have thus far only checked if `linear_combinations` contains one term, so this - /// method will return false, if the `Expression` has not been simplified. + /// need to contain the same variable and so it can be multivariate. However, we + /// have thus far only checked if `linear_combinations` contains one term, so this + /// method will return false, if the `Expression` has not been simplified. /// /// Hence, we check in the simplest case if an expression is a degree-1 univariate, /// by checking if it contains no `mul_terms` and it contains one `linear_combination` term. diff --git a/noir/noir-repo/acvm-repo/acir_field/Cargo.toml b/noir/noir-repo/acvm-repo/acir_field/Cargo.toml index 039aefe355e..f9ae3a4ea3f 100644 --- a/noir/noir-repo/acvm-repo/acir_field/Cargo.toml +++ b/noir/noir-repo/acvm-repo/acir_field/Cargo.toml @@ -2,7 +2,7 @@ name = "acir_field" description = "The field implementation being used by ACIR." # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors.workspace = true edition.workspace = true diff --git a/noir/noir-repo/acvm-repo/acvm/Cargo.toml b/noir/noir-repo/acvm-repo/acvm/Cargo.toml index ba01ac8ec16..e0948720b8c 100644 --- a/noir/noir-repo/acvm-repo/acvm/Cargo.toml +++ b/noir/noir-repo/acvm-repo/acvm/Cargo.toml @@ -2,7 +2,7 @@ name = "acvm" description = "The virtual machine that processes ACIR given a backend/proof system." # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors.workspace = true edition.workspace = true @@ -33,11 +33,17 @@ bls12_381 = [ ] [dev-dependencies] -ark-bls12-381 = { version = "^0.4.0", default-features = false, features = [ +ark-bls12-381 = { version = "^0.5.0", default-features = false, features = [ "curve", ] } +ark-ff.workspace = true ark-bn254.workspace = true bn254_blackbox_solver.workspace = true proptest.workspace = true -zkhash = { version = "^0.2.0", default-features = false } num-bigint.workspace = true +zkhash = { version = "^0.2.0", default-features = false } + +ark-bn254-v04 = { package = "ark-bn254", version = "^0.4.0", default-features = false, features = [ + "curve", +] } +ark-ff-v04 = { package = "ark-ff", version = "^0.4.0", default-features = false } diff --git a/noir/noir-repo/acvm-repo/acvm/src/compiler/mod.rs b/noir/noir-repo/acvm-repo/acvm/src/compiler/mod.rs index daedd77c4a0..ee84f7bf60b 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/compiler/mod.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/compiler/mod.rs @@ -16,10 +16,6 @@ pub use simulator::CircuitSimulator; use transformers::transform_internal; pub use transformers::{transform, MIN_EXPRESSION_WIDTH}; -/// We need multiple passes to stabilize the output. -/// The value was determined by running tests. -const MAX_OPTIMIZER_PASSES: usize = 3; - /// This module moves and decomposes acir opcodes. The transformation map allows consumers of this module to map /// metadata they had about the opcodes to the new opcode structure generated after the transformation. #[derive(Debug)] @@ -84,41 +80,10 @@ pub fn compile( ) -> (Circuit, AcirTransformationMap) { let acir_opcode_positions = (0..acir.opcodes.len()).collect::>(); - if MAX_OPTIMIZER_PASSES == 0 { - return (acir, AcirTransformationMap::new(&acir_opcode_positions)); - } - - let mut pass = 0; - let mut prev_opcodes_hash = fxhash::hash64(&acir.opcodes); - let mut prev_acir = acir; - let mut prev_acir_opcode_positions = acir_opcode_positions; - - // For most test programs it would be enough to only loop `transform_internal`, - // but some of them don't stabilize unless we also repeat the backend agnostic optimizations. - let (mut acir, acir_opcode_positions) = loop { - let (acir, acir_opcode_positions) = - optimize_internal(prev_acir, prev_acir_opcode_positions); - - // Stop if we have already done at least one transform and an extra optimization changed nothing. - if pass > 0 && prev_opcodes_hash == fxhash::hash64(&acir.opcodes) { - break (acir, acir_opcode_positions); - } - - let (acir, acir_opcode_positions) = - transform_internal(acir, expression_width, acir_opcode_positions); - - let opcodes_hash = fxhash::hash64(&acir.opcodes); - - // Stop if the output hasn't change in this loop or we went too long. - if pass == MAX_OPTIMIZER_PASSES - 1 || prev_opcodes_hash == opcodes_hash { - break (acir, acir_opcode_positions); - } + let (acir, acir_opcode_positions) = optimize_internal(acir, acir_opcode_positions); - pass += 1; - prev_acir = acir; - prev_opcodes_hash = opcodes_hash; - prev_acir_opcode_positions = acir_opcode_positions; - }; + let (mut acir, acir_opcode_positions) = + transform_internal(acir, expression_width, acir_opcode_positions); let transformation_map = AcirTransformationMap::new(&acir_opcode_positions); acir.assert_messages = transform_assert_messages(acir.assert_messages, &transformation_map); diff --git a/noir/noir-repo/acvm-repo/acvm/src/compiler/optimizers/merge_expressions.rs b/noir/noir-repo/acvm-repo/acvm/src/compiler/optimizers/merge_expressions.rs index 43e32101cc5..e95c6207c3c 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/compiler/optimizers/merge_expressions.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/compiler/optimizers/merge_expressions.rs @@ -404,7 +404,7 @@ mod tests { ], q_c: FieldElement::zero(), }), - Opcode::BlackBoxFuncCall(acir::circuit::opcodes::BlackBoxFuncCall::RANGE { + Opcode::BlackBoxFuncCall(BlackBoxFuncCall::RANGE { input: FunctionInput::witness(Witness(3), 32), }), ]; diff --git a/noir/noir-repo/acvm-repo/acvm/src/compiler/optimizers/unused_memory.rs b/noir/noir-repo/acvm-repo/acvm/src/compiler/optimizers/unused_memory.rs index 1963430210f..1325a3b03cd 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/compiler/optimizers/unused_memory.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/compiler/optimizers/unused_memory.rs @@ -1,4 +1,4 @@ -use acir::circuit::{opcodes::BlockId, Circuit, Opcode}; +use acir::circuit::{brillig::BrilligInputs, opcodes::BlockId, Circuit, Opcode}; use std::collections::HashSet; /// `UnusedMemoryOptimizer` will remove initializations of memory blocks which are unused. @@ -29,6 +29,13 @@ impl UnusedMemoryOptimizer { Opcode::MemoryOp { block_id, .. } => { unused_memory_initialization.remove(block_id); } + Opcode::BrilligCall { inputs, .. } => { + for input in inputs { + if let BrilligInputs::MemoryArray(block) = input { + unused_memory_initialization.remove(block); + } + } + } _ => (), } } diff --git a/noir/noir-repo/acvm-repo/acvm/src/compiler/transformers/mod.rs b/noir/noir-repo/acvm-repo/acvm/src/compiler/transformers/mod.rs index a499aec1b30..77a5d2da34e 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/compiler/transformers/mod.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/compiler/transformers/mod.rs @@ -14,12 +14,17 @@ mod csat; pub(crate) use csat::CSatTransformer; pub use csat::MIN_EXPRESSION_WIDTH; +use tracing::info; use super::{ - optimizers::MergeExpressionsOptimizer, transform_assert_messages, AcirTransformationMap, - MAX_OPTIMIZER_PASSES, + optimizers::{MergeExpressionsOptimizer, RangeOptimizer}, + transform_assert_messages, AcirTransformationMap, }; +/// We need multiple passes to stabilize the output. +/// The value was determined by running tests. +const MAX_TRANSFORMER_PASSES: usize = 3; + /// Applies [`ProofSystemCompiler`][crate::ProofSystemCompiler] specific optimizations to a [`Circuit`]. pub fn transform( acir: Circuit, @@ -50,12 +55,18 @@ pub(super) fn transform_internal( expression_width: ExpressionWidth, mut acir_opcode_positions: Vec, ) -> (Circuit, Vec) { + if acir.opcodes.len() == 1 && matches!(acir.opcodes[0], Opcode::BrilligCall { .. }) { + info!("Program is fully unconstrained, skipping transformation pass"); + return (acir, acir_opcode_positions); + } + // Allow multiple passes until we have stable output. let mut prev_opcodes_hash = fxhash::hash64(&acir.opcodes); // For most test programs it would be enough to loop here, but some of them // don't stabilize unless we also repeat the backend agnostic optimizations. - for _ in 0..MAX_OPTIMIZER_PASSES { + for _ in 0..MAX_TRANSFORMER_PASSES { + info!("Number of opcodes {}", acir.opcodes.len()); let (new_acir, new_acir_opcode_positions) = transform_internal_once(acir, expression_width, acir_opcode_positions); @@ -217,6 +228,12 @@ fn transform_internal_once( ..acir }; + // The `MergeOptimizer` can merge two witnesses which have range opcodes applied to them + // so we run the `RangeOptimizer` afterwards to clear these up. + let range_optimizer = RangeOptimizer::new(acir); + let (acir, new_acir_opcode_positions) = + range_optimizer.replace_redundant_ranges(new_acir_opcode_positions); + (acir, new_acir_opcode_positions) } diff --git a/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/bigint.rs b/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/bigint.rs index ccad3510682..eb5a6f81e7d 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/bigint.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/bigint.rs @@ -10,14 +10,18 @@ use crate::pwg::OpcodeResolutionError; /// Resolve BigInt opcodes by storing BigInt values (and their moduli) by their ID in the BigIntSolver /// - When it encounters a bigint operation opcode, it performs the operation on the stored values -/// and store the result using the provided ID. +/// and store the result using the provided ID. /// - When it gets a to_bytes opcode, it simply looks up the value and resolves the output witness accordingly. -#[derive(Default)] pub(crate) struct AcvmBigIntSolver { bigint_solver: BigIntSolver, } impl AcvmBigIntSolver { + pub(crate) fn with_pedantic_solving(pedantic_solving: bool) -> AcvmBigIntSolver { + let bigint_solver = BigIntSolver::with_pedantic_solving(pedantic_solving); + AcvmBigIntSolver { bigint_solver } + } + pub(crate) fn bigint_from_bytes( &mut self, inputs: &[FunctionInput], @@ -39,6 +43,9 @@ impl AcvmBigIntSolver { outputs: &[Witness], initial_witness: &mut WitnessMap, ) -> Result<(), OpcodeResolutionError> { + if self.bigint_solver.pedantic_solving() && outputs.len() != 32 { + panic!("--pedantic-solving: bigint_to_bytes: outputs.len() != 32: {}", outputs.len()); + } let mut bytes = self.bigint_solver.bigint_to_bytes(input)?; while bytes.len() < outputs.len() { bytes.push(0); diff --git a/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/logic.rs b/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/logic.rs index 8468b0ca27a..3fa72d9b215 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/logic.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/logic.rs @@ -14,13 +14,14 @@ pub(super) fn and( lhs: &FunctionInput, rhs: &FunctionInput, output: &Witness, + pedantic_solving: bool, ) -> Result<(), OpcodeResolutionError> { assert_eq!( lhs.num_bits(), rhs.num_bits(), "number of bits specified for each input must be the same" ); - solve_logic_opcode(initial_witness, lhs, rhs, *output, |left, right| { + solve_logic_opcode(initial_witness, lhs, rhs, *output, pedantic_solving, |left, right| { bit_and(left, right, lhs.num_bits()) }) } @@ -32,13 +33,14 @@ pub(super) fn xor( lhs: &FunctionInput, rhs: &FunctionInput, output: &Witness, + pedantic_solving: bool, ) -> Result<(), OpcodeResolutionError> { assert_eq!( lhs.num_bits(), rhs.num_bits(), "number of bits specified for each input must be the same" ); - solve_logic_opcode(initial_witness, lhs, rhs, *output, |left, right| { + solve_logic_opcode(initial_witness, lhs, rhs, *output, pedantic_solving, |left, right| { bit_xor(left, right, lhs.num_bits()) }) } @@ -49,11 +51,13 @@ fn solve_logic_opcode( a: &FunctionInput, b: &FunctionInput, result: Witness, + pedantic_solving: bool, logic_op: impl Fn(F, F) -> F, ) -> Result<(), OpcodeResolutionError> { - // TODO(https://github.com/noir-lang/noir/issues/5985): re-enable these once we figure out how to combine these with existing + // TODO(https://github.com/noir-lang/noir/issues/5985): re-enable these by + // default once we figure out how to combine these with existing // noirc_frontend/noirc_evaluator overflow error messages - let skip_bitsize_checks = true; + let skip_bitsize_checks = !pedantic_solving; let w_l_value = input_to_value(initial_witness, *a, skip_bitsize_checks)?; let w_r_value = input_to_value(initial_witness, *b, skip_bitsize_checks)?; let assignment = logic_op(w_l_value, w_r_value); diff --git a/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/mod.rs b/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/mod.rs index 5137b18179b..2f10e333c24 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/mod.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/mod.rs @@ -76,9 +76,15 @@ pub(crate) fn solve( BlackBoxFuncCall::AES128Encrypt { inputs, iv, key, outputs } => { solve_aes128_encryption_opcode(initial_witness, inputs, iv, key, outputs) } - BlackBoxFuncCall::AND { lhs, rhs, output } => and(initial_witness, lhs, rhs, output), - BlackBoxFuncCall::XOR { lhs, rhs, output } => xor(initial_witness, lhs, rhs, output), - BlackBoxFuncCall::RANGE { input } => solve_range_opcode(initial_witness, input), + BlackBoxFuncCall::AND { lhs, rhs, output } => { + and(initial_witness, lhs, rhs, output, backend.pedantic_solving()) + } + BlackBoxFuncCall::XOR { lhs, rhs, output } => { + xor(initial_witness, lhs, rhs, output, backend.pedantic_solving()) + } + BlackBoxFuncCall::RANGE { input } => { + solve_range_opcode(initial_witness, input, backend.pedantic_solving()) + } BlackBoxFuncCall::Blake2s { inputs, outputs } => { solve_generic_256_hash_opcode(initial_witness, inputs, None, outputs, blake2s) } diff --git a/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/range.rs b/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/range.rs index 4f9be14360e..d7083e4b9c0 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/range.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/pwg/blackbox/range.rs @@ -7,10 +7,11 @@ use acir::{circuit::opcodes::FunctionInput, native_types::WitnessMap, AcirField} pub(crate) fn solve_range_opcode( initial_witness: &WitnessMap, input: &FunctionInput, + pedantic_solving: bool, ) -> Result<(), OpcodeResolutionError> { // TODO(https://github.com/noir-lang/noir/issues/5985): - // re-enable bitsize checks - let skip_bitsize_checks = true; + // re-enable bitsize checks by default + let skip_bitsize_checks = !pedantic_solving; let w_value = input_to_value(initial_witness, *input, skip_bitsize_checks)?; if w_value.num_bits() > input.num_bits() { return Err(OpcodeResolutionError::UnsatisfiedConstrain { diff --git a/noir/noir-repo/acvm-repo/acvm/src/pwg/memory_op.rs b/noir/noir-repo/acvm-repo/acvm/src/pwg/memory_op.rs index a9ed7f5d15b..2a83bf2531c 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/pwg/memory_op.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/pwg/memory_op.rs @@ -21,6 +21,19 @@ pub(crate) struct MemoryOpSolver { } impl MemoryOpSolver { + fn index_from_field(&self, index: F) -> Result> { + if index.num_bits() <= 32 { + let memory_index = index.try_to_u64().unwrap() as MemoryIndex; + Ok(memory_index) + } else { + Err(OpcodeResolutionError::IndexOutOfBounds { + opcode_location: ErrorLocation::Unresolved, + index, + array_size: self.block_len, + }) + } + } + fn write_memory_index( &mut self, index: MemoryIndex, @@ -29,7 +42,7 @@ impl MemoryOpSolver { if index >= self.block_len { return Err(OpcodeResolutionError::IndexOutOfBounds { opcode_location: ErrorLocation::Unresolved, - index, + index: F::from(index as u128), array_size: self.block_len, }); } @@ -40,7 +53,7 @@ impl MemoryOpSolver { fn read_memory_index(&self, index: MemoryIndex) -> Result> { self.block_value.get(&index).copied().ok_or(OpcodeResolutionError::IndexOutOfBounds { opcode_location: ErrorLocation::Unresolved, - index, + index: F::from(index as u128), array_size: self.block_len, }) } @@ -66,12 +79,13 @@ impl MemoryOpSolver { op: &MemOp, initial_witness: &mut WitnessMap, predicate: &Option>, + pedantic_solving: bool, ) -> Result<(), OpcodeResolutionError> { let operation = get_value(&op.operation, initial_witness)?; // Find the memory index associated with this memory operation. let index = get_value(&op.index, initial_witness)?; - let memory_index = index.try_to_u64().unwrap() as MemoryIndex; + let memory_index = self.index_from_field(index)?; // Calculate the value associated with this memory operation. // @@ -83,7 +97,9 @@ impl MemoryOpSolver { let is_read_operation = operation.is_zero(); // Fetch whether or not the predicate is false (e.g. equal to zero) - let skip_operation = is_predicate_false(initial_witness, predicate)?; + let opcode_location = ErrorLocation::Unresolved; + let skip_operation = + is_predicate_false(initial_witness, predicate, pedantic_solving, &opcode_location)?; if is_read_operation { // `value_read = arr[memory_index]` @@ -131,6 +147,9 @@ mod tests { use super::MemoryOpSolver; + // use pedantic_solving for tests + const PEDANTIC_SOLVING: bool = true; + #[test] fn test_solver() { let mut initial_witness = WitnessMap::from(BTreeMap::from_iter([ @@ -150,7 +169,9 @@ mod tests { block_solver.init(&init, &initial_witness).unwrap(); for op in trace { - block_solver.solve_memory_op(&op, &mut initial_witness, &None).unwrap(); + block_solver + .solve_memory_op(&op, &mut initial_witness, &None, PEDANTIC_SOLVING) + .unwrap(); } assert_eq!(initial_witness[&Witness(4)], FieldElement::from(2u128)); @@ -175,7 +196,9 @@ mod tests { let mut err = None; for op in invalid_trace { if err.is_none() { - err = block_solver.solve_memory_op(&op, &mut initial_witness, &None).err(); + err = block_solver + .solve_memory_op(&op, &mut initial_witness, &None, PEDANTIC_SOLVING) + .err(); } } @@ -183,9 +206,9 @@ mod tests { err, Some(crate::pwg::OpcodeResolutionError::IndexOutOfBounds { opcode_location: _, - index: 2, + index, array_size: 2 - }) + }) if index == FieldElement::from(2u128) )); } @@ -209,7 +232,12 @@ mod tests { for op in invalid_trace { if err.is_none() { err = block_solver - .solve_memory_op(&op, &mut initial_witness, &Some(Expression::zero())) + .solve_memory_op( + &op, + &mut initial_witness, + &Some(Expression::zero()), + PEDANTIC_SOLVING, + ) .err(); } } @@ -241,7 +269,12 @@ mod tests { for op in invalid_trace { if err.is_none() { err = block_solver - .solve_memory_op(&op, &mut initial_witness, &Some(Expression::zero())) + .solve_memory_op( + &op, + &mut initial_witness, + &Some(Expression::zero()), + PEDANTIC_SOLVING, + ) .err(); } } diff --git a/noir/noir-repo/acvm-repo/acvm/src/pwg/mod.rs b/noir/noir-repo/acvm-repo/acvm/src/pwg/mod.rs index f9188cca700..6e0e28cf81d 100644 --- a/noir/noir-repo/acvm-repo/acvm/src/pwg/mod.rs +++ b/noir/noir-repo/acvm-repo/acvm/src/pwg/mod.rs @@ -126,7 +126,7 @@ pub enum OpcodeResolutionError { payload: Option>, }, #[error("Index out of bounds, array has size {array_size:?}, but index was {index:?}")] - IndexOutOfBounds { opcode_location: ErrorLocation, index: u32, array_size: u32 }, + IndexOutOfBounds { opcode_location: ErrorLocation, index: F, array_size: u32 }, #[error("Cannot solve opcode: {invalid_input_bit_size}")] InvalidInputBitSize { opcode_location: ErrorLocation, @@ -144,6 +144,8 @@ pub enum OpcodeResolutionError { AcirMainCallAttempted { opcode_location: ErrorLocation }, #[error("{results_size:?} result values were provided for {outputs_size:?} call output witnesses, most likely due to bad ACIR codegen")] AcirCallOutputsMismatch { opcode_location: ErrorLocation, results_size: u32, outputs_size: u32 }, + #[error("(--pedantic): Predicates are expected to be 0 or 1, but found: {pred_value}")] + PredicateLargerThanOne { opcode_location: ErrorLocation, pred_value: F }, } impl From for OpcodeResolutionError { @@ -218,11 +220,12 @@ impl<'a, F: AcirField, B: BlackBoxFunctionSolver> ACVM<'a, F, B> { assertion_payloads: &'a [(OpcodeLocation, AssertionPayload)], ) -> Self { let status = if opcodes.is_empty() { ACVMStatus::Solved } else { ACVMStatus::InProgress }; + let bigint_solver = AcvmBigIntSolver::with_pedantic_solving(backend.pedantic_solving()); ACVM { status, backend, block_solvers: HashMap::default(), - bigint_solver: AcvmBigIntSolver::default(), + bigint_solver, opcodes, instruction_pointer: 0, witness_map: initial_witness, @@ -373,7 +376,12 @@ impl<'a, F: AcirField, B: BlackBoxFunctionSolver> ACVM<'a, F, B> { } Opcode::MemoryOp { block_id, op, predicate } => { let solver = self.block_solvers.entry(*block_id).or_default(); - solver.solve_memory_op(op, &mut self.witness_map, predicate) + solver.solve_memory_op( + op, + &mut self.witness_map, + predicate, + self.backend.pedantic_solving(), + ) } Opcode::BrilligCall { .. } => match self.solve_brillig_call_opcode() { Ok(Some(foreign_call)) => return self.wait_for_foreign_call(foreign_call), @@ -477,7 +485,14 @@ impl<'a, F: AcirField, B: BlackBoxFunctionSolver> ACVM<'a, F, B> { unreachable!("Not executing a BrilligCall opcode"); }; - if is_predicate_false(&self.witness_map, predicate)? { + let opcode_location = + ErrorLocation::Resolved(OpcodeLocation::Acir(self.instruction_pointer())); + if is_predicate_false( + &self.witness_map, + predicate, + self.backend.pedantic_solving(), + &opcode_location, + )? { return BrilligSolver::::zero_out_brillig_outputs(&mut self.witness_map, outputs) .map(|_| None); } @@ -545,8 +560,15 @@ impl<'a, F: AcirField, B: BlackBoxFunctionSolver> ACVM<'a, F, B> { return StepResult::Status(self.solve_opcode()); }; + let opcode_location = + ErrorLocation::Resolved(OpcodeLocation::Acir(self.instruction_pointer())); let witness = &mut self.witness_map; - let should_skip = match is_predicate_false(witness, predicate) { + let should_skip = match is_predicate_false( + witness, + predicate, + self.backend.pedantic_solving(), + &opcode_location, + ) { Ok(result) => result, Err(err) => return StepResult::Status(self.handle_opcode_resolution(Err(err))), }; @@ -587,15 +609,19 @@ impl<'a, F: AcirField, B: BlackBoxFunctionSolver> ACVM<'a, F, B> { else { unreachable!("Not executing a Call opcode"); }; + + let opcode_location = + ErrorLocation::Resolved(OpcodeLocation::Acir(self.instruction_pointer())); if *id == AcirFunctionId(0) { - return Err(OpcodeResolutionError::AcirMainCallAttempted { - opcode_location: ErrorLocation::Resolved(OpcodeLocation::Acir( - self.instruction_pointer(), - )), - }); + return Err(OpcodeResolutionError::AcirMainCallAttempted { opcode_location }); } - if is_predicate_false(&self.witness_map, predicate)? { + if is_predicate_false( + &self.witness_map, + predicate, + self.backend.pedantic_solving(), + &opcode_location, + )? { // Zero out the outputs if we have a false predicate for output in outputs { insert_value(output, F::zero(), &mut self.witness_map)?; @@ -615,9 +641,7 @@ impl<'a, F: AcirField, B: BlackBoxFunctionSolver> ACVM<'a, F, B> { let result_values = &self.acir_call_results[self.acir_call_counter]; if outputs.len() != result_values.len() { return Err(OpcodeResolutionError::AcirCallOutputsMismatch { - opcode_location: ErrorLocation::Resolved(OpcodeLocation::Acir( - self.instruction_pointer(), - )), + opcode_location, results_size: result_values.len() as u32, outputs_size: outputs.len() as u32, }); @@ -736,9 +760,25 @@ fn any_witness_from_expression(expr: &Expression) -> Option { pub(crate) fn is_predicate_false( witness: &WitnessMap, predicate: &Option>, + pedantic_solving: bool, + opcode_location: &ErrorLocation, ) -> Result> { match predicate { - Some(pred) => get_value(pred, witness).map(|pred_value| pred_value.is_zero()), + Some(pred) => { + let pred_value = get_value(pred, witness)?; + let predicate_is_false = pred_value.is_zero(); + if pedantic_solving { + // We expect that the predicate should resolve to either 0 or 1. + if !predicate_is_false && !pred_value.is_one() { + let opcode_location = *opcode_location; + return Err(OpcodeResolutionError::PredicateLargerThanOne { + opcode_location, + pred_value, + }); + } + } + Ok(predicate_is_false) + } // If the predicate is `None`, then we treat it as an unconditional `true` None => Ok(false), } diff --git a/noir/noir-repo/acvm-repo/acvm/tests/solver.rs b/noir/noir-repo/acvm-repo/acvm/tests/solver.rs index 8b164b7c0f2..2fd61050377 100644 --- a/noir/noir-repo/acvm-repo/acvm/tests/solver.rs +++ b/noir/noir-repo/acvm-repo/acvm/tests/solver.rs @@ -15,7 +15,7 @@ use acir::{ }; use acvm::pwg::{ACVMStatus, ErrorLocation, ForeignCallWaitInfo, OpcodeResolutionError, ACVM}; -use acvm_blackbox_solver::StubbedBlackBoxSolver; +use acvm_blackbox_solver::{BigIntSolver, StubbedBlackBoxSolver}; use bn254_blackbox_solver::{field_from_hex, Bn254BlackBoxSolver, POSEIDON2_CONFIG}; use brillig_vm::brillig::HeapValueType; @@ -24,10 +24,10 @@ use proptest::arbitrary::any; use proptest::prelude::*; use proptest::result::maybe_ok; use proptest::sample::select; -use zkhash::poseidon2::poseidon2_params::Poseidon2Params; #[test] fn bls12_381_circuit() { + let solver = StubbedBlackBoxSolver::default(); type Bls12FieldElement = GenericFieldElement; let addition = Opcode::AssertZero(Expression { @@ -47,7 +47,7 @@ fn bls12_381_circuit() { ]) .into(); - let mut acvm = ACVM::new(&StubbedBlackBoxSolver, &opcodes, witness_assignments, &[], &[]); + let mut acvm = ACVM::new(&solver, &opcodes, witness_assignments, &[], &[]); // use the partial witness generation solver with our acir program let solver_status = acvm.solve(); assert_eq!(solver_status, ACVMStatus::Solved, "should be fully solved"); @@ -60,6 +60,7 @@ fn bls12_381_circuit() { #[test] fn inversion_brillig_oracle_equivalence() { + let solver = StubbedBlackBoxSolver::default(); // Opcodes below describe the following: // fn main(x : Field, y : pub Field) { // let z = x + y; @@ -169,13 +170,7 @@ fn inversion_brillig_oracle_equivalence() { ]) .into(); let unconstrained_functions = vec![brillig_bytecode]; - let mut acvm = ACVM::new( - &StubbedBlackBoxSolver, - &opcodes, - witness_assignments, - &unconstrained_functions, - &[], - ); + let mut acvm = ACVM::new(&solver, &opcodes, witness_assignments, &unconstrained_functions, &[]); // use the partial witness generation solver with our acir program let solver_status = acvm.solve(); @@ -204,6 +199,7 @@ fn inversion_brillig_oracle_equivalence() { #[test] fn double_inversion_brillig_oracle() { + let solver = StubbedBlackBoxSolver::default(); // Opcodes below describe the following: // fn main(x : Field, y : pub Field) { // let z = x + y; @@ -335,13 +331,7 @@ fn double_inversion_brillig_oracle() { ]) .into(); let unconstrained_functions = vec![brillig_bytecode]; - let mut acvm = ACVM::new( - &StubbedBlackBoxSolver, - &opcodes, - witness_assignments, - &unconstrained_functions, - &[], - ); + let mut acvm = ACVM::new(&solver, &opcodes, witness_assignments, &unconstrained_functions, &[]); // use the partial witness generation solver with our acir program let solver_status = acvm.solve(); @@ -388,6 +378,7 @@ fn double_inversion_brillig_oracle() { #[test] fn oracle_dependent_execution() { + let solver = StubbedBlackBoxSolver::default(); // This test ensures that we properly track the list of opcodes which still need to be resolved // across any brillig foreign calls we may have to perform. // @@ -492,13 +483,7 @@ fn oracle_dependent_execution() { let witness_assignments = BTreeMap::from([(w_x, FieldElement::from(2u128)), (w_y, FieldElement::from(2u128))]).into(); let unconstrained_functions = vec![brillig_bytecode]; - let mut acvm = ACVM::new( - &StubbedBlackBoxSolver, - &opcodes, - witness_assignments, - &unconstrained_functions, - &[], - ); + let mut acvm = ACVM::new(&solver, &opcodes, witness_assignments, &unconstrained_functions, &[]); // use the partial witness generation solver with our acir program let solver_status = acvm.solve(); @@ -544,6 +529,7 @@ fn oracle_dependent_execution() { #[test] fn brillig_oracle_predicate() { + let solver = StubbedBlackBoxSolver::default(); let fe_0 = FieldElement::zero(); let fe_1 = FieldElement::one(); let w_x = Witness(1); @@ -614,13 +600,7 @@ fn brillig_oracle_predicate() { ]) .into(); let unconstrained_functions = vec![brillig_bytecode]; - let mut acvm = ACVM::new( - &StubbedBlackBoxSolver, - &opcodes, - witness_assignments, - &unconstrained_functions, - &[], - ); + let mut acvm = ACVM::new(&solver, &opcodes, witness_assignments, &unconstrained_functions, &[]); let solver_status = acvm.solve(); assert_eq!(solver_status, ACVMStatus::Solved, "should be fully solved"); @@ -630,6 +610,7 @@ fn brillig_oracle_predicate() { #[test] fn unsatisfied_opcode_resolved() { + let solver = StubbedBlackBoxSolver::default(); let a = Witness(0); let b = Witness(1); let c = Witness(2); @@ -655,8 +636,7 @@ fn unsatisfied_opcode_resolved() { let opcodes = vec![Opcode::AssertZero(opcode_a)]; let unconstrained_functions = vec![]; - let mut acvm = - ACVM::new(&StubbedBlackBoxSolver, &opcodes, values, &unconstrained_functions, &[]); + let mut acvm = ACVM::new(&solver, &opcodes, values, &unconstrained_functions, &[]); let solver_status = acvm.solve(); assert_eq!( solver_status, @@ -670,6 +650,7 @@ fn unsatisfied_opcode_resolved() { #[test] fn unsatisfied_opcode_resolved_brillig() { + let solver = StubbedBlackBoxSolver::default(); let a = Witness(0); let b = Witness(1); let c = Witness(2); @@ -779,8 +760,7 @@ fn unsatisfied_opcode_resolved_brillig() { Opcode::AssertZero(opcode_a), ]; let unconstrained_functions = vec![brillig_bytecode]; - let mut acvm = - ACVM::new(&StubbedBlackBoxSolver, &opcodes, values, &unconstrained_functions, &[]); + let mut acvm = ACVM::new(&solver, &opcodes, values, &unconstrained_functions, &[]); let solver_status = acvm.solve(); assert_eq!( solver_status, @@ -795,6 +775,8 @@ fn unsatisfied_opcode_resolved_brillig() { #[test] fn memory_operations() { + let solver = StubbedBlackBoxSolver::default(); + let initial_witness = WitnessMap::from(BTreeMap::from_iter([ (Witness(1), FieldElement::from(1u128)), (Witness(2), FieldElement::from(2u128)), @@ -829,8 +811,7 @@ fn memory_operations() { let opcodes = vec![init, read_op, expression]; let unconstrained_functions = vec![]; - let mut acvm = - ACVM::new(&StubbedBlackBoxSolver, &opcodes, initial_witness, &unconstrained_functions, &[]); + let mut acvm = ACVM::new(&solver, &opcodes, initial_witness, &unconstrained_functions, &[]); let solver_status = acvm.solve(); assert_eq!(solver_status, ACVMStatus::Solved); let witness_map = acvm.finalize(); @@ -838,39 +819,6 @@ fn memory_operations() { assert_eq!(witness_map[&Witness(8)], FieldElement::from(6u128)); } -fn allowed_bigint_moduli() -> Vec> { - let bn254_fq: Vec = vec![ - 0x47, 0xFD, 0x7C, 0xD8, 0x16, 0x8C, 0x20, 0x3C, 0x8d, 0xca, 0x71, 0x68, 0x91, 0x6a, 0x81, - 0x97, 0x5d, 0x58, 0x81, 0x81, 0xb6, 0x45, 0x50, 0xb8, 0x29, 0xa0, 0x31, 0xe1, 0x72, 0x4e, - 0x64, 0x30, - ]; - let bn254_fr: Vec = vec![ - 1, 0, 0, 240, 147, 245, 225, 67, 145, 112, 185, 121, 72, 232, 51, 40, 93, 88, 129, 129, - 182, 69, 80, 184, 41, 160, 49, 225, 114, 78, 100, 48, - ]; - let secpk1_fr: Vec = vec![ - 0x41, 0x41, 0x36, 0xD0, 0x8C, 0x5E, 0xD2, 0xBF, 0x3B, 0xA0, 0x48, 0xAF, 0xE6, 0xDC, 0xAE, - 0xBA, 0xFE, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, - 0xFF, 0xFF, - ]; - let secpk1_fq: Vec = vec![ - 0x2F, 0xFC, 0xFF, 0xFF, 0xFE, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, - 0xFF, 0xFF, - ]; - let secpr1_fq: Vec = vec![ - 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, 0x00, - 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0xFF, 0xFF, - 0xFF, 0xFF, - ]; - let secpr1_fr: Vec = vec![ - 81, 37, 99, 252, 194, 202, 185, 243, 132, 158, 23, 167, 173, 250, 230, 188, 255, 255, 255, - 255, 255, 255, 255, 255, 0, 0, 0, 0, 255, 255, 255, 255, - ]; - - vec![bn254_fq, bn254_fr, secpk1_fr, secpk1_fq, secpr1_fq, secpr1_fr] -} - /// Whether to use a FunctionInput::constant or FunctionInput::witness: /// /// (value, use_constant) @@ -917,6 +865,7 @@ fn solve_array_input_blackbox_call( inputs: Vec, num_outputs: usize, num_bits: Option, + pedantic_solving: bool, f: F, ) -> Result, OpcodeResolutionError> where @@ -924,6 +873,7 @@ where (Vec>, Vec), ) -> Result, OpcodeResolutionError>, { + let solver = Bn254BlackBoxSolver(pedantic_solving); let initial_witness_vec: Vec<_> = inputs.iter().enumerate().map(|(i, (x, _))| (Witness(i as u32), *x)).collect(); let outputs: Vec<_> = (0..num_outputs) @@ -935,8 +885,7 @@ where let op = Opcode::BlackBoxFuncCall(f((inputs.clone(), outputs.clone()))?); let opcodes = vec![op]; let unconstrained_functions = vec![]; - let mut acvm = - ACVM::new(&Bn254BlackBoxSolver, &opcodes, initial_witness, &unconstrained_functions, &[]); + let mut acvm = ACVM::new(&solver, &opcodes, initial_witness, &unconstrained_functions, &[]); let solver_status = acvm.solve(); assert_eq!(solver_status, ACVMStatus::Solved); let witness_map = acvm.finalize(); @@ -948,7 +897,7 @@ where } prop_compose! { - fn bigint_with_modulus()(modulus in select(allowed_bigint_moduli())) + fn bigint_with_modulus()(modulus in select(BigIntSolver::allowed_bigint_moduli())) (inputs in proptest::collection::vec(any::<(u8, bool)>(), modulus.len()), modulus in Just(modulus)) -> (Vec, Vec) { let inputs = inputs.into_iter().zip(modulus.iter()).map(|((input, use_constant), modulus_byte)| { @@ -1029,7 +978,9 @@ fn bigint_solve_binary_op_opt( modulus: Vec, lhs: Vec, rhs: Vec, + pedantic_solving: bool, ) -> Result, OpcodeResolutionError> { + let solver = StubbedBlackBoxSolver(pedantic_solving); let initial_witness_vec: Vec<_> = lhs .iter() .chain(rhs.iter()) @@ -1071,8 +1022,7 @@ fn bigint_solve_binary_op_opt( opcodes.push(bigint_to_op); let unconstrained_functions = vec![]; - let mut acvm = - ACVM::new(&StubbedBlackBoxSolver, &opcodes, initial_witness, &unconstrained_functions, &[]); + let mut acvm = ACVM::new(&solver, &opcodes, initial_witness, &unconstrained_functions, &[]); let solver_status = acvm.solve(); assert_eq!(solver_status, ACVMStatus::Solved); let witness_map = acvm.finalize(); @@ -1095,6 +1045,7 @@ fn solve_blackbox_func_call( lhs: (FieldElement, bool), // if false, use a Witness rhs: (FieldElement, bool), // if false, use a Witness ) -> Result> { + let solver = StubbedBlackBoxSolver::default(); let (lhs, lhs_constant) = lhs; let (rhs, rhs_constant) = rhs; @@ -1114,8 +1065,7 @@ fn solve_blackbox_func_call( let op = Opcode::BlackBoxFuncCall(blackbox_func_call(lhs_opt, rhs_opt)?); let opcodes = vec![op]; let unconstrained_functions = vec![]; - let mut acvm = - ACVM::new(&StubbedBlackBoxSolver, &opcodes, initial_witness, &unconstrained_functions, &[]); + let mut acvm = ACVM::new(&solver, &opcodes, initial_witness, &unconstrained_functions, &[]); let solver_status = acvm.solve(); assert_eq!(solver_status, ACVMStatus::Solved); let witness_map = acvm.finalize(); @@ -1202,21 +1152,41 @@ where fields.into_iter().map(|field| field.into_repr()).collect() } -fn into_repr_mat(fields: T) -> Vec> +// fn into_repr_mat(fields: T) -> Vec> +// where +// T: IntoIterator, +// U: IntoIterator, +// { +// fields.into_iter().map(|field| into_repr_vec(field)).collect() +// } + +fn into_old_ark_field(field: T) -> U where - T: IntoIterator, - U: IntoIterator, + T: AcirField, + U: ark_ff_v04::PrimeField, { - fields.into_iter().map(|field| into_repr_vec(field)).collect() + U::from_be_bytes_mod_order(&field.to_be_bytes()) +} + +fn into_new_ark_field(field: T) -> U +where + T: ark_ff_v04::PrimeField, + U: ark_ff::PrimeField, +{ + use zkhash::ark_ff::BigInteger; + + U::from_be_bytes_mod_order(&field.into_bigint().to_bytes_be()) } fn run_both_poseidon2_permutations( inputs: Vec, ) -> Result<(Vec, Vec), OpcodeResolutionError> { + let pedantic_solving = true; let result = solve_array_input_blackbox_call( inputs.clone(), inputs.len(), None, + pedantic_solving, poseidon2_permutation_op, )?; @@ -1224,12 +1194,17 @@ fn run_both_poseidon2_permutations( let poseidon2_d = 5; let rounds_f = POSEIDON2_CONFIG.rounds_f as usize; let rounds_p = POSEIDON2_CONFIG.rounds_p as usize; - let mat_internal_diag_m_1 = into_repr_vec(POSEIDON2_CONFIG.internal_matrix_diagonal); + let mat_internal_diag_m_1: Vec = + POSEIDON2_CONFIG.internal_matrix_diagonal.into_iter().map(into_old_ark_field).collect(); let mat_internal = vec![]; - let round_constants = into_repr_mat(POSEIDON2_CONFIG.round_constant); + let round_constants: Vec> = POSEIDON2_CONFIG + .round_constant + .into_iter() + .map(|fields| fields.into_iter().map(into_old_ark_field).collect()) + .collect(); - let external_poseidon2 = - zkhash::poseidon2::poseidon2::Poseidon2::new(&Arc::new(Poseidon2Params::new( + let external_poseidon2 = zkhash::poseidon2::poseidon2::Poseidon2::new(&Arc::new( + zkhash::poseidon2::poseidon2_params::Poseidon2Params::new( poseidon2_t, poseidon2_d, rounds_f, @@ -1237,11 +1212,16 @@ fn run_both_poseidon2_permutations( &mat_internal_diag_m_1, &mat_internal, &round_constants, - ))); - - let expected_result = - external_poseidon2.permutation(&into_repr_vec(drop_use_constant(&inputs))); - Ok((into_repr_vec(result), expected_result)) + ), + )); + + let expected_result = external_poseidon2.permutation( + &drop_use_constant(&inputs) + .into_iter() + .map(into_old_ark_field) + .collect::>(), + ); + Ok((into_repr_vec(result), expected_result.into_iter().map(into_new_ark_field).collect())) } // Using the given BigInt modulus, solve the following circuit: @@ -1256,8 +1236,9 @@ fn bigint_solve_binary_op( modulus: Vec, lhs: Vec, rhs: Vec, + pedantic_solving: bool, ) -> Vec { - bigint_solve_binary_op_opt(Some(middle_op), modulus, lhs, rhs).unwrap() + bigint_solve_binary_op_opt(Some(middle_op), modulus, lhs, rhs, pedantic_solving).unwrap() } // Using the given BigInt modulus, solve the following circuit: @@ -1267,8 +1248,38 @@ fn bigint_solve_binary_op( fn bigint_solve_from_to_le_bytes( modulus: Vec, inputs: Vec, + pedantic_solving: bool, ) -> Vec { - bigint_solve_binary_op_opt(None, modulus, inputs, vec![]).unwrap() + bigint_solve_binary_op_opt(None, modulus, inputs, vec![], pedantic_solving).unwrap() +} + +// Test bigint_solve_from_to_le_bytes with a guaranteed-invalid modulus +// and optional pedantic_solving +fn bigint_from_to_le_bytes_disallowed_modulus_helper( + modulus: &mut Vec, + patch_location: usize, + patch_amount: u8, + zero_or_ones_constant: bool, + use_constant: bool, + pedantic_solving: bool, +) -> (Vec, Vec) { + let allowed_moduli: HashSet> = + BigIntSolver::allowed_bigint_moduli().into_iter().collect(); + let mut patch_location = patch_location % modulus.len(); + let patch_amount = patch_amount.clamp(1, u8::MAX); + while allowed_moduli.contains(modulus) { + modulus[patch_location] = patch_amount.wrapping_add(modulus[patch_location]); + patch_location += 1; + patch_location %= modulus.len(); + } + + let zero_function_input = + if zero_or_ones_constant { FieldElement::zero() } else { FieldElement::one() }; + let zero: Vec<_> = modulus.iter().map(|_| (zero_function_input, use_constant)).collect(); + let expected_results: Vec<_> = drop_use_constant(&zero); + let results = bigint_solve_from_to_le_bytes(modulus.clone(), zero, pedantic_solving); + + (results, expected_results) } fn function_input_from_option( @@ -1361,6 +1372,7 @@ fn prop_assert_injective( distinct_inputs: Vec, num_outputs: usize, num_bits: Option, + pedantic_solving: bool, op: F, ) -> (bool, String) where @@ -1372,11 +1384,22 @@ where { let equal_inputs = drop_use_constant_eq(&inputs, &distinct_inputs); let message = format!("not injective:\n{:?}\n{:?}", &inputs, &distinct_inputs); - let outputs_not_equal = - solve_array_input_blackbox_call(inputs, num_outputs, num_bits, op.clone()) - .expect("injectivity test operations to have valid input") - != solve_array_input_blackbox_call(distinct_inputs, num_outputs, num_bits, op) - .expect("injectivity test operations to have valid input"); + let outputs_not_equal = solve_array_input_blackbox_call( + inputs, + num_outputs, + num_bits, + pedantic_solving, + op.clone(), + ) + .expect("injectivity test operations to have valid input") + != solve_array_input_blackbox_call( + distinct_inputs, + num_outputs, + num_bits, + pedantic_solving, + op, + ) + .expect("injectivity test operations to have valid input"); (equal_inputs || outputs_not_equal, message) } @@ -1456,10 +1479,12 @@ fn poseidon2_permutation_zeroes() { #[test] fn sha256_compression_zeros() { + let pedantic_solving = true; let results = solve_array_input_blackbox_call( - [(FieldElement::zero(), false); 24].try_into().unwrap(), + [(FieldElement::zero(), false); 24].into(), 8, None, + pedantic_solving, sha256_compression_op, ); let expected_results: Vec<_> = vec![ @@ -1474,7 +1499,8 @@ fn sha256_compression_zeros() { #[test] fn blake2s_zeros() { - let results = solve_array_input_blackbox_call(vec![], 32, None, blake2s_op); + let pedantic_solving = true; + let results = solve_array_input_blackbox_call(vec![], 32, None, pedantic_solving, blake2s_op); let expected_results: Vec<_> = vec![ 105, 33, 122, 48, 121, 144, 128, 148, 225, 17, 33, 208, 66, 53, 74, 124, 31, 85, 182, 72, 44, 161, 165, 30, 27, 37, 13, 253, 30, 208, 238, 249, @@ -1487,7 +1513,8 @@ fn blake2s_zeros() { #[test] fn blake3_zeros() { - let results = solve_array_input_blackbox_call(vec![], 32, None, blake3_op); + let pedantic_solving = true; + let results = solve_array_input_blackbox_call(vec![], 32, None, pedantic_solving, blake3_op); let expected_results: Vec<_> = vec![ 175, 19, 73, 185, 245, 249, 161, 166, 160, 64, 77, 234, 54, 220, 201, 73, 155, 203, 37, 201, 173, 193, 18, 183, 204, 154, 147, 202, 228, 31, 50, 98, @@ -1500,10 +1527,12 @@ fn blake3_zeros() { #[test] fn keccakf1600_zeros() { + let pedantic_solving = true; let results = solve_array_input_blackbox_call( [(FieldElement::zero(), false); 25].into(), 25, Some(64), + pedantic_solving, keccakf1600_op, ); let expected_results: Vec<_> = vec![ @@ -1619,7 +1648,8 @@ proptest! { fn sha256_compression_injective(inputs_distinct_inputs in any_distinct_inputs(None, 24, 24)) { let (inputs, distinct_inputs) = inputs_distinct_inputs; if inputs.len() == 24 && distinct_inputs.len() == 24 { - let (result, message) = prop_assert_injective(inputs, distinct_inputs, 8, None, sha256_compression_op); + let pedantic_solving = true; + let (result, message) = prop_assert_injective(inputs, distinct_inputs, 8, None, pedantic_solving, sha256_compression_op); prop_assert!(result, "{}", message); } } @@ -1627,14 +1657,16 @@ proptest! { #[test] fn blake2s_injective(inputs_distinct_inputs in any_distinct_inputs(None, 0, 32)) { let (inputs, distinct_inputs) = inputs_distinct_inputs; - let (result, message) = prop_assert_injective(inputs, distinct_inputs, 32, None, blake2s_op); + let pedantic_solving = true; + let (result, message) = prop_assert_injective(inputs, distinct_inputs, 32, None, pedantic_solving, blake2s_op); prop_assert!(result, "{}", message); } #[test] fn blake3_injective(inputs_distinct_inputs in any_distinct_inputs(None, 0, 32)) { let (inputs, distinct_inputs) = inputs_distinct_inputs; - let (result, message) = prop_assert_injective(inputs, distinct_inputs, 32, None, blake3_op); + let pedantic_solving = true; + let (result, message) = prop_assert_injective(inputs, distinct_inputs, 32, None, pedantic_solving, blake3_op); prop_assert!(result, "{}", message); } @@ -1643,7 +1675,8 @@ proptest! { let (inputs, distinct_inputs) = inputs_distinct_inputs; assert_eq!(inputs.len(), 25); assert_eq!(distinct_inputs.len(), 25); - let (result, message) = prop_assert_injective(inputs, distinct_inputs, 25, Some(64), keccakf1600_op); + let pedantic_solving = true; + let (result, message) = prop_assert_injective(inputs, distinct_inputs, 25, Some(64), pedantic_solving, keccakf1600_op); prop_assert!(result, "{}", message); } @@ -1652,12 +1685,13 @@ proptest! { #[should_panic(expected = "Failure(BlackBoxFunctionFailed(Poseidon2Permutation, \"the number of inputs does not match specified length. 6 != 7\"))")] fn poseidon2_permutation_invalid_size_fails(inputs_distinct_inputs in any_distinct_inputs(None, 6, 6)) { let (inputs, distinct_inputs) = inputs_distinct_inputs; - let (result, message) = prop_assert_injective(inputs, distinct_inputs, 1, None, poseidon2_permutation_invalid_len_op); + let pedantic_solving = true; + let (result, message) = prop_assert_injective(inputs, distinct_inputs, 1, None, pedantic_solving, poseidon2_permutation_invalid_len_op); prop_assert!(result, "{}", message); } #[test] - fn bigint_from_to_le_bytes_zero_one(modulus in select(allowed_bigint_moduli()), zero_or_ones_constant: bool, use_constant: bool) { + fn bigint_from_to_le_bytes_zero_one(modulus in select(BigIntSolver::allowed_bigint_moduli()), zero_or_ones_constant: bool, use_constant: bool) { let zero_function_input = if zero_or_ones_constant { FieldElement::one() } else { @@ -1665,25 +1699,46 @@ proptest! { }; let zero_or_ones: Vec<_> = modulus.iter().map(|_| (zero_function_input, use_constant)).collect(); let expected_results = drop_use_constant(&zero_or_ones); - let results = bigint_solve_from_to_le_bytes(modulus.clone(), zero_or_ones); + let pedantic_solving = true; + let results = bigint_solve_from_to_le_bytes(modulus.clone(), zero_or_ones, pedantic_solving); prop_assert_eq!(results, expected_results) } #[test] fn bigint_from_to_le_bytes((input, modulus) in bigint_with_modulus()) { let expected_results: Vec<_> = drop_use_constant(&input); - let results = bigint_solve_from_to_le_bytes(modulus.clone(), input); + let pedantic_solving = true; + let results = bigint_solve_from_to_le_bytes(modulus.clone(), input, pedantic_solving); prop_assert_eq!(results, expected_results) } #[test] // TODO(https://github.com/noir-lang/noir/issues/5580): desired behavior? - fn bigint_from_to_le_bytes_extra_input_bytes((input, modulus) in bigint_with_modulus(), extra_bytes_len: u8, extra_bytes in proptest::collection::vec(any::<(u8, bool)>(), u8::MAX as usize)) { - let mut input = input; - let mut extra_bytes: Vec<_> = extra_bytes.into_iter().take(extra_bytes_len as usize).map(|(x, use_constant)| (FieldElement::from(x as u128), use_constant)).collect(); + fn bigint_from_to_le_bytes_extra_input_bytes((mut input, modulus) in bigint_with_modulus(), extra_bytes_len: u8, extra_bytes in proptest::collection::vec(any::<(u8, bool)>(), u8::MAX as usize)) { + let mut extra_bytes: Vec<_> = extra_bytes + .into_iter() + .take(extra_bytes_len as usize) + .map(|(x, use_constant)| (FieldElement::from(x as u128), use_constant)) + .collect(); + input.append(&mut extra_bytes); + let expected_results: Vec<_> = drop_use_constant(&input); + let pedantic_solving = false; + let results = bigint_solve_from_to_le_bytes(modulus.clone(), input, pedantic_solving); + prop_assert_eq!(results, expected_results) + } + + #[test] + #[should_panic(expected = "--pedantic-solving: bigint_from_bytes: inputs.len() > modulus.len()")] + fn bigint_from_to_le_bytes_extra_input_bytes_with_pedantic((mut input, modulus) in bigint_with_modulus(), extra_bytes_len: u8, extra_bytes in proptest::collection::vec(any::<(u8, bool)>(), u8::MAX as usize)) { + let mut extra_bytes: Vec<_> = extra_bytes + .into_iter() + .take(extra_bytes_len as usize) + .map(|(x, use_constant)| (FieldElement::from(x as u128), use_constant)) + .collect(); input.append(&mut extra_bytes); let expected_results: Vec<_> = drop_use_constant(&input); - let results = bigint_solve_from_to_le_bytes(modulus.clone(), input); + let pedantic_solving = true; + let results = bigint_solve_from_to_le_bytes(modulus.clone(), input, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1696,94 +1751,109 @@ proptest! { let larger_value = FieldElement::from(std::cmp::max((u8::MAX as u16) + 1, larger_value) as u128); input[patch_location] = (larger_value, use_constant); let expected_results: Vec<_> = drop_use_constant(&input); - let results = bigint_solve_from_to_le_bytes(modulus.clone(), input); + let pedantic_solving = true; + let results = bigint_solve_from_to_le_bytes(modulus.clone(), input, pedantic_solving); prop_assert_eq!(results, expected_results) } #[test] // TODO(https://github.com/noir-lang/noir/issues/5578): this test attempts to use a guaranteed-invalid BigInt modulus // #[should_panic(expected = "attempt to add with overflow")] - fn bigint_from_to_le_bytes_disallowed_modulus(mut modulus in select(allowed_bigint_moduli()), patch_location: usize, patch_amount: u8, zero_or_ones_constant: bool, use_constant: bool) { - let allowed_moduli: HashSet> = allowed_bigint_moduli().into_iter().collect(); - let mut patch_location = patch_location % modulus.len(); - let patch_amount = patch_amount.clamp(1, u8::MAX); - while allowed_moduli.contains(&modulus) { - modulus[patch_location] = patch_amount.wrapping_add(modulus[patch_location]); - patch_location += 1; - patch_location %= modulus.len(); - } - - let zero_function_input = if zero_or_ones_constant { - FieldElement::zero() - } else { - FieldElement::one() - }; - let zero: Vec<_> = modulus.iter().map(|_| (zero_function_input, use_constant)).collect(); - let expected_results: Vec<_> = drop_use_constant(&zero); - let results = bigint_solve_from_to_le_bytes(modulus.clone(), zero); + fn bigint_from_to_le_bytes_disallowed_modulus(mut modulus in select(BigIntSolver::allowed_bigint_moduli()), patch_location: usize, patch_amount: u8, zero_or_ones_constant: bool, use_constant: bool) { + let pedantic_solving = false; + let (results, expected_results) = bigint_from_to_le_bytes_disallowed_modulus_helper( + &mut modulus, + patch_location, + patch_amount, + zero_or_ones_constant, + use_constant, + pedantic_solving, + ); + prop_assert_eq!(results, expected_results) + } + #[test] + #[should_panic(expected = "--pedantic-solving: bigint_from_bytes: disallowed modulus [")] + fn bigint_from_to_le_bytes_disallowed_modulus_panics_with_pedantic(mut modulus in select(BigIntSolver::allowed_bigint_moduli()), patch_location: usize, patch_amount: u8, zero_or_ones_constant: bool, use_constant: bool) { + let pedantic_solving = true; + let (results, expected_results) = bigint_from_to_le_bytes_disallowed_modulus_helper( + &mut modulus, + patch_location, + patch_amount, + zero_or_ones_constant, + use_constant, + pedantic_solving, + ); prop_assert_eq!(results, expected_results) } #[test] fn bigint_add_commutative((xs, ys, modulus) in bigint_pair_with_modulus()) { - let lhs_results = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), xs.clone(), ys.clone()); - let rhs_results = bigint_solve_binary_op(bigint_add_op(), modulus, ys, xs); + let pedantic_solving = true; + let lhs_results = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), xs.clone(), ys.clone(), pedantic_solving); + let rhs_results = bigint_solve_binary_op(bigint_add_op(), modulus, ys, xs, pedantic_solving); prop_assert_eq!(lhs_results, rhs_results) } #[test] fn bigint_mul_commutative((xs, ys, modulus) in bigint_pair_with_modulus()) { - let lhs_results = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs.clone(), ys.clone()); - let rhs_results = bigint_solve_binary_op(bigint_mul_op(), modulus, ys, xs); + let pedantic_solving = true; + let lhs_results = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs.clone(), ys.clone(), pedantic_solving); + let rhs_results = bigint_solve_binary_op(bigint_mul_op(), modulus, ys, xs, pedantic_solving); prop_assert_eq!(lhs_results, rhs_results) } #[test] fn bigint_add_associative((xs, ys, zs, modulus) in bigint_triple_with_modulus()) { + let pedantic_solving = true; + // f(f(xs, ys), zs) == - let op_xs_ys = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), xs.clone(), ys.clone()); + let op_xs_ys = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), xs.clone(), ys.clone(), pedantic_solving); let xs_ys = use_witnesses(op_xs_ys); - let op_xs_ys_op_zs = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), xs_ys, zs.clone()); + let op_xs_ys_op_zs = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), xs_ys, zs.clone(), pedantic_solving); // f(xs, f(ys, zs)) - let op_ys_zs = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), ys.clone(), zs.clone()); + let op_ys_zs = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), ys.clone(), zs.clone(), pedantic_solving); let ys_zs = use_witnesses(op_ys_zs); - let op_xs_op_ys_zs = bigint_solve_binary_op(bigint_add_op(), modulus, xs, ys_zs); + let op_xs_op_ys_zs = bigint_solve_binary_op(bigint_add_op(), modulus, xs, ys_zs, pedantic_solving); prop_assert_eq!(op_xs_ys_op_zs, op_xs_op_ys_zs) } #[test] fn bigint_mul_associative((xs, ys, zs, modulus) in bigint_triple_with_modulus()) { + let pedantic_solving = true; + // f(f(xs, ys), zs) == - let op_xs_ys = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs.clone(), ys.clone()); + let op_xs_ys = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs.clone(), ys.clone(), pedantic_solving); let xs_ys = use_witnesses(op_xs_ys); - let op_xs_ys_op_zs = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs_ys, zs.clone()); + let op_xs_ys_op_zs = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs_ys, zs.clone(), pedantic_solving); // f(xs, f(ys, zs)) - let op_ys_zs = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), ys.clone(), zs.clone()); + let op_ys_zs = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), ys.clone(), zs.clone(), pedantic_solving); let ys_zs = use_witnesses(op_ys_zs); - let op_xs_op_ys_zs = bigint_solve_binary_op(bigint_mul_op(), modulus, xs, ys_zs); + let op_xs_op_ys_zs = bigint_solve_binary_op(bigint_mul_op(), modulus, xs, ys_zs, pedantic_solving); prop_assert_eq!(op_xs_ys_op_zs, op_xs_op_ys_zs) } #[test] fn bigint_mul_add_distributive((xs, ys, zs, modulus) in bigint_triple_with_modulus()) { + let pedantic_solving = true; + // xs * (ys + zs) == - let add_ys_zs = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), ys.clone(), zs.clone()); + let add_ys_zs = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), ys.clone(), zs.clone(), pedantic_solving); let add_ys_zs = use_witnesses(add_ys_zs); - let mul_xs_add_ys_zs = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs.clone(), add_ys_zs); + let mul_xs_add_ys_zs = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs.clone(), add_ys_zs, pedantic_solving); // xs * ys + xs * zs - let mul_xs_ys = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs.clone(), ys); + let mul_xs_ys = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs.clone(), ys, pedantic_solving); let mul_xs_ys = use_witnesses(mul_xs_ys); - let mul_xs_zs = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs, zs); + let mul_xs_zs = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs, zs, pedantic_solving); let mul_xs_zs = use_witnesses(mul_xs_zs); - let add_mul_xs_ys_mul_xs_zs = bigint_solve_binary_op(bigint_add_op(), modulus, mul_xs_ys, mul_xs_zs); + let add_mul_xs_ys_mul_xs_zs = bigint_solve_binary_op(bigint_add_op(), modulus, mul_xs_ys, mul_xs_zs, pedantic_solving); prop_assert_eq!(mul_xs_add_ys_zs, add_mul_xs_ys_mul_xs_zs) } @@ -1793,7 +1863,8 @@ proptest! { fn bigint_add_zero_l((xs, modulus) in bigint_with_modulus()) { let zero = bigint_zeroed(&xs); let expected_results = drop_use_constant(&xs); - let results = bigint_solve_binary_op(bigint_add_op(), modulus, zero, xs); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_add_op(), modulus, zero, xs, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1802,7 +1873,8 @@ proptest! { fn bigint_mul_zero_l((xs, modulus) in bigint_with_modulus()) { let zero = bigint_zeroed(&xs); let expected_results = drop_use_constant(&zero); - let results = bigint_solve_binary_op(bigint_mul_op(), modulus, zero, xs); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_mul_op(), modulus, zero, xs, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1810,14 +1882,16 @@ proptest! { fn bigint_mul_one_l((xs, modulus) in bigint_with_modulus()) { let one = bigint_to_one(&xs); let expected_results: Vec<_> = drop_use_constant(&xs); - let results = bigint_solve_binary_op(bigint_mul_op(), modulus, one, xs); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_mul_op(), modulus, one, xs, pedantic_solving); prop_assert_eq!(results, expected_results) } #[test] fn bigint_sub_self((xs, modulus) in bigint_with_modulus()) { let expected_results = drop_use_constant(&bigint_zeroed(&xs)); - let results = bigint_solve_binary_op(bigint_sub_op(), modulus, xs.clone(), xs); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_sub_op(), modulus, xs.clone(), xs, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1825,7 +1899,8 @@ proptest! { fn bigint_sub_zero((xs, modulus) in bigint_with_modulus()) { let zero = bigint_zeroed(&xs); let expected_results: Vec<_> = drop_use_constant(&xs); - let results = bigint_solve_binary_op(bigint_sub_op(), modulus, xs, zero); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_sub_op(), modulus, xs, zero, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1833,14 +1908,16 @@ proptest! { fn bigint_sub_one((xs, modulus) in bigint_with_modulus()) { let one = bigint_to_one(&xs); let expected_results: Vec<_> = drop_use_constant(&xs); - let results = bigint_solve_binary_op(bigint_sub_op(), modulus, xs, one); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_sub_op(), modulus, xs, one, pedantic_solving); prop_assert!(results != expected_results, "{:?} == {:?}", results, expected_results) } #[test] fn bigint_div_self((xs, modulus) in bigint_with_modulus()) { let one = drop_use_constant(&bigint_to_one(&xs)); - let results = bigint_solve_binary_op(bigint_div_op(), modulus, xs.clone(), xs); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_div_op(), modulus, xs.clone(), xs, pedantic_solving); prop_assert_eq!(results, one) } @@ -1849,7 +1926,18 @@ proptest! { fn bigint_div_by_zero((xs, modulus) in bigint_with_modulus()) { let zero = bigint_zeroed(&xs); let expected_results = drop_use_constant(&zero); - let results = bigint_solve_binary_op(bigint_div_op(), modulus, xs, zero); + let pedantic_solving = false; + let results = bigint_solve_binary_op(bigint_div_op(), modulus, xs, zero, pedantic_solving); + prop_assert_eq!(results, expected_results) + } + + #[test] + #[should_panic(expected = "Attempted to divide BigInt by zero")] + fn bigint_div_by_zero_with_pedantic((xs, modulus) in bigint_with_modulus()) { + let zero = bigint_zeroed(&xs); + let expected_results = drop_use_constant(&zero); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_div_op(), modulus, xs, zero, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1857,7 +1945,8 @@ proptest! { fn bigint_div_one((xs, modulus) in bigint_with_modulus()) { let one = bigint_to_one(&xs); let expected_results = drop_use_constant(&xs); - let results = bigint_solve_binary_op(bigint_div_op(), modulus, xs, one); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_div_op(), modulus, xs, one, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1865,16 +1954,18 @@ proptest! { fn bigint_div_zero((xs, modulus) in bigint_with_modulus()) { let zero = bigint_zeroed(&xs); let expected_results = drop_use_constant(&zero); - let results = bigint_solve_binary_op(bigint_div_op(), modulus, zero, xs); + let pedantic_solving = true; + let results = bigint_solve_binary_op(bigint_div_op(), modulus, zero, xs, pedantic_solving); prop_assert_eq!(results, expected_results) } #[test] fn bigint_add_sub((xs, ys, modulus) in bigint_pair_with_modulus()) { let expected_results = drop_use_constant(&xs); - let add_results = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), xs, ys.clone()); + let pedantic_solving = true; + let add_results = bigint_solve_binary_op(bigint_add_op(), modulus.clone(), xs, ys.clone(), pedantic_solving); let add_bigint = use_witnesses(add_results); - let results = bigint_solve_binary_op(bigint_sub_op(), modulus, add_bigint, ys); + let results = bigint_solve_binary_op(bigint_sub_op(), modulus, add_bigint, ys, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1882,9 +1973,10 @@ proptest! { #[test] fn bigint_sub_add((xs, ys, modulus) in bigint_pair_with_modulus()) { let expected_results = drop_use_constant(&xs); - let sub_results = bigint_solve_binary_op(bigint_sub_op(), modulus.clone(), xs, ys.clone()); + let pedantic_solving = true; + let sub_results = bigint_solve_binary_op(bigint_sub_op(), modulus.clone(), xs, ys.clone(), pedantic_solving); let add_bigint = use_witnesses(sub_results); - let results = bigint_solve_binary_op(bigint_add_op(), modulus, add_bigint, ys); + let results = bigint_solve_binary_op(bigint_add_op(), modulus, add_bigint, ys, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1892,9 +1984,10 @@ proptest! { #[test] fn bigint_div_mul((xs, ys, modulus) in bigint_pair_with_modulus()) { let expected_results = drop_use_constant(&xs); - let div_results = bigint_solve_binary_op(bigint_div_op(), modulus.clone(), xs, ys.clone()); + let pedantic_solving = true; + let div_results = bigint_solve_binary_op(bigint_div_op(), modulus.clone(), xs, ys.clone(), pedantic_solving); let div_bigint = use_witnesses(div_results); - let results = bigint_solve_binary_op(bigint_mul_op(), modulus, div_bigint, ys); + let results = bigint_solve_binary_op(bigint_mul_op(), modulus, div_bigint, ys, pedantic_solving); prop_assert_eq!(results, expected_results) } @@ -1902,9 +1995,10 @@ proptest! { #[test] fn bigint_mul_div((xs, ys, modulus) in bigint_pair_with_modulus()) { let expected_results = drop_use_constant(&xs); - let mul_results = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs, ys.clone()); + let pedantic_solving = true; + let mul_results = bigint_solve_binary_op(bigint_mul_op(), modulus.clone(), xs, ys.clone(), pedantic_solving); let mul_bigint = use_witnesses(mul_results); - let results = bigint_solve_binary_op(bigint_div_op(), modulus, mul_bigint, ys); + let results = bigint_solve_binary_op(bigint_div_op(), modulus, mul_bigint, ys, pedantic_solving); prop_assert_eq!(results, expected_results) } diff --git a/noir/noir-repo/acvm-repo/acvm_js/Cargo.toml b/noir/noir-repo/acvm-repo/acvm_js/Cargo.toml index bd536817428..a69c5e2adec 100644 --- a/noir/noir-repo/acvm-repo/acvm_js/Cargo.toml +++ b/noir/noir-repo/acvm-repo/acvm_js/Cargo.toml @@ -2,7 +2,7 @@ name = "acvm_js" description = "Typescript wrapper around the ACVM allowing execution of ACIR code" # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors.workspace = true edition.workspace = true diff --git a/noir/noir-repo/acvm-repo/acvm_js/build.sh b/noir/noir-repo/acvm-repo/acvm_js/build.sh index c07d2d8a4c1..16fb26e55db 100755 --- a/noir/noir-repo/acvm-repo/acvm_js/build.sh +++ b/noir/noir-repo/acvm-repo/acvm_js/build.sh @@ -25,7 +25,7 @@ function run_if_available { require_command jq require_command cargo require_command wasm-bindgen -#require_command wasm-opt +require_command wasm-opt self_path=$(dirname "$(readlink -f "$0")") pname=$(cargo read-manifest | jq -r '.name') diff --git a/noir/noir-repo/acvm-repo/acvm_js/package.json b/noir/noir-repo/acvm-repo/acvm_js/package.json index 904263b5e27..85fcfd7adfb 100644 --- a/noir/noir-repo/acvm-repo/acvm_js/package.json +++ b/noir/noir-repo/acvm-repo/acvm_js/package.json @@ -1,6 +1,6 @@ { "name": "@noir-lang/acvm_js", - "version": "1.0.0-beta.0", + "version": "1.0.0-beta.1", "publishConfig": { "access": "public" }, diff --git a/noir/noir-repo/acvm-repo/acvm_js/src/execute.rs b/noir/noir-repo/acvm-repo/acvm_js/src/execute.rs index c3627d0eff5..e4d52063977 100644 --- a/noir/noir-repo/acvm-repo/acvm_js/src/execute.rs +++ b/noir/noir-repo/acvm-repo/acvm_js/src/execute.rs @@ -30,12 +30,27 @@ pub async fn execute_circuit( program: Vec, initial_witness: JsWitnessMap, foreign_call_handler: ForeignCallHandler, +) -> Result { + let pedantic_solving = false; + execute_circuit_pedantic(program, initial_witness, foreign_call_handler, pedantic_solving).await +} + +/// `execute_circuit` with pedantic ACVM solving +async fn execute_circuit_pedantic( + program: Vec, + initial_witness: JsWitnessMap, + foreign_call_handler: ForeignCallHandler, + pedantic_solving: bool, ) -> Result { console_error_panic_hook::set_once(); - let mut witness_stack = - execute_program_with_native_type_return(program, initial_witness, &foreign_call_handler) - .await?; + let mut witness_stack = execute_program_with_native_type_return( + program, + initial_witness, + &foreign_call_handler, + pedantic_solving, + ) + .await?; let witness_map = witness_stack.pop().expect("Should have at least one witness on the stack").witness; Ok(witness_map.into()) @@ -53,6 +68,23 @@ pub async fn execute_circuit_with_return_witness( program: Vec, initial_witness: JsWitnessMap, foreign_call_handler: ForeignCallHandler, +) -> Result { + let pedantic_solving = false; + execute_circuit_with_return_witness_pedantic( + program, + initial_witness, + foreign_call_handler, + pedantic_solving, + ) + .await +} + +/// `executeCircuitWithReturnWitness` with pedantic ACVM execution +async fn execute_circuit_with_return_witness_pedantic( + program: Vec, + initial_witness: JsWitnessMap, + foreign_call_handler: ForeignCallHandler, + pedantic_solving: bool, ) -> Result { console_error_panic_hook::set_once(); @@ -63,6 +95,7 @@ pub async fn execute_circuit_with_return_witness( &program, initial_witness, &foreign_call_handler, + pedantic_solving, ) .await?; let solved_witness = @@ -87,12 +120,27 @@ pub async fn execute_program( program: Vec, initial_witness: JsWitnessMap, foreign_call_handler: ForeignCallHandler, +) -> Result { + let pedantic_solving = false; + execute_program_pedantic(program, initial_witness, foreign_call_handler, pedantic_solving).await +} + +/// `execute_program` with pedantic ACVM solving +async fn execute_program_pedantic( + program: Vec, + initial_witness: JsWitnessMap, + foreign_call_handler: ForeignCallHandler, + pedantic_solving: bool, ) -> Result { console_error_panic_hook::set_once(); - let witness_stack = - execute_program_with_native_type_return(program, initial_witness, &foreign_call_handler) - .await?; + let witness_stack = execute_program_with_native_type_return( + program, + initial_witness, + &foreign_call_handler, + pedantic_solving, + ) + .await?; Ok(witness_stack.into()) } @@ -101,6 +149,7 @@ async fn execute_program_with_native_type_return( program: Vec, initial_witness: JsWitnessMap, foreign_call_executor: &ForeignCallHandler, + pedantic_solving: bool, ) -> Result, Error> { let program: Program = Program::deserialize_program(&program) .map_err(|_| JsExecutionError::new( @@ -109,16 +158,22 @@ async fn execute_program_with_native_type_return( None, None))?; - execute_program_with_native_program_and_return(&program, initial_witness, foreign_call_executor) - .await + execute_program_with_native_program_and_return( + &program, + initial_witness, + foreign_call_executor, + pedantic_solving, + ) + .await } async fn execute_program_with_native_program_and_return( program: &Program, initial_witness: JsWitnessMap, foreign_call_executor: &ForeignCallHandler, + pedantic_solving: bool, ) -> Result, Error> { - let blackbox_solver = Bn254BlackBoxSolver; + let blackbox_solver = Bn254BlackBoxSolver(pedantic_solving); let executor = ProgramExecutor::new( &program.functions, &program.unconstrained_functions, diff --git a/noir/noir-repo/acvm-repo/blackbox_solver/Cargo.toml b/noir/noir-repo/acvm-repo/blackbox_solver/Cargo.toml index fe3a938c503..2c2e0d7675e 100644 --- a/noir/noir-repo/acvm-repo/blackbox_solver/Cargo.toml +++ b/noir/noir-repo/acvm-repo/blackbox_solver/Cargo.toml @@ -2,7 +2,7 @@ name = "acvm_blackbox_solver" description = "A solver for the blackbox functions found in ACIR and Brillig" # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors.workspace = true edition.workspace = true @@ -43,6 +43,7 @@ libaes = "0.7.0" [dev-dependencies] proptest.workspace = true +num-prime = { version = "0.4.4", default-features = false, features = ["big-int"] } [features] bn254 = ["acir/bn254"] diff --git a/noir/noir-repo/acvm-repo/blackbox_solver/src/bigint.rs b/noir/noir-repo/acvm-repo/blackbox_solver/src/bigint.rs index 540862843ab..f7be1e80a55 100644 --- a/noir/noir-repo/acvm-repo/blackbox_solver/src/bigint.rs +++ b/noir/noir-repo/acvm-repo/blackbox_solver/src/bigint.rs @@ -8,16 +8,30 @@ use crate::BlackBoxResolutionError; /// Resolve BigInt opcodes by storing BigInt values (and their moduli) by their ID in a HashMap: /// - When it encounters a bigint operation opcode, it performs the operation on the stored values -/// and store the result using the provided ID. +/// and store the result using the provided ID. /// - When it gets a to_bytes opcode, it simply looks up the value and resolves the output witness accordingly. -#[derive(Default, Debug, Clone, PartialEq, Eq)] - +#[derive(Debug, Clone, PartialEq, Eq)] pub struct BigIntSolver { bigint_id_to_value: HashMap, bigint_id_to_modulus: HashMap, + + // Use pedantic ACVM solving + pedantic_solving: bool, } impl BigIntSolver { + pub fn with_pedantic_solving(pedantic_solving: bool) -> BigIntSolver { + BigIntSolver { + bigint_id_to_value: Default::default(), + bigint_id_to_modulus: Default::default(), + pedantic_solving, + } + } + + pub fn pedantic_solving(&self) -> bool { + self.pedantic_solving + } + pub fn get_bigint( &self, id: u32, @@ -51,6 +65,17 @@ impl BigIntSolver { modulus: &[u8], output: u32, ) -> Result<(), BlackBoxResolutionError> { + if self.pedantic_solving { + if !self.is_valid_modulus(modulus) { + panic!("--pedantic-solving: bigint_from_bytes: disallowed modulus {:?}", modulus); + } + if inputs.len() > modulus.len() { + panic!( + "--pedantic-solving: bigint_from_bytes: inputs.len() > modulus.len() {:?}", + (inputs.len(), modulus.len()) + ); + } + } let bigint = BigUint::from_bytes_le(inputs); self.bigint_id_to_value.insert(output, bigint); let modulus = BigUint::from_bytes_le(modulus); @@ -84,8 +109,14 @@ impl BigIntSolver { } BlackBoxFunc::BigIntMul => lhs * rhs, BlackBoxFunc::BigIntDiv => { + if self.pedantic_solving && rhs == BigUint::ZERO { + return Err(BlackBoxResolutionError::Failed( + func, + "Attempted to divide BigInt by zero".to_string(), + )); + } lhs * rhs.modpow(&(&modulus - BigUint::from(2_u32)), &modulus) - } //TODO ensure that modulus is prime + } _ => unreachable!("ICE - bigint_op must be called for an operation"), }; if result > modulus { @@ -96,16 +127,57 @@ impl BigIntSolver { self.bigint_id_to_modulus.insert(output, modulus); Ok(()) } + + pub fn allowed_bigint_moduli() -> Vec> { + let bn254_fq: Vec = vec![ + 0x47, 0xFD, 0x7C, 0xD8, 0x16, 0x8C, 0x20, 0x3C, 0x8d, 0xca, 0x71, 0x68, 0x91, 0x6a, + 0x81, 0x97, 0x5d, 0x58, 0x81, 0x81, 0xb6, 0x45, 0x50, 0xb8, 0x29, 0xa0, 0x31, 0xe1, + 0x72, 0x4e, 0x64, 0x30, + ]; + let bn254_fr: Vec = vec![ + 1, 0, 0, 240, 147, 245, 225, 67, 145, 112, 185, 121, 72, 232, 51, 40, 93, 88, 129, 129, + 182, 69, 80, 184, 41, 160, 49, 225, 114, 78, 100, 48, + ]; + let secpk1_fr: Vec = vec![ + 0x41, 0x41, 0x36, 0xD0, 0x8C, 0x5E, 0xD2, 0xBF, 0x3B, 0xA0, 0x48, 0xAF, 0xE6, 0xDC, + 0xAE, 0xBA, 0xFE, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, + ]; + let secpk1_fq: Vec = vec![ + 0x2F, 0xFC, 0xFF, 0xFF, 0xFE, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, + 0xFF, 0xFF, 0xFF, 0xFF, + ]; + let secpr1_fq: Vec = vec![ + 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, + 0xFF, 0xFF, 0xFF, 0xFF, + ]; + let secpr1_fr: Vec = vec![ + 81, 37, 99, 252, 194, 202, 185, 243, 132, 158, 23, 167, 173, 250, 230, 188, 255, 255, + 255, 255, 255, 255, 255, 255, 0, 0, 0, 0, 255, 255, 255, 255, + ]; + vec![bn254_fq, bn254_fr, secpk1_fr, secpk1_fq, secpr1_fq, secpr1_fr] + } + + pub fn is_valid_modulus(&self, modulus: &[u8]) -> bool { + Self::allowed_bigint_moduli().into_iter().any(|allowed_modulus| allowed_modulus == modulus) + } } /// Wrapper over the generic bigint solver to automatically assign bigint IDs. -#[derive(Default, Debug, Clone, PartialEq, Eq)] +#[derive(Debug, Clone, PartialEq, Eq)] pub struct BigIntSolverWithId { solver: BigIntSolver, last_id: u32, } impl BigIntSolverWithId { + pub fn with_pedantic_solving(pedantic_solving: bool) -> BigIntSolverWithId { + let solver = BigIntSolver::with_pedantic_solving(pedantic_solving); + BigIntSolverWithId { solver, last_id: Default::default() } + } + pub fn create_bigint_id(&mut self) -> u32 { let output = self.last_id; self.last_id += 1; @@ -145,3 +217,23 @@ impl BigIntSolverWithId { Ok(id) } } + +#[test] +fn all_allowed_bigint_moduli_are_prime() { + use num_prime::nt_funcs::is_prime; + use num_prime::Primality; + + for modulus in BigIntSolver::allowed_bigint_moduli() { + let modulus = BigUint::from_bytes_le(&modulus); + let prime_test_config = None; + match is_prime(&modulus, prime_test_config) { + Primality::Yes => (), + Primality::No => panic!("not all allowed_bigint_moduli are prime: {modulus}"), + Primality::Probable(probability) => { + if probability < 0.90 { + panic!("not all allowed_bigint_moduli are prime within the allowed probability: {} < 0.90", probability); + } + } + } + } +} diff --git a/noir/noir-repo/acvm-repo/blackbox_solver/src/curve_specific_solver.rs b/noir/noir-repo/acvm-repo/blackbox_solver/src/curve_specific_solver.rs index b8fc3f47033..37fe5d05363 100644 --- a/noir/noir-repo/acvm-repo/blackbox_solver/src/curve_specific_solver.rs +++ b/noir/noir-repo/acvm-repo/blackbox_solver/src/curve_specific_solver.rs @@ -7,6 +7,7 @@ use crate::BlackBoxResolutionError; /// /// Returns an [`BlackBoxResolutionError`] if the backend does not support the given [`acir::BlackBoxFunc`]. pub trait BlackBoxFunctionSolver { + fn pedantic_solving(&self) -> bool; fn multi_scalar_mul( &self, points: &[F], @@ -29,7 +30,16 @@ pub trait BlackBoxFunctionSolver { ) -> Result, BlackBoxResolutionError>; } -pub struct StubbedBlackBoxSolver; +// pedantic_solving: bool +pub struct StubbedBlackBoxSolver(pub bool); + +// pedantic_solving enabled by default +impl Default for StubbedBlackBoxSolver { + fn default() -> StubbedBlackBoxSolver { + let pedantic_solving = true; + StubbedBlackBoxSolver(pedantic_solving) + } +} impl StubbedBlackBoxSolver { fn fail(black_box_function: BlackBoxFunc) -> BlackBoxResolutionError { @@ -41,6 +51,9 @@ impl StubbedBlackBoxSolver { } impl BlackBoxFunctionSolver for StubbedBlackBoxSolver { + fn pedantic_solving(&self) -> bool { + self.0 + } fn multi_scalar_mul( &self, _points: &[F], diff --git a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/Cargo.toml b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/Cargo.toml index 825a0ef0481..146759cfd65 100644 --- a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/Cargo.toml +++ b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/Cargo.toml @@ -2,7 +2,7 @@ name = "bn254_blackbox_solver" description = "Solvers for black box functions which are specific for the bn254 curve" # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors.workspace = true edition.workspace = true @@ -22,7 +22,7 @@ hex.workspace = true lazy_static.workspace = true ark-bn254.workspace = true -grumpkin.workspace = true +ark-grumpkin.workspace = true ark-ec.workspace = true ark-ff.workspace = true num-bigint.workspace = true diff --git a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/embedded_curve_ops.rs b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/embedded_curve_ops.rs index e599fd25593..ffe7fa50089 100644 --- a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/embedded_curve_ops.rs +++ b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/embedded_curve_ops.rs @@ -1,5 +1,6 @@ // TODO(https://github.com/noir-lang/noir/issues/4932): rename this file to something more generic use ark_ec::AffineRepr; +use ark_ff::MontConfig; use num_bigint::BigUint; use crate::FieldElement; @@ -13,6 +14,7 @@ pub fn multi_scalar_mul( points: &[FieldElement], scalars_lo: &[FieldElement], scalars_hi: &[FieldElement], + pedantic_solving: bool, ) -> Result<(FieldElement, FieldElement, FieldElement), BlackBoxResolutionError> { if points.len() != 3 * scalars_lo.len() || scalars_lo.len() != scalars_hi.len() { return Err(BlackBoxResolutionError::Failed( @@ -21,9 +23,12 @@ pub fn multi_scalar_mul( )); } - let mut output_point = grumpkin::SWAffine::zero(); + let mut output_point = ark_grumpkin::Affine::zero(); for i in (0..points.len()).step_by(3) { + if pedantic_solving && points[i + 2] > FieldElement::one() { + panic!("--pedantic-solving: is_infinity expected to be a bool, but found to be > 1") + } let point = create_point(points[i], points[i + 1], points[i + 2] == FieldElement::from(1_u128)) .map_err(|e| BlackBoxResolutionError::Failed(BlackBoxFunc::MultiScalarMul, e))?; @@ -48,23 +53,23 @@ pub fn multi_scalar_mul( let grumpkin_integer = BigUint::from_bytes_be(&bytes); // Check if this is smaller than the grumpkin modulus - // if grumpkin_integer >= grumpkin::FrConfig::MODULUS.into() { - // return Err(BlackBoxResolutionError::Failed( - // BlackBoxFunc::MultiScalarMul, - // format!("{} is not a valid grumpkin scalar", grumpkin_integer.to_str_radix(16)), - // )); - // } + if pedantic_solving && grumpkin_integer >= ark_grumpkin::FrConfig::MODULUS.into() { + return Err(BlackBoxResolutionError::Failed( + BlackBoxFunc::MultiScalarMul, + format!("{} is not a valid grumpkin scalar", grumpkin_integer.to_str_radix(16)), + )); + } let iteration_output_point = - grumpkin::SWAffine::from(point.mul_bigint(grumpkin_integer.to_u64_digits())); + ark_grumpkin::Affine::from(point.mul_bigint(grumpkin_integer.to_u64_digits())); - output_point = grumpkin::SWAffine::from(output_point + iteration_output_point); + output_point = ark_grumpkin::Affine::from(output_point + iteration_output_point); } if let Some((out_x, out_y)) = output_point.xy() { Ok(( - FieldElement::from_repr(*out_x), - FieldElement::from_repr(*out_y), + FieldElement::from_repr(out_x), + FieldElement::from_repr(out_y), FieldElement::from(output_point.is_zero() as u128), )) } else { @@ -75,16 +80,33 @@ pub fn multi_scalar_mul( pub fn embedded_curve_add( input1: [FieldElement; 3], input2: [FieldElement; 3], + pedantic_solving: bool, ) -> Result<(FieldElement, FieldElement, FieldElement), BlackBoxResolutionError> { + if pedantic_solving && (input1[2] > FieldElement::one() || input2[2] > FieldElement::one()) { + panic!("--pedantic-solving: is_infinity expected to be a bool, but found to be > 1") + } + let point1 = create_point(input1[0], input1[1], input1[2] == FieldElement::one()) .map_err(|e| BlackBoxResolutionError::Failed(BlackBoxFunc::EmbeddedCurveAdd, e))?; let point2 = create_point(input2[0], input2[1], input2[2] == FieldElement::one()) .map_err(|e| BlackBoxResolutionError::Failed(BlackBoxFunc::EmbeddedCurveAdd, e))?; - let res = grumpkin::SWAffine::from(point1 + point2); + + if pedantic_solving { + for point in [point1, point2] { + if point == ark_grumpkin::Affine::zero() { + return Err(BlackBoxResolutionError::Failed( + BlackBoxFunc::EmbeddedCurveAdd, + format!("Infinite input: embedded_curve_add({}, {})", point1, point2), + )); + } + } + } + + let res = ark_grumpkin::Affine::from(point1 + point2); if let Some((res_x, res_y)) = res.xy() { Ok(( - FieldElement::from_repr(*res_x), - FieldElement::from_repr(*res_y), + FieldElement::from_repr(res_x), + FieldElement::from_repr(res_y), FieldElement::from(res.is_zero() as u128), )) } else if res.is_zero() { @@ -101,11 +123,11 @@ fn create_point( x: FieldElement, y: FieldElement, is_infinite: bool, -) -> Result { +) -> Result { if is_infinite { - return Ok(grumpkin::SWAffine::zero()); + return Ok(ark_grumpkin::Affine::zero()); } - let point = grumpkin::SWAffine::new_unchecked(x.into_repr(), y.into_repr()); + let point = ark_grumpkin::Affine::new_unchecked(x.into_repr(), y.into_repr()); if !point.is_on_curve() { return Err(format!("Point ({}, {}) is not on curve", x.to_hex(), y.to_hex())); }; @@ -118,11 +140,12 @@ fn create_point( #[cfg(test)] mod tests { use super::*; + use ark_ff::BigInteger; fn get_generator() -> [FieldElement; 3] { - let generator = grumpkin::SWAffine::generator(); - let generator_x = FieldElement::from_repr(*generator.x().unwrap()); - let generator_y = FieldElement::from_repr(*generator.y().unwrap()); + let generator = ark_grumpkin::Affine::generator(); + let generator_x = FieldElement::from_repr(generator.x().unwrap()); + let generator_y = FieldElement::from_repr(generator.y().unwrap()); [generator_x, generator_y, FieldElement::zero()] } @@ -131,7 +154,13 @@ mod tests { // We check that multiplying 1 by generator results in the generator let generator = get_generator(); - let res = multi_scalar_mul(&generator, &[FieldElement::one()], &[FieldElement::zero()])?; + let pedantic_solving = true; + let res = multi_scalar_mul( + &generator, + &[FieldElement::one()], + &[FieldElement::zero()], + pedantic_solving, + )?; assert_eq!(generator[0], res.0); assert_eq!(generator[1], res.1); @@ -144,7 +173,8 @@ mod tests { let scalars_lo = [FieldElement::one()]; let scalars_hi = [FieldElement::from(2u128)]; - let res = multi_scalar_mul(&points, &scalars_lo, &scalars_hi)?; + let pedantic_solving = true; + let res = multi_scalar_mul(&points, &scalars_lo, &scalars_hi, pedantic_solving)?; let x = "0702ab9c7038eeecc179b4f209991bcb68c7cb05bf4c532d804ccac36199c9a9"; let y = "23f10e9e43a3ae8d75d24154e796aae12ae7af546716e8f81a2564f1b5814130"; @@ -165,30 +195,33 @@ mod tests { "Limb 0000000000000000000000000000000100000000000000000000000000000000 is not less than 2^128".into(), )); - let res = multi_scalar_mul(&points, &[FieldElement::one()], &[invalid_limb]); + let pedantic_solving = true; + let res = + multi_scalar_mul(&points, &[FieldElement::one()], &[invalid_limb], pedantic_solving); assert_eq!(res, expected_error); - let res = multi_scalar_mul(&points, &[invalid_limb], &[FieldElement::one()]); + let res = + multi_scalar_mul(&points, &[invalid_limb], &[FieldElement::one()], pedantic_solving); assert_eq!(res, expected_error); } - // #[test] - // fn rejects_grumpkin_modulus() { - // let x = grumpkin::FrConfig::MODULUS.to_bytes_be(); - - // let low = FieldElement::from_be_bytes_reduce(&x[16..32]); - // let high = FieldElement::from_be_bytes_reduce(&x[0..16]); + #[test] + fn rejects_grumpkin_modulus_when_pedantic() { + let x = ark_grumpkin::FrConfig::MODULUS.to_bytes_be(); + let low = FieldElement::from_be_bytes_reduce(&x[16..32]); + let high = FieldElement::from_be_bytes_reduce(&x[0..16]); - // let res = multi_scalar_mul(&get_generator(), &[low], &[high]); + let pedantic_solving = true; + let res = multi_scalar_mul(&get_generator(), &[low], &[high], pedantic_solving); - // assert_eq!( - // res, - // Err(BlackBoxResolutionError::Failed( - // BlackBoxFunc::MultiScalarMul, - // "30644e72e131a029b85045b68181585d97816a916871ca8d3c208c16d87cfd47 is not a valid grumpkin scalar".into(), - // )) - // ); - // } + assert_eq!( + res, + Err(BlackBoxResolutionError::Failed( + BlackBoxFunc::MultiScalarMul, + "30644e72e131a029b85045b68181585d97816a916871ca8d3c208c16d87cfd47 is not a valid grumpkin scalar".into(), + )) + ); + } #[test] fn rejects_invalid_point() { @@ -197,10 +230,12 @@ mod tests { let valid_scalar_low = FieldElement::zero(); let valid_scalar_high = FieldElement::zero(); + let pedantic_solving = true; let res = multi_scalar_mul( &[invalid_point_x, invalid_point_y, FieldElement::zero()], &[valid_scalar_low], &[valid_scalar_high], + pedantic_solving, ); assert_eq!( @@ -218,7 +253,8 @@ mod tests { let scalars_lo = [FieldElement::from(2u128)]; let scalars_hi = []; - let res = multi_scalar_mul(&points, &scalars_lo, &scalars_hi); + let pedantic_solving = true; + let res = multi_scalar_mul(&points, &scalars_lo, &scalars_hi, pedantic_solving); assert_eq!( res, @@ -234,9 +270,11 @@ mod tests { let x = FieldElement::from(1u128); let y = FieldElement::from(2u128); + let pedantic_solving = true; let res = embedded_curve_add( [x, y, FieldElement::from(0u128)], [x, y, FieldElement::from(0u128)], + pedantic_solving, ); assert_eq!( @@ -248,16 +286,39 @@ mod tests { ); } + #[test] + fn rejects_addition_of_infinite_points_when_pedantic() { + let x = FieldElement::from(1u128); + let y = FieldElement::from(1u128); + + let pedantic_solving = true; + let res = embedded_curve_add( + [x, y, FieldElement::from(1u128)], + [x, y, FieldElement::from(1u128)], + pedantic_solving, + ); + + assert_eq!( + res, + Err(BlackBoxResolutionError::Failed( + BlackBoxFunc::EmbeddedCurveAdd, + "Infinite input: embedded_curve_add(infinity, infinity)".into(), + )) + ); + } + #[test] fn output_of_msm_matches_add() -> Result<(), BlackBoxResolutionError> { let points = get_generator(); let scalars_lo = [FieldElement::from(2u128)]; let scalars_hi = [FieldElement::zero()]; - let msm_res = multi_scalar_mul(&points, &scalars_lo, &scalars_hi)?; + let pedantic_solving = true; + let msm_res = multi_scalar_mul(&points, &scalars_lo, &scalars_hi, pedantic_solving)?; let add_res = embedded_curve_add( [points[0], points[1], FieldElement::from(0u128)], [points[0], points[1], FieldElement::from(0u128)], + pedantic_solving, )?; assert_eq!(msm_res.0, add_res.0); diff --git a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/generator/generators.rs b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/generator/generators.rs index a4125014d56..22b88de8ecd 100644 --- a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/generator/generators.rs +++ b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/generator/generators.rs @@ -4,19 +4,16 @@ use std::sync::OnceLock; -use ark_ec::short_weierstrass::Affine; - use acvm_blackbox_solver::blake3; -use grumpkin::GrumpkinParameters; +use ark_grumpkin::Affine; use super::hash_to_curve::hash_to_curve; pub(crate) const DEFAULT_DOMAIN_SEPARATOR: &[u8] = "DEFAULT_DOMAIN_SEPARATOR".as_bytes(); const NUM_DEFAULT_GENERATORS: usize = 8; -fn default_generators() -> &'static [Affine; NUM_DEFAULT_GENERATORS] { - static INSTANCE: OnceLock<[Affine; NUM_DEFAULT_GENERATORS]> = - OnceLock::new(); +fn default_generators() -> &'static [Affine; NUM_DEFAULT_GENERATORS] { + static INSTANCE: OnceLock<[Affine; NUM_DEFAULT_GENERATORS]> = OnceLock::new(); INSTANCE.get_or_init(|| { _derive_generators(DEFAULT_DOMAIN_SEPARATOR, NUM_DEFAULT_GENERATORS as u32, 0) .try_into() @@ -42,7 +39,7 @@ pub fn derive_generators( domain_separator_bytes: &[u8], num_generators: u32, starting_index: u32, -) -> Vec> { +) -> Vec { // We cache a small number of the default generators so we can reuse them without needing to repeatedly recalculate them. if domain_separator_bytes == DEFAULT_DOMAIN_SEPARATOR && starting_index + num_generators <= NUM_DEFAULT_GENERATORS as u32 @@ -59,7 +56,7 @@ fn _derive_generators( domain_separator_bytes: &[u8], num_generators: u32, starting_index: u32, -) -> Vec> { +) -> Vec { let mut generator_preimage = [0u8; 64]; let domain_hash = blake3(domain_separator_bytes).expect("hash should succeed"); //1st 32 bytes are blake3 domain_hash @@ -91,7 +88,7 @@ mod test { fn test_derive_generators() { let res = derive_generators("test domain".as_bytes(), 128, 0); - let is_unique = |y: Affine, j: usize| -> bool { + let is_unique = |y: Affine, j: usize| -> bool { for (i, res) in res.iter().enumerate() { if i != j && *res == y { return false; diff --git a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/generator/hash_to_curve.rs b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/generator/hash_to_curve.rs index c0197883442..3c284fa811c 100644 --- a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/generator/hash_to_curve.rs +++ b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/generator/hash_to_curve.rs @@ -4,10 +4,10 @@ use acvm_blackbox_solver::blake3; -use ark_ec::{short_weierstrass::Affine, AffineRepr, CurveConfig}; +use ark_ec::AffineRepr; use ark_ff::Field; use ark_ff::{BigInteger, PrimeField}; -use grumpkin::GrumpkinParameters; +use ark_grumpkin::{Affine, Fq}; /// Hash a seed buffer into a point /// @@ -40,7 +40,7 @@ use grumpkin::GrumpkinParameters; /// /// N.B. steps c. and e. are because the `sqrt()` algorithm can return 2 values, /// we need to a way to canonically distinguish between these 2 values and select a "preferred" one -pub(crate) fn hash_to_curve(seed: &[u8], attempt_count: u8) -> Affine { +pub(crate) fn hash_to_curve(seed: &[u8], attempt_count: u8) -> Affine { let seed_size = seed.len(); // expand by 2 bytes to cover incremental hash attempts let mut target_seed = seed.to_vec(); @@ -56,10 +56,10 @@ pub(crate) fn hash_to_curve(seed: &[u8], attempt_count: u8) -> Affine::BaseField as Field>::BasePrimeField::from_be_bytes_mod_order(&hash); - let x = ::BaseField::from_base_prime_field(x); + let x = Fq::from_be_bytes_mod_order(&hash); + let x = Fq::from_base_prime_field(x); - if let Some(point) = Affine::::get_point_from_x_unchecked(x, false) { + if let Some(point) = Affine::get_point_from_x_unchecked(x, false) { let parity_bit = hash_hi[0] > 127; let y_bit_set = point.y().unwrap().into_bigint().get_bit(0); if (parity_bit && !y_bit_set) || (!parity_bit && y_bit_set) { diff --git a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/lib.rs b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/lib.rs index f738a375ab1..fc031faefa2 100644 --- a/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/lib.rs +++ b/noir/noir-repo/acvm-repo/bn254_blackbox_solver/src/lib.rs @@ -20,16 +20,21 @@ pub use poseidon2::{ type FieldElement = acir::acir_field::GenericFieldElement; #[derive(Default)] -pub struct Bn254BlackBoxSolver; +// pedantic_solving: bool +pub struct Bn254BlackBoxSolver(pub bool); impl BlackBoxFunctionSolver for Bn254BlackBoxSolver { + fn pedantic_solving(&self) -> bool { + self.0 + } + fn multi_scalar_mul( &self, points: &[FieldElement], scalars_lo: &[FieldElement], scalars_hi: &[FieldElement], ) -> Result<(FieldElement, FieldElement, FieldElement), BlackBoxResolutionError> { - multi_scalar_mul(points, scalars_lo, scalars_hi) + multi_scalar_mul(points, scalars_lo, scalars_hi, self.pedantic_solving()) } fn ec_add( @@ -44,6 +49,7 @@ impl BlackBoxFunctionSolver for Bn254BlackBoxSolver { embedded_curve_add( [*input1_x, *input1_y, *input1_infinite], [*input2_x, *input2_y, *input2_infinite], + self.pedantic_solving(), ) } diff --git a/noir/noir-repo/acvm-repo/brillig/Cargo.toml b/noir/noir-repo/acvm-repo/brillig/Cargo.toml index 9cc724f2b11..0d52b4b05ed 100644 --- a/noir/noir-repo/acvm-repo/brillig/Cargo.toml +++ b/noir/noir-repo/acvm-repo/brillig/Cargo.toml @@ -2,7 +2,7 @@ name = "brillig" description = "Brillig is the bytecode ACIR uses for non-determinism." # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors.workspace = true edition.workspace = true diff --git a/noir/noir-repo/acvm-repo/brillig_vm/Cargo.toml b/noir/noir-repo/acvm-repo/brillig_vm/Cargo.toml index 8225244f9a7..41d845f0d9e 100644 --- a/noir/noir-repo/acvm-repo/brillig_vm/Cargo.toml +++ b/noir/noir-repo/acvm-repo/brillig_vm/Cargo.toml @@ -2,7 +2,7 @@ name = "brillig_vm" description = "The virtual machine that processes Brillig bytecode, used to introduce non-determinism to the ACVM" # x-release-please-start-version -version = "1.0.0-beta.0" +version = "1.0.0-beta.1" # x-release-please-end authors.workspace = true edition.workspace = true diff --git a/noir/noir-repo/acvm-repo/brillig_vm/src/lib.rs b/noir/noir-repo/acvm-repo/brillig_vm/src/lib.rs index 5b3688339b5..8602f74e0d6 100644 --- a/noir/noir-repo/acvm-repo/brillig_vm/src/lib.rs +++ b/noir/noir-repo/acvm-repo/brillig_vm/src/lib.rs @@ -111,6 +111,8 @@ impl<'a, F: AcirField, B: BlackBoxFunctionSolver> VM<'a, F, B> { black_box_solver: &'a B, profiling_active: bool, ) -> Self { + let bigint_solver = + BrilligBigIntSolver::with_pedantic_solving(black_box_solver.pedantic_solving()); Self { calldata, program_counter: 0, @@ -121,7 +123,7 @@ impl<'a, F: AcirField, B: BlackBoxFunctionSolver> VM<'a, F, B> { memory: Memory::default(), call_stack: Vec::new(), black_box_solver, - bigint_solver: Default::default(), + bigint_solver, profiling_active, profiling_samples: Vec::with_capacity(bytecode.len()), } @@ -854,7 +856,8 @@ mod tests { }]; // Start VM - let mut vm = VM::new(calldata, &opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, &opcodes, vec![], &solver, false); let status = vm.process_opcode(); assert_eq!(status, VMStatus::Finished { return_data_offset: 0, return_data_size: 0 }); @@ -904,7 +907,8 @@ mod tests { Opcode::JumpIf { condition: destination, location: 6 }, ]; - let mut vm = VM::new(calldata, &opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, &opcodes, vec![], &solver, false); let status = vm.process_opcode(); assert_eq!(status, VMStatus::InProgress); @@ -972,7 +976,8 @@ mod tests { }, ]; - let mut vm = VM::new(calldata, &opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, &opcodes, vec![], &solver, false); let status = vm.process_opcode(); assert_eq!(status, VMStatus::InProgress); @@ -1044,7 +1049,8 @@ mod tests { }, }, ]; - let mut vm = VM::new(calldata, opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, opcodes, vec![], &solver, false); let status = vm.process_opcode(); assert_eq!(status, VMStatus::InProgress); @@ -1104,7 +1110,8 @@ mod tests { }, }, ]; - let mut vm = VM::new(calldata, opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, opcodes, vec![], &solver, false); let status = vm.process_opcode(); assert_eq!(status, VMStatus::InProgress); @@ -1150,7 +1157,8 @@ mod tests { }, Opcode::Mov { destination: MemoryAddress::direct(2), source: MemoryAddress::direct(0) }, ]; - let mut vm = VM::new(calldata, opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, opcodes, vec![], &solver, false); let status = vm.process_opcode(); assert_eq!(status, VMStatus::InProgress); @@ -1215,7 +1223,8 @@ mod tests { condition: MemoryAddress::direct(1), }, ]; - let mut vm = VM::new(calldata, opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, opcodes, vec![], &solver, false); let status = vm.process_opcode(); assert_eq!(status, VMStatus::InProgress); @@ -1311,7 +1320,8 @@ mod tests { .chain(cast_opcodes) .chain([equal_opcode, not_equal_opcode, less_than_opcode, less_than_equal_opcode]) .collect(); - let mut vm = VM::new(calldata, &opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, &opcodes, vec![], &solver, false); // Calldata copy let status = vm.process_opcode(); @@ -1411,7 +1421,8 @@ mod tests { ]; let opcodes = [&start[..], &loop_body[..]].concat(); - let vm = brillig_execute_and_get_vm(vec![], &opcodes); + let solver = StubbedBlackBoxSolver::default(); + let vm = brillig_execute_and_get_vm(vec![], &opcodes, &solver); vm.get_memory()[4..].to_vec() } @@ -1439,7 +1450,8 @@ mod tests { value: FieldElement::from(27_usize), }, ]; - let mut vm = VM::new(vec![], opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(vec![], opcodes, vec![], &solver, false); let status = vm.process_opcode(); assert_eq!(status, VMStatus::InProgress); @@ -1553,7 +1565,8 @@ mod tests { ]; let opcodes = [&start[..], &loop_body[..]].concat(); - let vm = brillig_execute_and_get_vm(memory, &opcodes); + let solver = StubbedBlackBoxSolver::default(); + let vm = brillig_execute_and_get_vm(memory, &opcodes, &solver); vm.memory.read(r_sum).to_field() } @@ -1644,7 +1657,8 @@ mod tests { ]; let opcodes = [&start[..], &recursive_fn[..]].concat(); - let vm = brillig_execute_and_get_vm(vec![], &opcodes); + let solver = StubbedBlackBoxSolver::default(); + let vm = brillig_execute_and_get_vm(vec![], &opcodes, &solver); vm.get_memory()[4..].to_vec() } @@ -1659,11 +1673,12 @@ mod tests { } /// Helper to execute brillig code - fn brillig_execute_and_get_vm( + fn brillig_execute_and_get_vm<'a, F: AcirField>( calldata: Vec, - opcodes: &[Opcode], - ) -> VM<'_, F, StubbedBlackBoxSolver> { - let mut vm = VM::new(calldata, opcodes, vec![], &StubbedBlackBoxSolver, false); + opcodes: &'a [Opcode], + solver: &'a StubbedBlackBoxSolver, + ) -> VM<'a, F, StubbedBlackBoxSolver> { + let mut vm = VM::new(calldata, opcodes, vec![], solver, false); brillig_execute(&mut vm); assert_eq!(vm.call_stack, vec![]); vm @@ -1705,7 +1720,8 @@ mod tests { }, ]; - let mut vm = brillig_execute_and_get_vm(vec![], &double_program); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = brillig_execute_and_get_vm(vec![], &double_program, &solver); // Check that VM is waiting assert_eq!( @@ -1798,7 +1814,8 @@ mod tests { }, ]; - let mut vm = brillig_execute_and_get_vm(initial_matrix.clone(), &invert_program); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = brillig_execute_and_get_vm(initial_matrix.clone(), &invert_program, &solver); // Check that VM is waiting assert_eq!( @@ -1908,7 +1925,9 @@ mod tests { }, ]; - let mut vm = brillig_execute_and_get_vm(input_string.clone(), &string_double_program); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = + brillig_execute_and_get_vm(input_string.clone(), &string_double_program, &solver); // Check that VM is waiting assert_eq!( @@ -2006,7 +2025,8 @@ mod tests { }, ]; - let mut vm = brillig_execute_and_get_vm(initial_matrix.clone(), &invert_program); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = brillig_execute_and_get_vm(initial_matrix.clone(), &invert_program, &solver); // Check that VM is waiting assert_eq!( @@ -2128,7 +2148,8 @@ mod tests { ]; let mut initial_memory = matrix_a.clone(); initial_memory.extend(matrix_b.clone()); - let mut vm = brillig_execute_and_get_vm(initial_memory, &matrix_mul_program); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = brillig_execute_and_get_vm(initial_memory, &matrix_mul_program, &solver); // Check that VM is waiting assert_eq!( @@ -2268,9 +2289,11 @@ mod tests { ]) .collect(); + let solver = StubbedBlackBoxSolver::default(); let mut vm = brillig_execute_and_get_vm( memory.into_iter().map(|mem_value| mem_value.to_field()).collect(), &program, + &solver, ); // Check that VM is waiting @@ -2342,7 +2365,8 @@ mod tests { }, ]; - let mut vm = VM::new(calldata, &opcodes, vec![], &StubbedBlackBoxSolver, false); + let solver = StubbedBlackBoxSolver::default(); + let mut vm = VM::new(calldata, &opcodes, vec![], &solver, false); vm.process_opcode(); vm.process_opcode(); diff --git a/noir/noir-repo/compiler/integration-tests/package.json b/noir/noir-repo/compiler/integration-tests/package.json index e33179f31e7..bfaa1cabd16 100644 --- a/noir/noir-repo/compiler/integration-tests/package.json +++ b/noir/noir-repo/compiler/integration-tests/package.json @@ -13,7 +13,7 @@ "lint": "NODE_NO_WARNINGS=1 eslint . --ext .ts --ignore-path ./.eslintignore --max-warnings 0" }, "dependencies": { - "@aztec/bb.js": "portal:../../../../barretenberg/ts", + "@aztec/bb.js": "0.66.0", "@noir-lang/noir_js": "workspace:*", "@noir-lang/noir_wasm": "workspace:*", "@nomicfoundation/hardhat-chai-matchers": "^2.0.0", diff --git a/noir/noir-repo/compiler/noirc_driver/src/lib.rs b/noir/noir-repo/compiler/noirc_driver/src/lib.rs index 9318e4d2b5c..a7e7e2d4e2f 100644 --- a/noir/noir-repo/compiler/noirc_driver/src/lib.rs +++ b/noir/noir-repo/compiler/noirc_driver/src/lib.rs @@ -73,7 +73,11 @@ pub struct CompileOptions { /// Only show SSA passes whose name contains the provided string. /// This setting takes precedence over `show_ssa` if it's not empty. #[arg(long, hide = true)] - pub show_ssa_pass_name: Option, + pub show_ssa_pass: Option, + + /// Only show the SSA and ACIR for the contract function with a given name. + #[arg(long, hide = true)] + pub show_contract_fn: Option, /// Emit the unoptimized SSA IR to file. /// The IR will be dumped into the workspace target directory, @@ -150,6 +154,16 @@ pub struct CompileOptions { /// A lower value keeps the original program if it was smaller, even if it has more jumps. #[arg(long, hide = true, allow_hyphen_values = true)] pub max_bytecode_increase_percent: Option, + + /// Use pedantic ACVM solving, i.e. double-check some black-box function + /// assumptions when solving. + /// This is disabled by default. + #[arg(long, default_value = "false")] + pub pedantic_solving: bool, + + /// Used internally to test for non-determinism in the compiler. + #[clap(long, hide = true)] + pub check_non_determinism: bool, } pub fn parse_expression_width(input: &str) -> Result { @@ -307,13 +321,13 @@ pub fn check_crate( crate_id: CrateId, options: &CompileOptions, ) -> CompilationResult<()> { - let error_on_unused_imports = true; let diagnostics = CrateDefMap::collect_defs( crate_id, context, options.debug_comptime_in_file.as_deref(), - error_on_unused_imports, + options.pedantic_solving, ); + let crate_files = context.crate_files(&crate_id); let warnings_and_errors: Vec = diagnostics .into_iter() .map(|(error, file_id)| { @@ -324,6 +338,14 @@ pub fn check_crate( // We filter out any warnings if they're going to be ignored later on to free up memory. !options.silence_warnings || diagnostic.diagnostic.kind != DiagnosticKind::Warning }) + .filter(|error| { + // Only keep warnings from the crate we are checking + if error.diagnostic.is_warning() { + crate_files.contains(&error.file_id) + } else { + true + } + }) .collect(); if has_errors(&warnings_and_errors, options.deny_warnings) { @@ -424,6 +446,11 @@ pub fn compile_contract( if options.print_acir { for contract_function in &compiled_contract.functions { + if let Some(ref name) = options.show_contract_fn { + if name != &contract_function.name { + continue; + } + } println!( "Compiled ACIR for {}::{} (unoptimized):", compiled_contract.name, contract_function.name @@ -468,7 +495,15 @@ fn compile_contract_inner( continue; } - let function = match compile_no_check(context, options, function_id, None, true) { + let mut options = options.clone(); + + if let Some(ref name_filter) = options.show_contract_fn { + let show = name == *name_filter; + options.show_ssa &= show; + options.show_ssa_pass = options.show_ssa_pass.filter(|_| show); + }; + + let function = match compile_no_check(context, &options, function_id, None, true) { Ok(function) => function, Err(new_error) => { errors.push(FileDiagnostic::from(new_error)); @@ -586,10 +621,17 @@ pub fn compile_no_check( cached_program: Option, force_compile: bool, ) -> Result { + let force_unconstrained = options.force_brillig; + let program = if options.instrument_debug { - monomorphize_debug(main_function, &mut context.def_interner, &context.debug_instrumenter)? + monomorphize_debug( + main_function, + &mut context.def_interner, + &context.debug_instrumenter, + force_unconstrained, + )? } else { - monomorphize(main_function, &mut context.def_interner)? + monomorphize(main_function, &mut context.def_interner, force_unconstrained)? }; if options.show_monomorphized { @@ -617,7 +659,7 @@ pub fn compile_no_check( let return_visibility = program.return_visibility; let ssa_evaluator_options = noirc_evaluator::ssa::SsaEvaluatorOptions { - ssa_logging: match &options.show_ssa_pass_name { + ssa_logging: match &options.show_ssa_pass { Some(string) => SsaLogging::Contains(string.clone()), None => { if options.show_ssa { @@ -628,7 +670,6 @@ pub fn compile_no_check( } }, enable_brillig_logging: options.show_brillig, - force_brillig_output: options.force_brillig, print_codegen_timings: options.benchmark_codegen, expression_width: if options.bounded_codegen { options.expression_width.unwrap_or(DEFAULT_EXPRESSION_WIDTH) diff --git a/noir/noir-repo/compiler/noirc_errors/src/reporter.rs b/noir/noir-repo/compiler/noirc_errors/src/reporter.rs index e57775d9a7f..20a765fdaa5 100644 --- a/noir/noir-repo/compiler/noirc_errors/src/reporter.rs +++ b/noir/noir-repo/compiler/noirc_errors/src/reporter.rs @@ -230,7 +230,7 @@ pub fn report<'files>( let color_choice = if std::io::stderr().is_terminal() { ColorChoice::Auto } else { ColorChoice::Never }; let writer = StandardStream::stderr(color_choice); - let config = codespan_reporting::term::Config::default(); + let config = term::Config::default(); let stack_trace = stack_trace(files, &custom_diagnostic.call_stack); let diagnostic = convert_diagnostic(custom_diagnostic, file, stack_trace, deny_warnings); diff --git a/noir/noir-repo/compiler/noirc_evaluator/Cargo.toml b/noir/noir-repo/compiler/noirc_evaluator/Cargo.toml index 72fba8aadc2..3e30fa6673d 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/Cargo.toml +++ b/noir/noir-repo/compiler/noirc_evaluator/Cargo.toml @@ -28,6 +28,8 @@ tracing.workspace = true chrono = "0.4.37" rayon.workspace = true cfg-if.workspace = true +smallvec = { version = "1.13.2", features = ["serde"] } +vec-collections = "0.4.3" [dev-dependencies] proptest.workspace = true diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/acir/acir_variable.rs b/noir/noir-repo/compiler/noirc_evaluator/src/acir/acir_variable.rs index 78188ea2b65..41e2c2dad1e 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/acir/acir_variable.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/acir/acir_variable.rs @@ -112,9 +112,6 @@ impl From for AcirType { pub(crate) struct AcirContext> { blackbox_solver: B, - /// Two-way map that links `AcirVar` to `AcirVarData`. - /// - /// The vars object is an instance of the `TwoWayMap`, which provides a bidirectional mapping between `AcirVar` and `AcirVarData`. vars: HashMap>, constant_witnesses: HashMap, @@ -544,6 +541,29 @@ impl> AcirContext { Ok(()) } + /// Constrains the `lhs` and `rhs` to be non-equal. + /// + /// This is done by asserting the existence of an inverse for the value `lhs - rhs`. + /// The constraint `(lhs - rhs) * inverse == 1` will only be satisfiable if `lhs` and `rhs` are non-equal. + pub(crate) fn assert_neq_var( + &mut self, + lhs: AcirVar, + rhs: AcirVar, + assert_message: Option>, + ) -> Result<(), RuntimeError> { + let diff_var = self.sub_var(lhs, rhs)?; + + let one = self.add_constant(F::one()); + let _ = self.inv_var(diff_var, one)?; + if let Some(payload) = assert_message { + self.acir_ir + .assertion_payloads + .insert(self.acir_ir.last_acir_opcode_location(), payload); + } + + Ok(()) + } + pub(crate) fn vars_to_expressions_or_memory( &self, values: &[AcirValue], @@ -954,6 +974,7 @@ impl> AcirContext { offset: AcirVar, bits: u32, ) -> Result<(), RuntimeError> { + #[allow(unused_qualifications)] const fn num_bits() -> usize { std::mem::size_of::() * 8 } @@ -1423,7 +1444,7 @@ impl> AcirContext { } }?; output_count = input_size + (16 - input_size % 16); - (vec![], vec![F::from(output_count as u128)]) + (vec![], vec![]) } BlackBoxFunc::RecursiveAggregation => { let proof_type_var = match inputs.pop() { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/acir/generated_acir.rs b/noir/noir-repo/compiler/noirc_evaluator/src/acir/generated_acir.rs index b6a5a817ea7..14ceac62461 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/acir/generated_acir.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/acir/generated_acir.rs @@ -479,7 +479,7 @@ impl GeneratedAcir { pub(crate) fn is_equal(&mut self, lhs: &Expression, rhs: &Expression) -> Witness { let t = lhs - rhs; - self.is_zero(&t) + self.is_zero(t) } /// Returns a `Witness` that is constrained to be: @@ -497,7 +497,7 @@ impl GeneratedAcir { /// This implies that either `y` or `t` or both is `0`. /// - If `t == 0`, then by definition `t == 0`. /// - If `y == 0`, this does not mean anything at this point in time, due to it having no - /// constraints. + /// constraints. /// /// Naively, we could apply the following constraint: `y == 1 - t`. /// This along with the previous `y * t == 0` constraint means that @@ -534,36 +534,32 @@ impl GeneratedAcir { /// By setting `z` to be `0`, we can make `y` equal to `1`. /// This is easily observed: `y = 1 - t * 0` /// Now since `y` is one, this means that `t` needs to be zero, or else `y * t == 0` will fail. - fn is_zero(&mut self, t_expr: &Expression) -> Witness { - // We're checking for equality with zero so we can negate the expression without changing the result. - // This is useful as it will sometimes allow us to simplify an expression down to a witness. - let t_witness = if let Some(witness) = t_expr.to_witness() { - witness + fn is_zero(&mut self, t_expr: Expression) -> Witness { + // We're going to be multiplying this expression by two different witnesses in a second so we want to + // ensure that this expression only contains a single witness. We can tolerate coefficients and constant terms however. + let linear_t = if t_expr.is_degree_one_univariate() { + t_expr } else { - let negated_expr = t_expr * -F::one(); - self.get_or_create_witness(&negated_expr) + Expression::::from(self.get_or_create_witness(&t_expr)) }; // Call the inversion directive, since we do not apply a constraint // the prover can choose anything here. - let z = self.brillig_inverse(t_witness.into()); + let z = self.brillig_inverse(linear_t.clone()); + let z_expr = Expression::::from(z); let y = self.next_witness_index(); + let y_expr = Expression::::from(y); // Add constraint y == 1 - tz => y + tz - 1 == 0 - let y_is_boolean_constraint = Expression { - mul_terms: vec![(F::one(), t_witness, z)], - linear_combinations: vec![(F::one(), y)], - q_c: -F::one(), - }; + let mut y_is_boolean_constraint = + (&z_expr * &linear_t).expect("multiplying two linear expressions"); + y_is_boolean_constraint.push_addition_term(F::one(), y); + let y_is_boolean_constraint = y_is_boolean_constraint - F::one(); self.assert_is_zero(y_is_boolean_constraint); // Add constraint that y * t == 0; - let ty_zero_constraint = Expression { - mul_terms: vec![(F::one(), t_witness, y)], - linear_combinations: vec![], - q_c: F::zero(), - }; + let ty_zero_constraint = (&y_expr * &linear_t).expect("multiplying two linear expressions"); self.assert_is_zero(ty_zero_constraint); y @@ -604,7 +600,7 @@ impl GeneratedAcir { ) { // Check whether we have a call to this Brillig function already exists. // This helps us optimize the Brillig metadata to only be stored once per Brillig entry point. - let inserted_func_before = self.brillig_locations.get(&brillig_function_index).is_some(); + let inserted_func_before = self.brillig_locations.contains_key(&brillig_function_index); let opcode = AcirOpcode::BrilligCall { id: brillig_function_index, inputs, outputs, predicate }; diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/acir/mod.rs b/noir/noir-repo/compiler/noirc_evaluator/src/acir/mod.rs index e7b011b6d7b..b573b7136f3 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/acir/mod.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/acir/mod.rs @@ -1,7 +1,6 @@ //! This file holds the pass to convert from Noir's SSA IR to ACIR. use fxhash::FxHashMap as HashMap; -use im::Vector; use std::collections::{BTreeMap, HashSet}; use std::fmt::Debug; @@ -248,7 +247,7 @@ impl Debug for AcirDynamicArray { #[derive(Debug, Clone)] pub(crate) enum AcirValue { Var(AcirVar, AcirType), - Array(Vector), + Array(im::Vector), DynamicArray(AcirDynamicArray), } @@ -724,6 +723,47 @@ impl<'a> Context<'a> { self.acir_context.assert_eq_var(lhs, rhs, assert_payload)?; } + Instruction::ConstrainNotEqual(lhs, rhs, assert_message) => { + let lhs = self.convert_numeric_value(*lhs, dfg)?; + let rhs = self.convert_numeric_value(*rhs, dfg)?; + + let assert_payload = if let Some(error) = assert_message { + match error { + ConstrainError::StaticString(string) => Some( + self.acir_context.generate_assertion_message_payload(string.clone()), + ), + ConstrainError::Dynamic(error_selector, is_string_type, values) => { + if let Some(constant_string) = try_to_extract_string_from_error_payload( + *is_string_type, + values, + dfg, + ) { + Some( + self.acir_context + .generate_assertion_message_payload(constant_string), + ) + } else { + let acir_vars: Vec<_> = values + .iter() + .map(|value| self.convert_value(*value, dfg)) + .collect(); + + let expressions_or_memory = + self.acir_context.vars_to_expressions_or_memory(&acir_vars)?; + + Some(AssertionPayload { + error_selector: error_selector.as_u64(), + payload: expressions_or_memory, + }) + } + } + } + } else { + None + }; + + self.acir_context.assert_neq_var(lhs, rhs, assert_payload)?; + } Instruction::Cast(value_id, _) => { let acir_var = self.convert_numeric_value(*value_id, dfg)?; self.define_result_var(dfg, instruction_id, acir_var); @@ -770,7 +810,8 @@ impl<'a> Context<'a> { unreachable!("Expected all load instructions to be removed before acir_gen") } Instruction::IncrementRc { .. } | Instruction::DecrementRc { .. } => { - // Do nothing. Only Brillig needs to worry about reference counted arrays + // Only Brillig needs to worry about reference counted arrays + unreachable!("Expected all Rc instructions to be removed before acir_gen") } Instruction::RangeCheck { value, max_bit_size, assert_message } => { let acir_var = self.convert_numeric_value(*value, dfg)?; @@ -789,6 +830,7 @@ impl<'a> Context<'a> { let result = dfg.instruction_results(instruction_id)[0]; self.ssa_values.insert(result, value); } + Instruction::Noop => (), } self.acir_context.set_call_stack(CallStack::new()); @@ -1118,7 +1160,7 @@ impl<'a> Context<'a> { &mut self, instruction: InstructionId, dfg: &DataFlowGraph, - array: Vector, + array: im::Vector, index: FieldElement, store_value: Option, ) -> Result { @@ -1303,7 +1345,7 @@ impl<'a> Context<'a> { match typ { Type::Numeric(_) => self.array_get_value(&Type::field(), call_data_block, offset), Type::Array(arc, len) => { - let mut result = Vector::new(); + let mut result = im::Vector::new(); for _i in 0..*len { for sub_type in arc.iter() { let element = self.get_from_call_data(offset, call_data_block, sub_type)?; @@ -1394,7 +1436,7 @@ impl<'a> Context<'a> { Ok(AcirValue::Var(read, typ)) } Type::Array(element_types, len) => { - let mut values = Vector::new(); + let mut values = im::Vector::new(); for _ in 0..len { for typ in element_types.as_ref() { values.push_back(self.array_get_value(typ, block_id, var_index)?); @@ -1682,7 +1724,7 @@ impl<'a> Context<'a> { let read = self.acir_context.read_from_memory(source, &index_var)?; Ok::(AcirValue::Var(read, AcirType::field())) })?; - let array: Vector = init_values.into(); + let array: im::Vector = init_values.into(); self.initialize_array(destination, array_len, Some(AcirValue::Array(array)))?; Ok(()) } @@ -1891,6 +1933,9 @@ impl<'a> Context<'a> { Value::Instruction { .. } | Value::Param { .. } => { unreachable!("ICE: Should have been in cache {value_id} {value:?}") } + Value::Global(_) => { + unreachable!("ICE: All globals should have been inlined"); + } }; self.ssa_values.insert(value_id, acir_value.clone()); acir_value @@ -1948,9 +1993,9 @@ impl<'a> Context<'a> { let bit_count = binary_type.bit_size::(); let num_type = binary_type.to_numeric_type(); let result = match binary.operator { - BinaryOp::Add => self.acir_context.add_var(lhs, rhs), - BinaryOp::Sub => self.acir_context.sub_var(lhs, rhs), - BinaryOp::Mul => self.acir_context.mul_var(lhs, rhs), + BinaryOp::Add { .. } => self.acir_context.add_var(lhs, rhs), + BinaryOp::Sub { .. } => self.acir_context.sub_var(lhs, rhs), + BinaryOp::Mul { .. } => self.acir_context.mul_var(lhs, rhs), BinaryOp::Div => self.acir_context.div_var( lhs, rhs, @@ -1983,14 +2028,7 @@ impl<'a> Context<'a> { if let NumericType::Unsigned { bit_size } = &num_type { // Check for integer overflow - self.check_unsigned_overflow( - result, - *bit_size, - binary.lhs, - binary.rhs, - dfg, - binary.operator, - )?; + self.check_unsigned_overflow(result, *bit_size, binary, dfg)?; } Ok(result) @@ -2001,47 +2039,18 @@ impl<'a> Context<'a> { &mut self, result: AcirVar, bit_size: u32, - lhs: ValueId, - rhs: ValueId, + binary: &Binary, dfg: &DataFlowGraph, - op: BinaryOp, ) -> Result<(), RuntimeError> { - // We try to optimize away operations that are guaranteed not to overflow - let max_lhs_bits = dfg.get_value_max_num_bits(lhs); - let max_rhs_bits = dfg.get_value_max_num_bits(rhs); - - let msg = match op { - BinaryOp::Add => { - if std::cmp::max(max_lhs_bits, max_rhs_bits) < bit_size { - // `lhs` and `rhs` have both been casted up from smaller types and so cannot overflow. - return Ok(()); - } - "attempt to add with overflow".to_string() - } - BinaryOp::Sub => { - if dfg.is_constant(lhs) && max_lhs_bits > max_rhs_bits { - // `lhs` is a fixed constant and `rhs` is restricted such that `lhs - rhs > 0` - // Note strict inequality as `rhs > lhs` while `max_lhs_bits == max_rhs_bits` is possible. - return Ok(()); - } - "attempt to subtract with overflow".to_string() - } - BinaryOp::Mul => { - if bit_size == 1 || max_lhs_bits + max_rhs_bits <= bit_size { - // Either performing boolean multiplication (which cannot overflow), - // or `lhs` and `rhs` have both been casted up from smaller types and so cannot overflow. - return Ok(()); - } - "attempt to multiply with overflow".to_string() - } - _ => return Ok(()), + let Some(msg) = binary.check_unsigned_overflow_msg(dfg, bit_size) else { + return Ok(()); }; let with_pred = self.acir_context.mul_var(result, self.current_side_effects_enabled_var)?; self.acir_context.range_constrain_var( with_pred, &NumericType::Unsigned { bit_size }, - Some(msg), + Some(msg.to_string()), )?; Ok(()) } @@ -2053,8 +2062,9 @@ impl<'a> Context<'a> { /// /// There are some edge cases to consider: /// - Constants are not explicitly type casted, so we need to check for this and - /// return the type of the other operand, if we have a constant. + /// return the type of the other operand, if we have a constant. /// - 0 is not seen as `Field 0` but instead as `Unit 0` + /// /// TODO: The latter seems like a bug, if we cannot differentiate between a function returning /// TODO nothing and a 0. /// @@ -2104,7 +2114,7 @@ impl<'a> Context<'a> { Value::Instruction { instruction, .. } => { if matches!( &dfg[*instruction], - Instruction::Binary(Binary { operator: BinaryOp::Sub, .. }) + Instruction::Binary(Binary { operator: BinaryOp::Sub { .. }, .. }) ) { // Subtractions must first have the integer modulus added before truncation can be // applied. This is done in order to prevent underflow. @@ -2203,7 +2213,7 @@ impl<'a> Context<'a> { let Type::Array(result_type, array_length) = dfg.type_of_value(result_ids[0]) else { - unreachable!("ICE: ToRadix result must be an array"); + unreachable!("ICE: ToBits result must be an array"); }; self.acir_context @@ -2273,7 +2283,7 @@ impl<'a> Context<'a> { let slice = self.convert_value(slice_contents, dfg); let mut new_elem_size = Self::flattened_value_size(&slice); - let mut new_slice = Vector::new(); + let mut new_slice = im::Vector::new(); self.slice_intrinsic_input(&mut new_slice, slice)?; let elements_to_push = &arguments[2..]; @@ -2344,7 +2354,7 @@ impl<'a> Context<'a> { let one = self.acir_context.add_constant(FieldElement::one()); let new_slice_length = self.acir_context.add_var(slice_length, one)?; - let mut new_slice = Vector::new(); + let mut new_slice = im::Vector::new(); self.slice_intrinsic_input(&mut new_slice, slice)?; let elements_to_push = &arguments[2..]; @@ -2418,7 +2428,7 @@ impl<'a> Context<'a> { } let slice = self.convert_value(slice_contents, dfg); - let mut new_slice = Vector::new(); + let mut new_slice = im::Vector::new(); self.slice_intrinsic_input(&mut new_slice, slice)?; let mut results = vec![ @@ -2444,7 +2454,7 @@ impl<'a> Context<'a> { let slice = self.convert_value(slice_contents, dfg); - let mut new_slice = Vector::new(); + let mut new_slice = im::Vector::new(); self.slice_intrinsic_input(&mut new_slice, slice)?; let element_size = slice_typ.element_size(); @@ -2631,7 +2641,7 @@ impl<'a> Context<'a> { let slice_size = Self::flattened_value_size(&slice); - let mut new_slice = Vector::new(); + let mut new_slice = im::Vector::new(); self.slice_intrinsic_input(&mut new_slice, slice)?; // Compiler sanity check @@ -2760,8 +2770,6 @@ impl<'a> Context<'a> { unreachable!("Expected static_assert to be removed by this point") } Intrinsic::StrAsBytes => unreachable!("Expected as_bytes to be removed by this point"), - Intrinsic::FromField => unreachable!("Expected from_field to be removed by this point"), - Intrinsic::AsField => unreachable!("Expected as_field to be removed by this point"), Intrinsic::IsUnconstrained => { unreachable!("Expected is_unconstrained to be removed by this point") } @@ -2783,7 +2791,7 @@ impl<'a> Context<'a> { fn slice_intrinsic_input( &mut self, - old_slice: &mut Vector, + old_slice: &mut im::Vector, input: AcirValue, ) -> Result<(), RuntimeError> { match input { @@ -2888,8 +2896,9 @@ mod test { use acvm::{ acir::{ circuit::{ - brillig::BrilligFunctionId, opcodes::AcirFunctionId, ExpressionWidth, Opcode, - OpcodeLocation, + brillig::BrilligFunctionId, + opcodes::{AcirFunctionId, BlackBoxFuncCall}, + ExpressionWidth, Opcode, OpcodeLocation, }, native_types::Witness, }, @@ -2900,12 +2909,11 @@ mod test { use std::collections::BTreeMap; use crate::{ - acir::BrilligStdlibFunc, + acir::{BrilligStdlibFunc, Function}, brillig::Brillig, ssa::{ function_builder::FunctionBuilder, ir::{ - call_stack::CallStack, function::FunctionId, instruction::BinaryOp, map::Id, @@ -2914,6 +2922,8 @@ mod test { }, }; + use super::Ssa; + fn build_basic_foo_with_return( builder: &mut FunctionBuilder, foo_id: FunctionId, @@ -2932,8 +2942,7 @@ mod test { builder.new_function("foo".into(), foo_id, inline_type); } // Set a call stack for testing whether `brillig_locations` in the `GeneratedAcir` was accurately set. - let mut stack = CallStack::unit(Location::dummy()); - stack.push_back(Location::dummy()); + let stack = vec![Location::dummy(), Location::dummy()]; let call_stack = builder.current_function.dfg.call_stack_data.get_or_insert_locations(stack); builder.set_call_stack(call_stack); @@ -3194,7 +3203,11 @@ mod test { let func_with_nested_call_v1 = builder.add_parameter(Type::field()); let two = builder.field_constant(2u128); - let v0_plus_two = builder.insert_binary(func_with_nested_call_v0, BinaryOp::Add, two); + let v0_plus_two = builder.insert_binary( + func_with_nested_call_v0, + BinaryOp::Add { unchecked: false }, + two, + ); let foo_id = Id::test_new(2); let foo_call = builder.import_function(foo_id); @@ -3328,7 +3341,8 @@ mod test { build_basic_foo_with_return(&mut builder, foo_id, true, InlineType::default()); build_basic_foo_with_return(&mut builder, bar_id, true, InlineType::default()); - let ssa = builder.finish(); + let mut ssa = builder.finish(); + ssa.globals = Function::new("globals".to_owned(), ssa.main_id); let brillig = ssa.to_brillig(false); let (acir_functions, brillig_functions, _, _) = ssa @@ -3358,8 +3372,8 @@ mod test { // We have two normal Brillig functions that was called multiple times. // We should have a single locations map for each function's debug metadata. assert_eq!(main_acir.brillig_locations.len(), 2); - assert!(main_acir.brillig_locations.get(&BrilligFunctionId(0)).is_some()); - assert!(main_acir.brillig_locations.get(&BrilligFunctionId(1)).is_some()); + assert!(main_acir.brillig_locations.contains_key(&BrilligFunctionId(0))); + assert!(main_acir.brillig_locations.contains_key(&BrilligFunctionId(1))); } // Test that given multiple primitive operations that are represented by Brillig directives (e.g. invert/quotient), @@ -3466,7 +3480,8 @@ mod test { build_basic_foo_with_return(&mut builder, foo_id, true, InlineType::default()); - let ssa = builder.finish(); + let mut ssa = builder.finish(); + ssa.globals = Function::new("globals".to_owned(), ssa.main_id); // We need to generate Brillig artifacts for the regular Brillig function and pass them to the ACIR generation pass. let brillig = ssa.to_brillig(false); println!("{}", ssa); @@ -3494,7 +3509,7 @@ mod test { // We have one normal Brillig functions that was called twice. // We should have a single locations map for each function's debug metadata. assert_eq!(main_acir.brillig_locations.len(), 1); - assert!(main_acir.brillig_locations.get(&BrilligFunctionId(0)).is_some()); + assert!(main_acir.brillig_locations.contains_key(&BrilligFunctionId(0))); } // Test that given both normal Brillig calls, Brillig stdlib calls, and non-inlined ACIR calls, that we accurately generate ACIR. @@ -3555,7 +3570,8 @@ mod test { // Build an ACIR function which has the same logic as the Brillig function above build_basic_foo_with_return(&mut builder, bar_id, false, InlineType::Fold); - let ssa = builder.finish(); + let mut ssa = builder.finish(); + ssa.globals = Function::new("globals".to_owned(), ssa.main_id); // We need to generate Brillig artifacts for the regular Brillig function and pass them to the ACIR generation pass. let brillig = ssa.to_brillig(false); println!("{}", ssa); @@ -3587,7 +3603,7 @@ mod test { ); assert_eq!(main_acir.brillig_locations.len(), 1); - assert!(main_acir.brillig_locations.get(&BrilligFunctionId(0)).is_some()); + assert!(main_acir.brillig_locations.contains_key(&BrilligFunctionId(0))); let foo_acir = &acir_functions[1]; let foo_opcodes = foo_acir.opcodes(); @@ -3661,4 +3677,36 @@ mod test { "Should have {expected_num_normal_calls} BrilligCall opcodes to normal Brillig functions but got {num_normal_brillig_calls}" ); } + + #[test] + fn multiply_with_bool_should_not_emit_range_check() { + let src = " + acir(inline) fn main f0 { + b0(v0: bool, v1: u32): + enable_side_effects v0 + v2 = cast v0 as u32 + v3 = mul v2, v1 + return v3 + } + "; + let ssa = Ssa::from_str(src).unwrap(); + let brillig = ssa.to_brillig(false); + + let (mut acir_functions, _brillig_functions, _, _) = ssa + .into_acir(&brillig, ExpressionWidth::default()) + .expect("Should compile manually written SSA into ACIR"); + + assert_eq!(acir_functions.len(), 1); + + let opcodes = acir_functions[0].take_opcodes(); + + for opcode in opcodes { + if let Opcode::BlackBoxFuncCall(BlackBoxFuncCall::RANGE { input }) = opcode { + assert!( + input.to_witness().0 <= 1, + "only input witnesses should have range checks: {opcode:?}" + ); + } + } + } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen.rs index 5a81c79ae0d..b51a3445a1b 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen.rs @@ -2,11 +2,13 @@ pub(crate) mod brillig_black_box; pub(crate) mod brillig_block; pub(crate) mod brillig_block_variables; pub(crate) mod brillig_fn; +pub(crate) mod brillig_globals; pub(crate) mod brillig_slice_ops; mod constant_allocation; mod variable_liveness; use acvm::FieldElement; +use fxhash::FxHashMap as HashMap; use self::{brillig_block::BrilligBlock, brillig_fn::FunctionContext}; use super::{ @@ -14,7 +16,7 @@ use super::{ artifact::{BrilligArtifact, BrilligParameter, GeneratedBrillig, Label}, BrilligContext, }, - Brillig, + Brillig, BrilligVariable, ValueId, }; use crate::{ errors::InternalError, @@ -25,6 +27,7 @@ use crate::{ pub(crate) fn convert_ssa_function( func: &Function, enable_debug_trace: bool, + globals: &HashMap, ) -> BrilligArtifact { let mut brillig_context = BrilligContext::new(enable_debug_trace); @@ -35,7 +38,13 @@ pub(crate) fn convert_ssa_function( brillig_context.call_check_max_stack_depth_procedure(); for block in function_context.blocks.clone() { - BrilligBlock::compile(&mut function_context, &mut brillig_context, block, &func.dfg); + BrilligBlock::compile( + &mut function_context, + &mut brillig_context, + block, + &func.dfg, + globals, + ); } let mut artifact = brillig_context.artifact(); @@ -53,6 +62,7 @@ pub(crate) fn gen_brillig_for( arguments, FunctionContext::return_values(func), func.id(), + true, ); entry_point.name = func.name().to_string(); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_black_box.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_black_box.rs index 2ddcea26570..1fc39b58223 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_black_box.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_black_box.rs @@ -355,15 +355,14 @@ pub(crate) fn convert_black_box_call { if let ( [inputs, BrilligVariable::BrilligArray(iv), BrilligVariable::BrilligArray(key)], - [BrilligVariable::SingleAddr(out_len), BrilligVariable::BrilligVector(outputs)], + [outputs], ) = (function_arguments, function_results) { let inputs = convert_array_or_vector(brillig_context, *inputs, bb_func); let iv = brillig_context.codegen_brillig_array_to_heap_array(*iv); let key = brillig_context.codegen_brillig_array_to_heap_array(*key); - let outputs_vector = - brillig_context.codegen_brillig_vector_to_heap_vector(*outputs); + let outputs_vector = convert_array_or_vector(brillig_context, *outputs, bb_func); brillig_context.black_box_op_instruction(BlackBoxOp::AES128Encrypt { inputs, @@ -372,11 +371,6 @@ pub(crate) fn convert_black_box_call { +pub(crate) struct BrilligBlock<'block, Registers: RegisterAllocator> { pub(crate) function_context: &'block mut FunctionContext, /// The basic block that is being converted pub(crate) block_id: BasicBlockId, /// Context for creating brillig opcodes - pub(crate) brillig_context: &'block mut BrilligContext, + pub(crate) brillig_context: &'block mut BrilligContext, /// Tracks the available variable during the codegen of the block pub(crate) variables: BlockVariables, /// For each instruction, the set of values that are not used anymore after it. pub(crate) last_uses: HashMap>, + + pub(crate) globals: &'block HashMap, + + pub(crate) building_globals: bool, } -impl<'block> BrilligBlock<'block> { +impl<'block, Registers: RegisterAllocator> BrilligBlock<'block, Registers> { /// Converts an SSA Basic block into a sequence of Brillig opcodes pub(crate) fn compile( function_context: &'block mut FunctionContext, - brillig_context: &'block mut BrilligContext, + brillig_context: &'block mut BrilligContext, block_id: BasicBlockId, dfg: &DataFlowGraph, + globals: &'block HashMap, ) { let live_in = function_context.liveness.get_live_in(&block_id); - let variables = BlockVariables::new(live_in.clone()); + + let mut live_in_no_globals = HashSet::default(); + for value in live_in { + if !dfg.is_global(*value) { + live_in_no_globals.insert(*value); + } + } + + let variables = BlockVariables::new(live_in_no_globals); brillig_context.set_allocated_registers( variables @@ -64,12 +77,44 @@ impl<'block> BrilligBlock<'block> { ); let last_uses = function_context.liveness.get_last_uses(&block_id).clone(); - let mut brillig_block = - BrilligBlock { function_context, block_id, brillig_context, variables, last_uses }; + let mut brillig_block = BrilligBlock { + function_context, + block_id, + brillig_context, + variables, + last_uses, + globals, + building_globals: false, + }; brillig_block.convert_block(dfg); } + pub(crate) fn compile_globals( + &mut self, + globals: &DataFlowGraph, + used_globals: &HashSet, + ) { + for (id, value) in globals.values_iter() { + if !used_globals.contains(&id) { + continue; + } + match value { + Value::NumericConstant { .. } => { + self.convert_ssa_value(id, globals); + } + Value::Instruction { instruction, .. } => { + self.convert_ssa_instruction(*instruction, globals); + } + _ => { + panic!( + "Expected either an instruction or a numeric constant for a global value" + ) + } + } + } + } + fn convert_block(&mut self, dfg: &DataFlowGraph) { // Add a label for this block let block_label = self.create_block_label_for_current_function(self.block_id); @@ -199,7 +244,11 @@ impl<'block> BrilligBlock<'block> { } /// Converts an SSA instruction into a sequence of Brillig opcodes. - fn convert_ssa_instruction(&mut self, instruction_id: InstructionId, dfg: &DataFlowGraph) { + pub(crate) fn convert_ssa_instruction( + &mut self, + instruction_id: InstructionId, + dfg: &DataFlowGraph, + ) { let instruction = &dfg[instruction_id]; self.brillig_context.set_call_stack(dfg.get_instruction_call_stack(instruction_id)); @@ -279,6 +328,10 @@ impl<'block> BrilligBlock<'block> { self.brillig_context.deallocate_single_addr(condition); } } + Instruction::ConstrainNotEqual(..) => { + unreachable!("only implemented in ACIR") + } + Instruction::Allocate => { let result_value = dfg.instruction_results(instruction_id)[0]; let pointer = self.variables.define_single_addr_variable( @@ -635,9 +688,7 @@ impl<'block> BrilligBlock<'block> { let array = array.extract_register(); self.brillig_context.load_instruction(destination, array); } - Intrinsic::FromField - | Intrinsic::AsField - | Intrinsic::IsUnconstrained + Intrinsic::IsUnconstrained | Intrinsic::DerivePedersenGenerators | Intrinsic::ApplyRangeConstraint | Intrinsic::StrAsBytes @@ -648,7 +699,10 @@ impl<'block> BrilligBlock<'block> { } } } - Value::Instruction { .. } | Value::Param { .. } | Value::NumericConstant { .. } => { + Value::Instruction { .. } + | Value::Param { .. } + | Value::NumericConstant { .. } + | Value::Global(_) => { unreachable!("unsupported function call type {:?}", dfg[*func]) } }, @@ -797,7 +851,7 @@ impl<'block> BrilligBlock<'block> { self.brillig_context.deallocate_register(rc_register); } Instruction::EnableSideEffectsIf { .. } => { - todo!("enable_side_effects not supported by brillig") + unreachable!("enable_side_effects not supported by brillig") } Instruction::IfElse { .. } => { unreachable!("IfElse instructions should not be possible in brillig") @@ -839,20 +893,27 @@ impl<'block> BrilligBlock<'block> { self.brillig_context.deallocate_register(items_pointer); } } + Instruction::Noop => (), }; - let dead_variables = self - .last_uses - .get(&instruction_id) - .expect("Last uses for instruction should have been computed"); - - for dead_variable in dead_variables { - self.variables.remove_variable( - dead_variable, - self.function_context, - self.brillig_context, - ); + if !self.building_globals { + let dead_variables = self + .last_uses + .get(&instruction_id) + .expect("Last uses for instruction should have been computed"); + + for dead_variable in dead_variables { + // Globals are reserved throughout the entirety of the program + if !dfg.is_global(*dead_variable) { + self.variables.remove_variable( + dead_variable, + self.function_context, + self.brillig_context, + ); + } + } } + self.brillig_context.set_call_stack(CallStack::new()); } @@ -1319,9 +1380,9 @@ impl<'block> BrilligBlock<'block> { BrilligBinaryOp::Modulo } } - BinaryOp::Add => BrilligBinaryOp::Add, - BinaryOp::Sub => BrilligBinaryOp::Sub, - BinaryOp::Mul => BrilligBinaryOp::Mul, + BinaryOp::Add { .. } => BrilligBinaryOp::Add, + BinaryOp::Sub { .. } => BrilligBinaryOp::Sub, + BinaryOp::Mul { .. } => BrilligBinaryOp::Mul, BinaryOp::Eq => BrilligBinaryOp::Equals, BinaryOp::Lt => { if is_signed { @@ -1479,88 +1540,70 @@ impl<'block> BrilligBlock<'block> { is_signed: bool, ) { let bit_size = left.bit_size; - let max_lhs_bits = dfg.get_value_max_num_bits(binary.lhs); - let max_rhs_bits = dfg.get_value_max_num_bits(binary.rhs); - if bit_size == FieldElement::max_num_bits() { + if bit_size == FieldElement::max_num_bits() || is_signed { return; } - match (binary_operation, is_signed) { - (BrilligBinaryOp::Add, false) => { - if std::cmp::max(max_lhs_bits, max_rhs_bits) < bit_size { - // `left` and `right` have both been casted up from smaller types and so cannot overflow. - return; - } - - let condition = - SingleAddrVariable::new(self.brillig_context.allocate_register(), 1); - // Check that lhs <= result - self.brillig_context.binary_instruction( - left, - result, - condition, - BrilligBinaryOp::LessThanEquals, - ); - self.brillig_context - .codegen_constrain(condition, Some("attempt to add with overflow".to_string())); - self.brillig_context.deallocate_single_addr(condition); - } - (BrilligBinaryOp::Sub, false) => { - if dfg.is_constant(binary.lhs) && max_lhs_bits > max_rhs_bits { - // `left` is a fixed constant and `right` is restricted such that `left - right > 0` - // Note strict inequality as `right > left` while `max_lhs_bits == max_rhs_bits` is possible. - return; - } - - let condition = - SingleAddrVariable::new(self.brillig_context.allocate_register(), 1); - // Check that rhs <= lhs - self.brillig_context.binary_instruction( - right, - left, - condition, - BrilligBinaryOp::LessThanEquals, - ); - self.brillig_context.codegen_constrain( - condition, - Some("attempt to subtract with overflow".to_string()), - ); - self.brillig_context.deallocate_single_addr(condition); - } - (BrilligBinaryOp::Mul, false) => { - if bit_size == 1 || max_lhs_bits + max_rhs_bits <= bit_size { - // Either performing boolean multiplication (which cannot overflow), - // or `left` and `right` have both been casted up from smaller types and so cannot overflow. - return; + if let Some(msg) = binary.check_unsigned_overflow_msg(dfg, bit_size) { + match binary_operation { + BrilligBinaryOp::Add => { + let condition = + SingleAddrVariable::new(self.brillig_context.allocate_register(), 1); + // Check that lhs <= result + self.brillig_context.binary_instruction( + left, + result, + condition, + BrilligBinaryOp::LessThanEquals, + ); + self.brillig_context.codegen_constrain(condition, Some(msg.to_string())); + self.brillig_context.deallocate_single_addr(condition); } - - let is_right_zero = - SingleAddrVariable::new(self.brillig_context.allocate_register(), 1); - let zero = self.brillig_context.make_constant_instruction(0_usize.into(), bit_size); - self.brillig_context.binary_instruction( - zero, - right, - is_right_zero, - BrilligBinaryOp::Equals, - ); - self.brillig_context.codegen_if_not(is_right_zero.address, |ctx| { - let condition = SingleAddrVariable::new(ctx.allocate_register(), 1); - let division = SingleAddrVariable::new(ctx.allocate_register(), bit_size); - // Check that result / rhs == lhs - ctx.binary_instruction(result, right, division, BrilligBinaryOp::UnsignedDiv); - ctx.binary_instruction(division, left, condition, BrilligBinaryOp::Equals); - ctx.codegen_constrain( + BrilligBinaryOp::Sub => { + let condition = + SingleAddrVariable::new(self.brillig_context.allocate_register(), 1); + // Check that rhs <= lhs + self.brillig_context.binary_instruction( + right, + left, condition, - Some("attempt to multiply with overflow".to_string()), + BrilligBinaryOp::LessThanEquals, ); - ctx.deallocate_single_addr(condition); - ctx.deallocate_single_addr(division); - }); - self.brillig_context.deallocate_single_addr(is_right_zero); - self.brillig_context.deallocate_single_addr(zero); + self.brillig_context.codegen_constrain(condition, Some(msg.to_string())); + self.brillig_context.deallocate_single_addr(condition); + } + BrilligBinaryOp::Mul => { + let is_right_zero = + SingleAddrVariable::new(self.brillig_context.allocate_register(), 1); + let zero = + self.brillig_context.make_constant_instruction(0_usize.into(), bit_size); + self.brillig_context.binary_instruction( + zero, + right, + is_right_zero, + BrilligBinaryOp::Equals, + ); + self.brillig_context.codegen_if_not(is_right_zero.address, |ctx| { + let condition = SingleAddrVariable::new(ctx.allocate_register(), 1); + let division = SingleAddrVariable::new(ctx.allocate_register(), bit_size); + // Check that result / rhs == lhs + ctx.binary_instruction( + result, + right, + division, + BrilligBinaryOp::UnsignedDiv, + ); + ctx.binary_instruction(division, left, condition, BrilligBinaryOp::Equals); + ctx.codegen_constrain(condition, Some(msg.to_string())); + ctx.deallocate_single_addr(condition); + ctx.deallocate_single_addr(division); + }); + self.brillig_context.deallocate_single_addr(is_right_zero); + self.brillig_context.deallocate_single_addr(zero); + } + _ => {} } - _ => {} } } @@ -1571,22 +1614,38 @@ impl<'block> BrilligBlock<'block> { } /// Converts an SSA `ValueId` into a `RegisterOrMemory`. Initializes if necessary. - fn convert_ssa_value(&mut self, value_id: ValueId, dfg: &DataFlowGraph) -> BrilligVariable { + pub(crate) fn convert_ssa_value( + &mut self, + value_id: ValueId, + dfg: &DataFlowGraph, + ) -> BrilligVariable { let value_id = dfg.resolve(value_id); let value = &dfg[value_id]; match value { + Value::Global(_) => { + unreachable!("Expected global value to be resolve to its inner value"); + } Value::Param { .. } | Value::Instruction { .. } => { // All block parameters and instruction results should have already been // converted to registers so we fetch from the cache. - - self.variables.get_allocation(self.function_context, value_id, dfg) + if dfg.is_global(value_id) { + *self.globals.get(&value_id).unwrap_or_else(|| { + panic!("ICE: Global value not found in cache {value_id}") + }) + } else { + self.variables.get_allocation(self.function_context, value_id, dfg) + } } Value::NumericConstant { constant, .. } => { // Constants might have been converted previously or not, so we get or create and // (re)initialize the value inside. if self.variables.is_allocated(&value_id) { self.variables.get_allocation(self.function_context, value_id, dfg) + } else if dfg.is_global(value_id) { + *self.globals.get(&value_id).unwrap_or_else(|| { + panic!("ICE: Global value not found in cache {value_id}") + }) } else { let new_variable = self.variables.define_variable( self.function_context, diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_block_variables.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_block_variables.rs index bf0a1bc7347..4cf8e921483 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_block_variables.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_block_variables.rs @@ -7,7 +7,7 @@ use crate::{ get_bit_size_from_ssa_type, BrilligArray, BrilligVariable, BrilligVector, SingleAddrVariable, }, - registers::{RegisterAllocator, Stack}, + registers::RegisterAllocator, BrilligContext, }, ssa::ir::{ @@ -48,10 +48,10 @@ impl BlockVariables { } /// For a given SSA value id, define the variable and return the corresponding cached allocation. - pub(crate) fn define_variable( + pub(crate) fn define_variable( &mut self, function_context: &mut FunctionContext, - brillig_context: &mut BrilligContext, + brillig_context: &mut BrilligContext, value_id: ValueId, dfg: &DataFlowGraph, ) -> BrilligVariable { @@ -68,10 +68,10 @@ impl BlockVariables { } /// Defines a variable that fits in a single register and returns the allocated register. - pub(crate) fn define_single_addr_variable( + pub(crate) fn define_single_addr_variable( &mut self, function_context: &mut FunctionContext, - brillig_context: &mut BrilligContext, + brillig_context: &mut BrilligContext, value: ValueId, dfg: &DataFlowGraph, ) -> SingleAddrVariable { @@ -80,11 +80,11 @@ impl BlockVariables { } /// Removes a variable so it's not used anymore within this block. - pub(crate) fn remove_variable( + pub(crate) fn remove_variable( &mut self, value_id: &ValueId, function_context: &mut FunctionContext, - brillig_context: &mut BrilligContext, + brillig_context: &mut BrilligContext, ) { assert!(self.available_variables.remove(value_id), "ICE: Variable is not available"); let variable = function_context @@ -133,6 +133,14 @@ pub(crate) fn allocate_value( ) -> BrilligVariable { let typ = dfg.type_of_value(value_id); + allocate_value_with_type(brillig_context, typ) +} + +/// For a given value_id, allocates the necessary registers to hold it. +pub(crate) fn allocate_value_with_type( + brillig_context: &mut BrilligContext, + typ: Type, +) -> BrilligVariable { match typ { Type::Numeric(_) | Type::Reference(_) | Type::Function => { BrilligVariable::SingleAddr(SingleAddrVariable { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_globals.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_globals.rs new file mode 100644 index 00000000000..99c8ee0fded --- /dev/null +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_globals.rs @@ -0,0 +1,39 @@ +use acvm::FieldElement; +use fxhash::{FxHashMap as HashMap, FxHashSet as HashSet}; + +use super::{ + BrilligArtifact, BrilligBlock, BrilligVariable, Function, FunctionContext, Label, ValueId, +}; +use crate::brillig::{brillig_ir::BrilligContext, DataFlowGraph}; + +pub(crate) fn convert_ssa_globals( + enable_debug_trace: bool, + globals: &Function, + used_globals: &HashSet, +) -> (BrilligArtifact, HashMap) { + let mut brillig_context = BrilligContext::new_for_global_init(enable_debug_trace); + // The global space does not have globals itself + let empty_globals = HashMap::default(); + // We can use any ID here as this context is only going to be used for globals which does not differentiate + // by functions and blocks. The only Label that should be used in the globals context is `Label::globals_init()` + let mut function_context = FunctionContext::new(globals); + brillig_context.enter_context(Label::globals_init()); + + let block_id = DataFlowGraph::default().make_block(); + let mut brillig_block = BrilligBlock { + function_context: &mut function_context, + block_id, + brillig_context: &mut brillig_context, + variables: Default::default(), + last_uses: HashMap::default(), + globals: &empty_globals, + building_globals: true, + }; + + brillig_block.compile_globals(&globals.dfg, used_globals); + + brillig_context.return_instruction(); + + let artifact = brillig_context.artifact(); + (artifact, function_context.ssa_value_allocations) +} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_slice_ops.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_slice_ops.rs index 26c7151bf07..1ec2d165b12 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_slice_ops.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/brillig_slice_ops.rs @@ -2,12 +2,13 @@ use acvm::acir::brillig::MemoryAddress; use crate::brillig::brillig_ir::{ brillig_variable::{BrilligVariable, BrilligVector, SingleAddrVariable}, + registers::RegisterAllocator, BrilligBinaryOp, }; use super::brillig_block::BrilligBlock; -impl<'block> BrilligBlock<'block> { +impl<'block, Registers: RegisterAllocator> BrilligBlock<'block, Registers> { fn write_variables(&mut self, write_pointer: MemoryAddress, variables: &[BrilligVariable]) { for (index, variable) in variables.iter().enumerate() { self.brillig_context.store_instruction(write_pointer, variable.extract_register()); @@ -159,6 +160,7 @@ mod tests { use std::vec; use acvm::FieldElement; + use fxhash::FxHashMap as HashMap; use noirc_frontend::monomorphization::ast::InlineType; use crate::brillig::brillig_gen::brillig_block::BrilligBlock; @@ -173,6 +175,7 @@ mod tests { create_and_run_vm, create_context, create_entry_point_bytecode, }; use crate::brillig::brillig_ir::{BrilligContext, BRILLIG_MEMORY_ADDRESSING_BIT_SIZE}; + use crate::brillig::ValueId; use crate::ssa::function_builder::FunctionBuilder; use crate::ssa::ir::function::RuntimeType; use crate::ssa::ir::map::Id; @@ -193,7 +196,8 @@ mod tests { fn create_brillig_block<'a>( function_context: &'a mut FunctionContext, brillig_context: &'a mut BrilligContext, - ) -> BrilligBlock<'a> { + globals: &'a HashMap, + ) -> BrilligBlock<'a, Stack> { let variables = BlockVariables::default(); BrilligBlock { function_context, @@ -201,6 +205,8 @@ mod tests { brillig_context, variables, last_uses: Default::default(), + globals, + building_globals: false, } } @@ -242,7 +248,9 @@ mod tests { // Allocate the results let target_vector = BrilligVector { pointer: context.allocate_register() }; - let mut block = create_brillig_block(&mut function_context, &mut context); + let brillig_globals = HashMap::default(); + let mut block = + create_brillig_block(&mut function_context, &mut context, &brillig_globals); if push_back { block.slice_push_back_operation( @@ -358,7 +366,9 @@ mod tests { bit_size: BRILLIG_MEMORY_ADDRESSING_BIT_SIZE, }; - let mut block = create_brillig_block(&mut function_context, &mut context); + let brillig_globals = HashMap::default(); + let mut block = + create_brillig_block(&mut function_context, &mut context, &brillig_globals); if pop_back { block.slice_pop_back_operation( @@ -464,7 +474,9 @@ mod tests { // Allocate the results let target_vector = BrilligVector { pointer: context.allocate_register() }; - let mut block = create_brillig_block(&mut function_context, &mut context); + let brillig_globals = HashMap::default(); + let mut block = + create_brillig_block(&mut function_context, &mut context, &brillig_globals); block.slice_insert_operation( target_vector, @@ -604,7 +616,9 @@ mod tests { bit_size: BRILLIG_MEMORY_ADDRESSING_BIT_SIZE, }; - let mut block = create_brillig_block(&mut function_context, &mut context); + let brillig_globals = HashMap::default(); + let mut block = + create_brillig_block(&mut function_context, &mut context, &brillig_globals); block.slice_remove_operation( target_vector, diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/constant_allocation.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/constant_allocation.rs index 61ca20be2f5..64741393dd7 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/constant_allocation.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/constant_allocation.rs @@ -22,6 +22,7 @@ pub(crate) enum InstructionLocation { Terminator, } +#[derive(Default)] pub(crate) struct ConstantAllocation { constant_usage: HashMap>>, allocation_points: HashMap>>, diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/variable_liveness.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/variable_liveness.rs index d6851a9ecf9..37a63466119 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/variable_liveness.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_gen/variable_liveness.rs @@ -53,9 +53,10 @@ pub(crate) fn collect_variables_of_value( let value = &dfg[value_id]; match value { - Value::Instruction { .. } | Value::Param { .. } | Value::NumericConstant { .. } => { - Some(value_id) - } + Value::Instruction { .. } + | Value::Param { .. } + | Value::NumericConstant { .. } + | Value::Global(_) => Some(value_id), // Functions are not variables in a defunctionalized SSA. Only constant function values should appear. Value::ForeignFunction(_) | Value::Function(_) | Value::Intrinsic(..) => None, } @@ -113,6 +114,7 @@ fn compute_used_before_def( type LastUses = HashMap; /// A struct representing the liveness of variables throughout a function. +#[derive(Default)] pub(crate) struct VariableLiveness { cfg: ControlFlowGraph, post_order: PostOrder, @@ -382,14 +384,14 @@ mod test { builder.switch_to_block(b2); let twenty_seven = builder.field_constant(27u128); - let v7 = builder.insert_binary(v0, BinaryOp::Add, twenty_seven); + let v7 = builder.insert_binary(v0, BinaryOp::Add { unchecked: false }, twenty_seven); builder.insert_store(v3, v7); builder.terminate_with_jmp(b3, vec![]); builder.switch_to_block(b1); - let v6 = builder.insert_binary(v1, BinaryOp::Add, twenty_seven); + let v6 = builder.insert_binary(v1, BinaryOp::Add { unchecked: false }, twenty_seven); builder.insert_store(v3, v6); builder.terminate_with_jmp(b3, vec![]); @@ -501,7 +503,7 @@ mod test { builder.switch_to_block(b2); - let v6 = builder.insert_binary(v4, BinaryOp::Mul, v4); + let v6 = builder.insert_binary(v4, BinaryOp::Mul { unchecked: false }, v4); builder.terminate_with_jmp(b4, vec![v0]); @@ -526,7 +528,7 @@ mod test { let v12 = builder.insert_load(v3, Type::field()); - let v13 = builder.insert_binary(v12, BinaryOp::Add, v6); + let v13 = builder.insert_binary(v12, BinaryOp::Add { unchecked: false }, v6); builder.insert_store(v3, v13); @@ -535,13 +537,13 @@ mod test { builder.switch_to_block(b8); let one = builder.field_constant(1u128); - let v15 = builder.insert_binary(v7, BinaryOp::Add, one); + let v15 = builder.insert_binary(v7, BinaryOp::Add { unchecked: false }, one); builder.terminate_with_jmp(b4, vec![v15]); builder.switch_to_block(b6); - let v16 = builder.insert_binary(v4, BinaryOp::Add, one); + let v16 = builder.insert_binary(v4, BinaryOp::Add { unchecked: false }, one); builder.terminate_with_jmp(b1, vec![v16]); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir.rs index 3c100d229a6..06f61948337 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir.rs @@ -37,7 +37,7 @@ use acvm::{ }; use debug_show::DebugShow; -use super::ProcedureId; +use super::{GlobalSpace, ProcedureId}; /// The Brillig VM does not apply a limit to the memory address space, /// As a convention, we take use 32 bits. This means that we assume that @@ -110,7 +110,9 @@ impl BrilligContext { can_call_procedures: true, } } +} +impl BrilligContext { /// Splits a two's complement signed integer in the sign bit and the absolute value. /// For example, -6 i8 (11111010) is split to 00000110 (6, absolute value) and 1 (is_negative). pub(crate) fn absolute_value( @@ -213,6 +215,21 @@ impl BrilligContext { } } +/// Special brillig context to codegen global values initialization +impl BrilligContext { + pub(crate) fn new_for_global_init(enable_debug_trace: bool) -> BrilligContext { + BrilligContext { + obj: BrilligArtifact::default(), + registers: GlobalSpace::new(), + context_label: Label::globals_init(), + current_section: 0, + next_section: 1, + debug_show: DebugShow::new(enable_debug_trace), + can_call_procedures: false, + } + } +} + impl BrilligContext { /// Adds a brillig instruction to the brillig byte code fn push_opcode(&mut self, opcode: BrilligOpcode) { @@ -253,6 +270,10 @@ pub(crate) mod tests { pub(crate) struct DummyBlackBoxSolver; impl BlackBoxFunctionSolver for DummyBlackBoxSolver { + fn pedantic_solving(&self) -> bool { + true + } + fn multi_scalar_mul( &self, _points: &[FieldElement], @@ -295,8 +316,12 @@ pub(crate) mod tests { returns: Vec, ) -> GeneratedBrillig { let artifact = context.artifact(); - let mut entry_point_artifact = - BrilligContext::new_entry_point_artifact(arguments, returns, FunctionId::test_new(0)); + let mut entry_point_artifact = BrilligContext::new_entry_point_artifact( + arguments, + returns, + FunctionId::test_new(0), + false, + ); entry_point_artifact.link_with(&artifact); while let Some(unresolved_fn_label) = entry_point_artifact.first_unresolved_function_call() { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/artifact.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/artifact.rs index be4a6b84bc1..4c48675d1e7 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/artifact.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/artifact.rs @@ -27,7 +27,7 @@ pub(crate) struct GeneratedBrillig { pub(crate) locations: BTreeMap, pub(crate) error_types: BTreeMap, pub(crate) name: String, - pub(crate) procedure_locations: HashMap, + pub(crate) procedure_locations: BTreeMap, } #[derive(Default, Debug, Clone)] @@ -61,7 +61,7 @@ pub(crate) struct BrilligArtifact { /// This is created as artifacts are linked together and allows us to determine /// which opcodes originate from reusable procedures.s /// The range is inclusive for both start and end opcode locations. - pub(crate) procedure_locations: HashMap, + pub(crate) procedure_locations: BTreeMap, } /// A pointer to a location in the opcode. @@ -75,6 +75,8 @@ pub(crate) enum LabelType { Function(FunctionId, Option), /// Labels for intrinsic procedures Procedure(ProcedureId), + /// Label for initialization of globals + GlobalInit, } impl std::fmt::Display for LabelType { @@ -89,6 +91,7 @@ impl std::fmt::Display for LabelType { } LabelType::Entrypoint => write!(f, "Entrypoint"), LabelType::Procedure(procedure_id) => write!(f, "Procedure({:?})", procedure_id), + LabelType::GlobalInit => write!(f, "Globals Initialization"), } } } @@ -123,6 +126,10 @@ impl Label { pub(crate) fn procedure(procedure_id: ProcedureId) -> Self { Label { label_type: LabelType::Procedure(procedure_id), section: None } } + + pub(crate) fn globals_init() -> Self { + Label { label_type: LabelType::GlobalInit, section: None } + } } impl std::fmt::Display for Label { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/codegen_calls.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/codegen_calls.rs index da310873cff..4da3aa4d6d2 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/codegen_calls.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/codegen_calls.rs @@ -9,7 +9,8 @@ use super::{ BrilligBinaryOp, BrilligContext, ReservedRegisters, }; -impl BrilligContext { +impl BrilligContext { + // impl BrilligContext { pub(crate) fn codegen_call( &mut self, func_id: FunctionId, @@ -17,7 +18,7 @@ impl BrilligContext { returns: &[BrilligVariable], ) { let stack_size_register = SingleAddrVariable::new_usize(self.allocate_register()); - let previous_stack_pointer = self.registers.empty_stack_start(); + let previous_stack_pointer = self.registers.empty_registers_start(); let stack_size = previous_stack_pointer.unwrap_relative(); // Write the stack size self.const_instruction(stack_size_register, stack_size.into()); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/entry_point.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/entry_point.rs index 2dbee48b277..b84a15db4ad 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/entry_point.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/entry_point.rs @@ -15,6 +15,7 @@ use acvm::acir::{ pub(crate) const MAX_STACK_SIZE: usize = 16 * MAX_STACK_FRAME_SIZE; pub(crate) const MAX_STACK_FRAME_SIZE: usize = 2048; pub(crate) const MAX_SCRATCH_SPACE: usize = 64; +pub(crate) const MAX_GLOBAL_SPACE: usize = 16384; impl BrilligContext { /// Creates an entry point artifact that will jump to the function label provided. @@ -22,11 +23,16 @@ impl BrilligContext { arguments: Vec, return_parameters: Vec, target_function: FunctionId, + globals_init: bool, ) -> BrilligArtifact { let mut context = BrilligContext::new(false); context.codegen_entry_point(&arguments, &return_parameters); + if globals_init { + context.add_globals_init_instruction(); + } + context.add_external_call_instruction(target_function); context.codegen_exit_point(&arguments, &return_parameters); @@ -34,7 +40,7 @@ impl BrilligContext { } fn calldata_start_offset() -> usize { - ReservedRegisters::len() + MAX_STACK_SIZE + MAX_SCRATCH_SPACE + ReservedRegisters::len() + MAX_STACK_SIZE + MAX_SCRATCH_SPACE + MAX_GLOBAL_SPACE } fn return_data_start_offset(calldata_size: usize) -> usize { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/instructions.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/instructions.rs index 2bf5364414c..d67da423d44 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/instructions.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/instructions.rs @@ -200,6 +200,13 @@ impl BrilligContext< self.obj.add_unresolved_external_call(BrilligOpcode::Call { location: 0 }, proc_label); } + pub(super) fn add_globals_init_instruction(&mut self) { + let globals_init_label = Label::globals_init(); + self.debug_show.add_external_call_instruction(globals_init_label.to_string()); + self.obj + .add_unresolved_external_call(BrilligOpcode::Call { location: 0 }, globals_init_label); + } + /// Adds a unresolved `Jump` instruction to the bytecode. pub(crate) fn jump_instruction(&mut self, target_label: Label) { self.debug_show.jump_instruction(target_label.to_string()); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/registers.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/registers.rs index dd7766f40aa..b83c03b297a 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/registers.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/brillig_ir/registers.rs @@ -7,7 +7,7 @@ use crate::brillig::brillig_ir::entry_point::MAX_STACK_SIZE; use super::{ brillig_variable::SingleAddrVariable, - entry_point::{MAX_SCRATCH_SPACE, MAX_STACK_FRAME_SIZE}, + entry_point::{MAX_GLOBAL_SPACE, MAX_SCRATCH_SPACE, MAX_STACK_FRAME_SIZE}, BrilligContext, ReservedRegisters, }; @@ -24,6 +24,8 @@ pub(crate) trait RegisterAllocator { fn ensure_register_is_allocated(&mut self, register: MemoryAddress); /// Creates a new register context from a set of registers allocated previously. fn from_preallocated_registers(preallocated_registers: Vec) -> Self; + /// Finds the first register that is available based upon the deallocation list + fn empty_registers_start(&self) -> MemoryAddress; } /// Every brillig stack frame/call context has its own view of register space. @@ -41,10 +43,6 @@ impl Stack { let offset = register.unwrap_relative(); offset >= Self::start() && offset < Self::end() } - - pub(crate) fn empty_stack_start(&self) -> MemoryAddress { - MemoryAddress::relative(self.storage.empty_registers_start(Self::start())) - } } impl RegisterAllocator for Stack { @@ -83,6 +81,10 @@ impl RegisterAllocator for Stack { ), } } + + fn empty_registers_start(&self) -> MemoryAddress { + MemoryAddress::relative(self.storage.empty_registers_start(Self::start())) + } } /// Procedure arguments and returns are passed through scratch space. @@ -109,7 +111,7 @@ impl RegisterAllocator for ScratchSpace { } fn end() -> usize { - ReservedRegisters::len() + MAX_STACK_SIZE + MAX_SCRATCH_SPACE + Self::start() + MAX_SCRATCH_SPACE } fn ensure_register_is_allocated(&mut self, register: MemoryAddress) { @@ -139,6 +141,70 @@ impl RegisterAllocator for ScratchSpace { ), } } + + fn empty_registers_start(&self) -> MemoryAddress { + MemoryAddress::direct(self.storage.empty_registers_start(Self::start())) + } +} + +/// Globals have a separate memory space +/// This memory space is initialized once at the beginning of a program +/// and is read-only. +pub(crate) struct GlobalSpace { + storage: DeallocationListAllocator, +} + +impl GlobalSpace { + pub(crate) fn new() -> Self { + Self { storage: DeallocationListAllocator::new(Self::start()) } + } + + fn is_within_bounds(register: MemoryAddress) -> bool { + let index = register.unwrap_direct(); + index >= Self::start() && index < Self::end() + } +} + +impl RegisterAllocator for GlobalSpace { + fn start() -> usize { + ScratchSpace::end() + } + + fn end() -> usize { + Self::start() + MAX_GLOBAL_SPACE + } + + fn allocate_register(&mut self) -> MemoryAddress { + let allocated = MemoryAddress::direct(self.storage.allocate_register()); + assert!(Self::is_within_bounds(allocated), "Global space too deep"); + allocated + } + + fn deallocate_register(&mut self, register_index: MemoryAddress) { + self.storage.deallocate_register(register_index.unwrap_direct()); + } + + fn ensure_register_is_allocated(&mut self, register: MemoryAddress) { + assert!(Self::is_within_bounds(register), "Register out of global space bounds"); + self.storage.ensure_register_is_allocated(register.unwrap_direct()); + } + + fn from_preallocated_registers(preallocated_registers: Vec) -> Self { + for register in &preallocated_registers { + assert!(Self::is_within_bounds(*register), "Register out of global space bounds"); + } + + Self { + storage: DeallocationListAllocator::from_preallocated_registers( + Self::start(), + vecmap(preallocated_registers, |r| r.unwrap_direct()), + ), + } + } + + fn empty_registers_start(&self) -> MemoryAddress { + MemoryAddress::direct(self.storage.empty_registers_start(Self::start())) + } } struct DeallocationListAllocator { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/mod.rs b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/mod.rs index cb8c35cd8e0..3d96a855aa0 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/brillig/mod.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/brillig/mod.rs @@ -2,7 +2,8 @@ pub(crate) mod brillig_gen; pub(crate) mod brillig_ir; use acvm::FieldElement; -use brillig_ir::artifact::LabelType; +use brillig_gen::brillig_globals::convert_ssa_globals; +use brillig_ir::{artifact::LabelType, brillig_variable::BrilligVariable, registers::GlobalSpace}; use self::{ brillig_gen::convert_ssa_function, @@ -12,7 +13,11 @@ use self::{ }, }; use crate::ssa::{ - ir::function::{Function, FunctionId}, + ir::{ + dfg::DataFlowGraph, + function::{Function, FunctionId}, + value::ValueId, + }, ssa_gen::Ssa, }; use fxhash::FxHashMap as HashMap; @@ -26,12 +31,18 @@ pub use self::brillig_ir::procedures::ProcedureId; pub struct Brillig { /// Maps SSA function labels to their brillig artifact ssa_function_to_brillig: HashMap>, + globals: BrilligArtifact, } impl Brillig { /// Compiles a function into brillig and store the compilation artifacts - pub(crate) fn compile(&mut self, func: &Function, enable_debug_trace: bool) { - let obj = convert_ssa_function(func, enable_debug_trace); + pub(crate) fn compile( + &mut self, + func: &Function, + enable_debug_trace: bool, + globals: &HashMap, + ) { + let obj = convert_ssa_function(func, enable_debug_trace, globals); self.ssa_function_to_brillig.insert(func.id(), obj); } @@ -46,6 +57,7 @@ impl Brillig { } // Procedures are compiled as needed LabelType::Procedure(procedure_id) => Some(Cow::Owned(compile_procedure(procedure_id))), + LabelType::GlobalInit => Some(Cow::Borrowed(&self.globals)), _ => unreachable!("ICE: Expected a function or procedure label"), } } @@ -71,9 +83,14 @@ impl Ssa { .collect::>(); let mut brillig = Brillig::default(); + + let (artifact, brillig_globals) = + convert_ssa_globals(enable_debug_trace, &self.globals, &self.used_global_values); + brillig.globals = artifact; + for brillig_function_id in brillig_reachable_function_ids { let func = &self.functions[&brillig_function_id]; - brillig.compile(func, enable_debug_trace); + brillig.compile(func, enable_debug_trace, &brillig_globals); } brillig diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/errors.rs b/noir/noir-repo/compiler/noirc_evaluator/src/errors.rs index ee662c50b75..94c0e0554a4 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/errors.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/errors.rs @@ -49,8 +49,8 @@ pub enum RuntimeError { StaticAssertDynamicMessage { call_stack: CallStack }, #[error("Argument is dynamic")] StaticAssertDynamicPredicate { call_stack: CallStack }, - #[error("Argument is false")] - StaticAssertFailed { call_stack: CallStack }, + #[error("{message}")] + StaticAssertFailed { message: String, call_stack: CallStack }, #[error("Nested slices, i.e. slices within an array or slice, are not supported")] NestedSlice { call_stack: CallStack }, #[error("Big Integer modulus do no match")] @@ -165,7 +165,7 @@ impl RuntimeError { | RuntimeError::AssertConstantFailed { call_stack } | RuntimeError::StaticAssertDynamicMessage { call_stack } | RuntimeError::StaticAssertDynamicPredicate { call_stack } - | RuntimeError::StaticAssertFailed { call_stack } + | RuntimeError::StaticAssertFailed { call_stack, .. } | RuntimeError::IntegerOutOfBounds { call_stack, .. } | RuntimeError::UnsupportedIntegerSize { call_stack, .. } | RuntimeError::InvalidBlackBoxInputBitSize { call_stack, .. } @@ -201,7 +201,7 @@ impl RuntimeError { RuntimeError::UnknownLoopBound { .. } => { let primary_message = self.to_string(); let location = - self.call_stack().back().expect("Expected RuntimeError to have a location"); + self.call_stack().last().expect("Expected RuntimeError to have a location"); Diagnostic::simple_error( primary_message, @@ -212,7 +212,7 @@ impl RuntimeError { _ => { let message = self.to_string(); let location = - self.call_stack().back().unwrap_or_else(|| panic!("Expected RuntimeError to have a location. Error message: {message}")); + self.call_stack().last().unwrap_or_else(|| panic!("Expected RuntimeError to have a location. Error message: {message}")); Diagnostic::simple_error(message, String::new(), location.span) } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa.rs index 9377cadb260..ed515bbe98c 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa.rs @@ -5,7 +5,6 @@ //! elimination and constant folding. //! //! This module heavily borrows from Cranelift -#![allow(dead_code)] use std::{ collections::{BTreeMap, BTreeSet}, @@ -57,9 +56,6 @@ pub struct SsaEvaluatorOptions { pub enable_brillig_logging: bool, - /// Force Brillig output (for step debugging) - pub force_brillig_output: bool, - /// Pretty print benchmark times of each code generation pass pub print_codegen_timings: bool, @@ -100,7 +96,6 @@ pub(crate) fn optimize_into_acir( let builder = SsaBuilder::new( program, options.ssa_logging.clone(), - options.force_brillig_output, options.print_codegen_timings, &options.emit_ssa, )?; @@ -155,15 +150,15 @@ pub(crate) fn optimize_into_acir( /// Run all SSA passes. fn optimize_all(builder: SsaBuilder, options: &SsaEvaluatorOptions) -> Result { Ok(builder + .run_pass(Ssa::remove_unreachable_functions, "Removing Unreachable Functions") .run_pass(Ssa::defunctionalize, "Defunctionalization") .run_pass(Ssa::remove_paired_rc, "Removing Paired rc_inc & rc_decs") - .run_pass(Ssa::separate_runtime, "Runtime Separation") - .run_pass(Ssa::resolve_is_unconstrained, "Resolving IsUnconstrained") .run_pass(|ssa| ssa.inline_functions(options.inliner_aggressiveness), "Inlining (1st)") // Run mem2reg with the CFG separated into blocks .run_pass(Ssa::mem2reg, "Mem2Reg (1st)") .run_pass(Ssa::simplify_cfg, "Simplifying (1st)") .run_pass(Ssa::as_slice_optimization, "`as_slice` optimization") + .run_pass(Ssa::remove_unreachable_functions, "Removing Unreachable Functions") .try_run_pass( Ssa::evaluate_static_assert_and_assert_constant, "`static_assert` and `assert_constant`", @@ -174,10 +169,11 @@ fn optimize_all(builder: SsaBuilder, options: &SsaEvaluatorOptions) -> Result Result, ) -> Result { - let ssa = ssa_gen::generate_ssa(program, force_brillig_runtime)?; + let ssa = ssa_gen::generate_ssa(program)?; if let Some(emit_ssa) = emit_ssa { let mut emit_ssa_dir = emit_ssa.clone(); // We expect the full package artifact path to be passed in here, @@ -465,7 +454,7 @@ impl SsaBuilder { let ssa_path = emit_ssa.with_extension("ssa.json"); write_to_file(&serde_json::to_vec(&ssa).unwrap(), &ssa_path); } - Ok(SsaBuilder { ssa_logging, print_codegen_timings, ssa }.print("Initial SSA:")) + Ok(SsaBuilder { ssa_logging, print_codegen_timings, ssa }.print("Initial SSA")) } fn finish(self) -> Ssa { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/checks/check_for_underconstrained_values.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/checks/check_for_underconstrained_values.rs index 48be8eb7ad8..f44f726bfc7 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/checks/check_for_underconstrained_values.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/checks/check_for_underconstrained_values.rs @@ -10,7 +10,7 @@ use crate::ssa::ir::value::{Value, ValueId}; use crate::ssa::ssa_gen::Ssa; use im::HashMap; use rayon::prelude::*; -use std::collections::{BTreeMap, HashSet}; +use std::collections::{BTreeMap, BTreeSet, HashSet}; use tracing::trace; impl Ssa { @@ -73,7 +73,7 @@ fn check_for_underconstrained_values_within_function( context.compute_sets_of_connected_value_ids(function, all_functions); - let all_brillig_generated_values: HashSet = + let all_brillig_generated_values: BTreeSet = context.brillig_return_to_argument.keys().copied().collect(); let connected_sets_indices = @@ -81,7 +81,7 @@ fn check_for_underconstrained_values_within_function( // Go through each disconnected set, find brillig calls that caused it and form warnings for set_index in - HashSet::from_iter(0..(context.value_sets.len())).difference(&connected_sets_indices) + BTreeSet::from_iter(0..(context.value_sets.len())).difference(&connected_sets_indices) { let current_set = &context.value_sets[*set_index]; warnings.append(&mut context.find_disconnecting_brillig_calls_with_results_in_set( @@ -104,7 +104,7 @@ struct DependencyContext { array_elements: HashMap, // Map of brillig call ids to sets of the value ids descending // from their arguments and results - tainted: HashMap, + tainted: BTreeMap, } /// Structure keeping track of value ids descending from Brillig calls' @@ -267,7 +267,8 @@ impl DependencyContext { } // Check the constrain instruction arguments against those // involved in Brillig calls, remove covered calls - Instruction::Constrain(value_id1, value_id2, _) => { + Instruction::Constrain(value_id1, value_id2, _) + | Instruction::ConstrainNotEqual(value_id1, value_id2, _) => { self.clear_constrained( &[function.dfg.resolve(*value_id1), function.dfg.resolve(*value_id2)], function, @@ -294,11 +295,9 @@ impl DependencyContext { Intrinsic::ArrayLen | Intrinsic::ArrayRefCount | Intrinsic::ArrayAsStrUnchecked - | Intrinsic::AsField | Intrinsic::AsSlice | Intrinsic::BlackBox(..) | Intrinsic::DerivePedersenGenerators - | Intrinsic::FromField | Intrinsic::Hint(..) | Intrinsic::SlicePushBack | Intrinsic::SlicePushFront @@ -316,10 +315,9 @@ impl DependencyContext { self.update_children(&arguments, &results); } }, - Value::Function(callee) => match all_functions[&callee].runtime() { + Value::Function(callee) => match all_functions[callee].runtime() { RuntimeType::Brillig(_) => { // Record arguments/results for each Brillig call for the check - self.tainted.insert( *instruction, BrilligTaintedIds::new(&arguments, &results), @@ -335,7 +333,8 @@ impl DependencyContext { } Value::Instruction { .. } | Value::NumericConstant { .. } - | Value::Param { .. } => { + | Value::Param { .. } + | Value::Global(_) => { panic!( "calling non-function value with ID {func_id} in function {}", function.name() @@ -368,6 +367,7 @@ impl DependencyContext { | Instruction::DecrementRc { .. } | Instruction::EnableSideEffectsIf { .. } | Instruction::IncrementRc { .. } + | Instruction::Noop | Instruction::MakeArray { .. } => {} } } @@ -436,7 +436,7 @@ impl DependencyContext { struct Context { visited_blocks: HashSet, block_queue: Vec, - value_sets: Vec>, + value_sets: Vec>, brillig_return_to_argument: HashMap>, brillig_return_to_instruction_id: HashMap, } @@ -469,7 +469,7 @@ impl Context { fn find_sets_connected_to_function_inputs_or_outputs( &mut self, function: &Function, - ) -> HashSet { + ) -> BTreeSet { let variable_parameters_and_return_values = function .parameters() .iter() @@ -477,7 +477,7 @@ impl Context { .filter(|id| function.dfg.get_numeric_constant(**id).is_none()) .map(|value_id| function.dfg.resolve(*value_id)); - let mut connected_sets_indices: HashSet = HashSet::new(); + let mut connected_sets_indices: BTreeSet = BTreeSet::default(); // Go through each parameter and each set and check if the set contains the parameter // If it's the case, then that set doesn't present an issue @@ -494,8 +494,8 @@ impl Context { /// Find which Brillig calls separate this set from others and return bug warnings about them fn find_disconnecting_brillig_calls_with_results_in_set( &self, - current_set: &HashSet, - all_brillig_generated_values: &HashSet, + current_set: &BTreeSet, + all_brillig_generated_values: &BTreeSet, function: &Function, ) -> Vec { let mut warnings = Vec::new(); @@ -505,7 +505,7 @@ impl Context { // Go through all Brillig outputs in the set for brillig_output_in_set in intersection { // Get the inputs that correspond to the output - let inputs: HashSet = + let inputs: BTreeSet = self.brillig_return_to_argument[&brillig_output_in_set].iter().copied().collect(); // Check if any of them are not in the set @@ -534,7 +534,7 @@ impl Context { let instructions = function.dfg[block].instructions(); for instruction in instructions.iter() { - let mut instruction_arguments_and_results = HashSet::new(); + let mut instruction_arguments_and_results = BTreeSet::new(); // Insert non-constant instruction arguments function.dfg[*instruction].for_each_value(|value_id| { @@ -556,6 +556,7 @@ impl Context { | Instruction::Binary(..) | Instruction::Cast(..) | Instruction::Constrain(..) + | Instruction::ConstrainNotEqual(..) | Instruction::IfElse { .. } | Instruction::Load { .. } | Instruction::Not(..) @@ -575,12 +576,10 @@ impl Context { Intrinsic::ArrayLen | Intrinsic::ArrayAsStrUnchecked | Intrinsic::ArrayRefCount - | Intrinsic::AsField | Intrinsic::AsSlice | Intrinsic::BlackBox(..) | Intrinsic::Hint(Hint::BlackBox) | Intrinsic::DerivePedersenGenerators - | Intrinsic::FromField | Intrinsic::SliceInsert | Intrinsic::SlicePushBack | Intrinsic::SlicePushFront @@ -596,7 +595,7 @@ impl Context { self.value_sets.push(instruction_arguments_and_results); } }, - Value::Function(callee) => match all_functions[&callee].runtime() { + Value::Function(callee) => match all_functions[callee].runtime() { RuntimeType::Brillig(_) => { // For calls to Brillig functions we memorize the mapping of results to argument ValueId's and InstructionId's // The latter are needed to produce the callstack later @@ -622,7 +621,8 @@ impl Context { } Value::Instruction { .. } | Value::NumericConstant { .. } - | Value::Param { .. } => { + | Value::Param { .. } + | Value::Global(_) => { panic!("At the point we are running disconnect there shouldn't be any other values as arguments") } } @@ -631,6 +631,7 @@ impl Context { | Instruction::DecrementRc { .. } | Instruction::EnableSideEffectsIf { .. } | Instruction::IncrementRc { .. } + | Instruction::Noop | Instruction::RangeCheck { .. } => {} } } @@ -641,15 +642,15 @@ impl Context { /// Merge all small sets into larger ones based on whether the sets intersect or not /// /// If two small sets have a common ValueId, we merge them into one - fn merge_sets(current: &[HashSet]) -> Vec> { + fn merge_sets(current: &[BTreeSet]) -> Vec> { let mut new_set_id: usize = 0; - let mut updated_sets: HashMap> = HashMap::new(); - let mut value_dictionary: HashMap = HashMap::new(); - let mut parsed_value_set: HashSet = HashSet::new(); + let mut updated_sets: BTreeMap> = BTreeMap::default(); + let mut value_dictionary: HashMap = HashMap::default(); + let mut parsed_value_set: BTreeSet = BTreeSet::default(); for set in current.iter() { // Check if the set has any of the ValueIds we've encountered at previous iterations - let intersection: HashSet = + let intersection: BTreeSet = set.intersection(&parsed_value_set).copied().collect(); parsed_value_set.extend(set.iter()); @@ -666,7 +667,7 @@ impl Context { } // If there is an intersection, we have to join the sets - let mut joining_sets_ids: HashSet = + let mut joining_sets_ids: BTreeSet = intersection.iter().map(|x| value_dictionary[x]).collect(); let mut largest_set_size = usize::MIN; let mut largest_set_index = usize::MAX; @@ -680,7 +681,7 @@ impl Context { joining_sets_ids.remove(&largest_set_index); let mut largest_set = - updated_sets.extract(&largest_set_index).expect("Set should be in the hashmap").0; + updated_sets.remove(&largest_set_index).expect("Set should be in the hashmap"); // For each of other sets that need to be joined for set_id in joining_sets_ids.iter() { @@ -705,7 +706,7 @@ impl Context { /// Parallel version of merge_sets /// The sets are merged by chunks, and then the chunks are merged together - fn merge_sets_par(sets: &[HashSet]) -> Vec> { + fn merge_sets_par(sets: &[BTreeSet]) -> Vec> { let mut sets = sets.to_owned(); let mut len = sets.len(); let mut prev_len = len + 1; diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/function_builder/data_bus.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/function_builder/data_bus.rs index 48af34d466c..068fff7d284 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/function_builder/data_bus.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/function_builder/data_bus.rs @@ -252,7 +252,7 @@ impl FunctionBuilder { for size in ssa_param_sizes { let visibilities: Vec = flattened_params_databus_visibility.drain(0..size).collect(); - let visibility = visibilities.get(0).copied().unwrap_or(DatabusVisibility::None); + let visibility = visibilities.first().copied().unwrap_or(DatabusVisibility::None); assert!( visibilities.iter().all(|v| *v == visibility), "inconsistent databus visibility for ssa param" diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/function_builder/mod.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/function_builder/mod.rs index 855034cedd2..154d485b143 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/function_builder/mod.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/function_builder/mod.rs @@ -19,7 +19,7 @@ use super::{ ir::{ basic_block::BasicBlock, call_stack::{CallStack, CallStackId}, - dfg::InsertInstructionResult, + dfg::{GlobalsGraph, InsertInstructionResult}, function::RuntimeType, instruction::{ConstrainError, InstructionId, Intrinsic}, types::NumericType, @@ -40,6 +40,10 @@ pub(crate) struct FunctionBuilder { finished_functions: Vec, call_stack: CallStackId, error_types: BTreeMap, + + /// Whether instructions are simplified as soon as they are inserted into this builder. + /// This is true by default unless changed to false after constructing a builder. + pub(crate) simplify: bool, } impl FunctionBuilder { @@ -56,6 +60,7 @@ impl FunctionBuilder { finished_functions: Vec::new(), call_stack: CallStackId::root(), error_types: BTreeMap::default(), + simplify: true, } } @@ -68,6 +73,13 @@ impl FunctionBuilder { self.current_function.set_runtime(runtime); } + pub(crate) fn set_globals(&mut self, globals: Arc) { + for (_, value) in globals.values_iter() { + self.current_function.dfg.make_global(value.get_type().into_owned()); + } + self.current_function.set_globals(globals); + } + /// Finish the current function and create a new function. /// /// A FunctionBuilder can always only work on one function at a time, so care @@ -133,6 +145,7 @@ impl FunctionBuilder { } /// Insert a numeric constant into the current function of type Field + #[cfg(test)] pub(crate) fn field_constant(&mut self, value: impl Into) -> ValueId { self.numeric_constant(value.into(), NumericType::NativeField) } @@ -171,12 +184,21 @@ impl FunctionBuilder { ctrl_typevars: Option>, ) -> InsertInstructionResult { let block = self.current_block(); - self.current_function.dfg.insert_instruction_and_results( - instruction, - block, - ctrl_typevars, - self.call_stack, - ) + if self.simplify { + self.current_function.dfg.insert_instruction_and_results( + instruction, + block, + ctrl_typevars, + self.call_stack, + ) + } else { + self.current_function.dfg.insert_instruction_and_results_without_simplification( + instruction, + block, + ctrl_typevars, + self.call_stack, + ) + } } /// Switch to inserting instructions in the given block. @@ -236,14 +258,6 @@ impl FunctionBuilder { operator: BinaryOp, rhs: ValueId, ) -> ValueId { - let lhs_type = self.type_of_value(lhs); - let rhs_type = self.type_of_value(rhs); - if operator != BinaryOp::Shl && operator != BinaryOp::Shr { - assert_eq!( - lhs_type, rhs_type, - "ICE - Binary instruction operands must have the same type" - ); - } let instruction = Instruction::Binary(Binary { lhs, rhs, operator }); self.insert_instruction(instruction, None).first() } @@ -328,6 +342,7 @@ impl FunctionBuilder { .first() } + #[cfg(test)] pub(crate) fn insert_mutable_array_set( &mut self, array: ValueId, @@ -467,29 +482,33 @@ impl FunctionBuilder { /// /// Returns whether a reference count instruction was issued. fn update_array_reference_count(&mut self, value: ValueId, increment: bool) -> bool { - match self.type_of_value(value) { - Type::Numeric(_) => false, - Type::Function => false, - Type::Reference(element) => { - if element.contains_an_array() { - let reference = value; - let value = self.insert_load(reference, element.as_ref().clone()); - self.update_array_reference_count(value, increment); - true - } else { - false + if self.current_function.runtime().is_brillig() { + match self.type_of_value(value) { + Type::Numeric(_) => false, + Type::Function => false, + Type::Reference(element) => { + if element.contains_an_array() { + let reference = value; + let value = self.insert_load(reference, element.as_ref().clone()); + self.update_array_reference_count(value, increment); + true + } else { + false + } } - } - Type::Array(..) | Type::Slice(..) => { - // If there are nested arrays or slices, we wait until ArrayGet - // is issued to increment the count of that array. - if increment { - self.insert_inc_rc(value); - } else { - self.insert_dec_rc(value); + Type::Array(..) | Type::Slice(..) => { + // If there are nested arrays or slices, we wait until ArrayGet + // is issued to increment the count of that array. + if increment { + self.insert_inc_rc(value); + } else { + self.insert_dec_rc(value); + } + true } - true } + } else { + false } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/call_stack.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/call_stack.rs index 113609f4a11..e1df616bc66 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/call_stack.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/call_stack.rs @@ -3,7 +3,7 @@ use serde::{Deserialize, Serialize}; use noirc_errors::Location; -pub(crate) type CallStack = im::Vector; +pub(crate) type CallStack = Vec; #[derive(Debug, Default, Clone, Copy, PartialEq, Eq, Hash, Serialize, Deserialize)] pub(crate) struct CallStackId(u32); @@ -57,9 +57,9 @@ impl Default for CallStackHelper { impl CallStackHelper { /// Construct a CallStack from a CallStackId pub(crate) fn get_call_stack(&self, mut call_stack: CallStackId) -> CallStack { - let mut result = im::Vector::new(); + let mut result = Vec::new(); while let Some(parent) = self.locations[call_stack.index()].parent { - result.push_back(self.locations[call_stack.index()].value); + result.push(self.locations[call_stack.index()].value); call_stack = parent; } result diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/cfg.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/cfg.rs index 788b1a7d302..5253c68c72c 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/cfg.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/cfg.rs @@ -18,7 +18,7 @@ struct CfgNode { pub(crate) successors: BTreeSet, } -#[derive(Clone)] +#[derive(Clone, Default)] /// The Control Flow Graph maintains a mapping of blocks to their predecessors /// and successors where predecessors are basic blocks and successors are /// basic blocks. diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/dfg.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/dfg.rs index 72ba369f98c..83b8f2a57ff 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/dfg.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/dfg.rs @@ -1,11 +1,11 @@ -use std::borrow::Cow; +use std::{borrow::Cow, sync::Arc}; use crate::ssa::{function_builder::data_bus::DataBus, ir::instruction::SimplifyResult}; use super::{ basic_block::{BasicBlock, BasicBlockId}, call_stack::{CallStack, CallStackHelper, CallStackId}, - function::FunctionId, + function::{FunctionId, RuntimeType}, instruction::{ Instruction, InstructionId, InstructionResultType, Intrinsic, TerminatorInstruction, }, @@ -17,7 +17,6 @@ use super::{ use acvm::{acir::AcirField, FieldElement}; use fxhash::FxHashMap as HashMap; use iter_extended::vecmap; -use noirc_errors::Location; use serde::{Deserialize, Serialize}; use serde_with::serde_as; use serde_with::DisplayFromStr; @@ -27,8 +26,13 @@ use serde_with::DisplayFromStr; /// owning most data in a function and handing out Ids to this data that can be /// shared without worrying about ownership. #[serde_as] -#[derive(Debug, Default, Clone, Serialize, Deserialize)] +#[derive(Debug, Clone, Serialize, Deserialize, Default)] pub(crate) struct DataFlowGraph { + /// Runtime of the [Function] that owns this [DataFlowGraph]. + /// This might change during the `runtime_separation` pass where + /// ACIR functions are cloned as Brillig functions. + runtime: RuntimeType, + /// All of the instructions in a function instructions: DenseMap, @@ -42,7 +46,7 @@ pub(crate) struct DataFlowGraph { /// Call instructions require the func signature, but /// other instructions may need some more reading on my part #[serde_as(as = "HashMap")] - results: HashMap>, + results: HashMap>, /// Storage for all of the values defined in this /// function. @@ -98,9 +102,44 @@ pub(crate) struct DataFlowGraph { #[serde(skip)] pub(crate) data_bus: DataBus, + + pub(crate) globals: Arc, +} + +/// The GlobalsGraph contains the actual global data. +/// Global data is expected to only be numeric constants or array constants (which are represented by Instruction::MakeArray). +/// The global's data will shared across functions and should be accessible inside of a function's DataFlowGraph. +#[derive(Debug, Clone, Serialize, Deserialize, Default)] +pub(crate) struct GlobalsGraph { + /// Storage for all of the global values + values: DenseMap, + /// All of the instructions in the global value space. + /// These are expected to all be Instruction::MakeArray + instructions: DenseMap, +} + +impl GlobalsGraph { + pub(crate) fn from_dfg(dfg: DataFlowGraph) -> Self { + Self { values: dfg.values, instructions: dfg.instructions } + } + + /// Iterate over every Value in this DFG in no particular order, including unused Values + pub(crate) fn values_iter(&self) -> impl DoubleEndedIterator { + self.values.iter() + } } impl DataFlowGraph { + /// Runtime type of the function. + pub(crate) fn runtime(&self) -> RuntimeType { + self.runtime + } + + /// Set runtime type of the function. + pub(crate) fn set_runtime(&mut self, runtime: RuntimeType) { + self.runtime = runtime; + } + /// Creates a new basic block with no parameters. /// After being created, the block is unreachable in the current function /// until another block is made to jump to it. @@ -134,12 +173,12 @@ impl DataFlowGraph { /// The pairs are order by id, which is not guaranteed to be meaningful. pub(crate) fn basic_blocks_iter( &self, - ) -> impl ExactSizeIterator { + ) -> impl DoubleEndedIterator { self.blocks.iter() } /// Iterate over every Value in this DFG in no particular order, including unused Values - pub(crate) fn values_iter(&self) -> impl ExactSizeIterator { + pub(crate) fn values_iter(&self) -> impl DoubleEndedIterator { self.values.iter() } @@ -165,7 +204,56 @@ impl DataFlowGraph { id } - /// Inserts a new instruction at the end of the given block and returns its results + /// Check if the function runtime would simply ignore this instruction. + pub(crate) fn is_handled_by_runtime(&self, instruction: &Instruction) -> bool { + match self.runtime() { + RuntimeType::Acir(_) => !matches!( + instruction, + Instruction::IncrementRc { .. } | Instruction::DecrementRc { .. } + ), + RuntimeType::Brillig(_) => !matches!( + instruction, + Instruction::EnableSideEffectsIf { .. } | Instruction::IfElse { .. } + ), + } + } + + fn insert_instruction_without_simplification( + &mut self, + instruction_data: Instruction, + block: BasicBlockId, + ctrl_typevars: Option>, + call_stack: CallStackId, + ) -> InstructionId { + let id = self.make_instruction(instruction_data, ctrl_typevars); + self.blocks[block].insert_instruction(id); + self.locations.insert(id, call_stack); + id + } + + pub(crate) fn insert_instruction_and_results_without_simplification( + &mut self, + instruction_data: Instruction, + block: BasicBlockId, + ctrl_typevars: Option>, + call_stack: CallStackId, + ) -> InsertInstructionResult { + if !self.is_handled_by_runtime(&instruction_data) { + panic!("Attempted to insert instruction not handled by runtime: {instruction_data:?}"); + } + + let id = self.insert_instruction_without_simplification( + instruction_data, + block, + ctrl_typevars, + call_stack, + ); + + InsertInstructionResult::Results(id, self.instruction_results(id)) + } + + /// Simplifies a new instruction and inserts it at the end of the given block and returns its results. + /// If the instruction is not handled by the current runtime, `InstructionRemoved` is returned. pub(crate) fn insert_instruction_and_results( &mut self, instruction: Instruction, @@ -173,6 +261,28 @@ impl DataFlowGraph { ctrl_typevars: Option>, call_stack: CallStackId, ) -> InsertInstructionResult { + self.insert_instruction_and_results_if_simplified( + instruction, + block, + ctrl_typevars, + call_stack, + None, + ) + } + + /// Simplifies a potentially existing instruction and inserts it only if it changed. + pub(crate) fn insert_instruction_and_results_if_simplified( + &mut self, + instruction: Instruction, + block: BasicBlockId, + ctrl_typevars: Option>, + call_stack: CallStackId, + existing_id: Option, + ) -> InsertInstructionResult { + if !self.is_handled_by_runtime(&instruction) { + panic!("Attempted to insert instruction not handled by runtime: {instruction:?}"); + } + match instruction.simplify(self, block, ctrl_typevars.clone(), call_stack) { SimplifyResult::SimplifiedTo(simplification) => { InsertInstructionResult::SimplifiedTo(simplification) @@ -184,7 +294,21 @@ impl DataFlowGraph { result @ (SimplifyResult::SimplifiedToInstruction(_) | SimplifyResult::SimplifiedToInstructionMultiple(_) | SimplifyResult::None) => { - let instructions = result.instructions().unwrap_or(vec![instruction]); + let instructions = result.instructions(); + if instructions.is_none() { + if let Some(id) = existing_id { + if self[id] == instruction { + // Just (re)insert into the block, no need to redefine. + self.blocks[block].insert_instruction(id); + return InsertInstructionResult::Results( + id, + self.instruction_results(id), + ); + } + } + } + let mut instructions = instructions.unwrap_or(vec![instruction]); + assert!(!instructions.is_empty(), "`SimplifyResult::SimplifiedToInstructionMultiple` must not return empty vector"); if instructions.len() > 1 { // There's currently no way to pass results from one instruction in `instructions` on to the next. @@ -196,27 +320,26 @@ impl DataFlowGraph { ); } - let mut last_id = None; - + // Pull off the last instruction as we want to return its results. + let last_instruction = instructions.pop().expect("`instructions` can't be empty"); for instruction in instructions { - let id = self.make_instruction(instruction, ctrl_typevars.clone()); - self.blocks[block].insert_instruction(id); - self.locations.insert(id, call_stack); - last_id = Some(id); + self.insert_instruction_without_simplification( + instruction, + block, + ctrl_typevars.clone(), + call_stack, + ); } - - let id = last_id.expect("There should be at least 1 simplified instruction"); - InsertInstructionResult::Results(id, self.instruction_results(id)) + self.insert_instruction_and_results_without_simplification( + last_instruction, + block, + ctrl_typevars, + call_stack, + ) } } } - /// Insert a value into the dfg's storage and return an id to reference it. - /// Until the value is used in an instruction it is unreachable. - pub(crate) fn make_value(&mut self, value: Value) -> ValueId { - self.values.insert(value) - } - /// Set the value of value_to_replace to refer to the value referred to by new_value. /// /// This is the preferred method to call for optimizations simplifying @@ -268,6 +391,10 @@ impl DataFlowGraph { id } + pub(crate) fn make_global(&mut self, typ: Type) -> ValueId { + self.values.insert(Value::Global(typ)) + } + /// Gets or creates a ValueId for the given FunctionId. pub(crate) fn import_function(&mut self, function: FunctionId) -> ValueId { if let Some(existing) = self.functions.get(&function) { @@ -306,16 +433,18 @@ impl DataFlowGraph { /// Returns the results of the instruction pub(crate) fn make_instruction_results( &mut self, - instruction_id: InstructionId, + instruction: InstructionId, ctrl_typevars: Option>, ) { - let result_types = self.instruction_result_types(instruction_id, ctrl_typevars); - let results = vecmap(result_types.into_iter().enumerate(), |(position, typ)| { - let instruction = instruction_id; - self.values.insert(Value::Instruction { typ, position, instruction }) + let mut results = smallvec::SmallVec::new(); + let mut position = 0; + self.for_each_instruction_result_type(instruction, ctrl_typevars, |this, typ| { + let result = this.values.insert(Value::Instruction { typ, position, instruction }); + position += 1; + results.push(result); }); - self.results.insert(instruction_id, results); + self.results.insert(instruction, results); } /// Return the result types of this instruction. @@ -326,18 +455,21 @@ impl DataFlowGraph { /// the type of an instruction that does not require them. Compared to passing an empty Vec, /// Option has the benefit of panicking if it is accidentally used for a Call instruction, /// rather than silently returning the empty Vec and continuing. - fn instruction_result_types( - &self, + fn for_each_instruction_result_type( + &mut self, instruction_id: InstructionId, ctrl_typevars: Option>, - ) -> Vec { + mut f: impl FnMut(&mut Self, Type), + ) { let instruction = &self.instructions[instruction_id]; match instruction.result_type() { - InstructionResultType::Known(typ) => vec![typ], - InstructionResultType::Operand(value) => vec![self.type_of_value(value)], - InstructionResultType::None => vec![], + InstructionResultType::Known(typ) => f(self, typ), + InstructionResultType::Operand(value) => f(self, self.type_of_value(value)), + InstructionResultType::None => (), InstructionResultType::Unknown => { - ctrl_typevars.expect("Control typevars required but not given") + for typ in ctrl_typevars.expect("Control typevars required but not given") { + f(self, typ); + } } } } @@ -354,10 +486,15 @@ impl DataFlowGraph { pub(crate) fn get_value_max_num_bits(&self, value: ValueId) -> u32 { match self[value] { Value::Instruction { instruction, .. } => { + let value_bit_size = self.type_of_value(value).bit_size(); if let Instruction::Cast(original_value, _) = self[instruction] { - self.type_of_value(original_value).bit_size() + let original_bit_size = self.type_of_value(original_value).bit_size(); + // We might have cast e.g. `u1` to `u8` to be able to do arithmetic, + // in which case we want to recover the original smaller bit size; + // OTOH if we cast down, then we don't need the higher original size. + value_bit_size.min(original_bit_size) } else { - self.type_of_value(value).bit_size() + value_bit_size } } @@ -396,17 +533,19 @@ impl DataFlowGraph { value_id } - /// Returns the number of instructions - /// inserted into functions. - pub(crate) fn num_instructions(&self) -> usize { - self.instructions.len() - } - /// Returns all of result values which are attached to this instruction. pub(crate) fn instruction_results(&self, instruction_id: InstructionId) -> &[ValueId] { self.results.get(&instruction_id).expect("expected a list of Values").as_slice() } + /// Remove an instruction by replacing it with a `Noop` instruction. + /// Doing this avoids shifting over each instruction after this one in its block's instructions vector. + #[allow(unused)] + pub(crate) fn remove_instruction(&mut self, instruction: InstructionId) { + self.instructions[instruction] = Instruction::Noop; + self.results.insert(instruction, smallvec::SmallVec::new()); + } + /// Add a parameter to the given block pub(crate) fn add_block_parameter(&mut self, block_id: BasicBlockId, typ: Type) -> ValueId { let block = &mut self.blocks[block_id]; @@ -428,7 +567,7 @@ impl DataFlowGraph { &self, value: ValueId, ) -> Option<(FieldElement, NumericType)> { - match &self.values[self.resolve(value)] { + match &self[self.resolve(value)] { Value::NumericConstant { constant, typ } => Some((*constant, *typ)), _ => None, } @@ -437,13 +576,15 @@ impl DataFlowGraph { /// Returns the Value::Array associated with this ValueId if it refers to an array constant. /// Otherwise, this returns None. pub(crate) fn get_array_constant(&self, value: ValueId) -> Option<(im::Vector, Type)> { - match &self.values[self.resolve(value)] { - Value::Instruction { instruction, .. } => match &self.instructions[*instruction] { + let value = self.resolve(value); + if let Some(instruction) = self.get_local_or_global_instruction(value) { + match instruction { Instruction::MakeArray { elements, typ } => Some((elements.clone(), typ.clone())), _ => None, - }, + } + } else { // Arrays are shared, so cloning them is cheap - _ => None, + None } } @@ -456,6 +597,24 @@ impl DataFlowGraph { } } + /// If this value points to an array of constant bytes, returns a string + /// consisting of those bytes if they form a valid UTF-8 string. + pub(crate) fn get_string(&self, value: ValueId) -> Option { + let (value_ids, _typ) = self.get_array_constant(value)?; + + let mut bytes = Vec::new(); + for value_id in value_ids { + let field_value = self.get_numeric_constant(value_id)?; + let u64_value = field_value.try_to_u64()?; + if u64_value > 255 { + return None; + }; + let byte = u64_value as u8; + bytes.push(byte); + } + String::from_utf8(bytes).ok() + } + /// A constant index less than the array length is safe pub(crate) fn is_safe_index(&self, index: ValueId, array: ValueId) -> bool { #[allow(clippy::match_like_matches_macro)] @@ -496,15 +655,6 @@ impl DataFlowGraph { self.locations.get(&instruction).cloned().unwrap_or_default() } - pub(crate) fn add_location_to_instruction( - &mut self, - instruction: InstructionId, - location: Location, - ) { - let call_stack = self.locations.entry(instruction).or_default(); - *call_stack = self.call_stack_data.add_child(*call_stack, location); - } - pub(crate) fn get_call_stack(&self, call_stack: CallStackId) -> CallStack { self.call_stack_data.get_call_stack(call_stack) } @@ -527,14 +677,23 @@ impl DataFlowGraph { /// True if the given ValueId refers to a (recursively) constant value pub(crate) fn is_constant(&self, argument: ValueId) -> bool { - match &self[self.resolve(argument)] { + let argument = self.resolve(argument); + match &self[argument] { Value::Param { .. } => false, - Value::Instruction { instruction, .. } => match &self[*instruction] { - Instruction::MakeArray { elements, .. } => { - elements.iter().all(|element| self.is_constant(*element)) + Value::Instruction { .. } => { + let Some(instruction) = self.get_local_or_global_instruction(argument) else { + return false; + }; + match &instruction { + Instruction::MakeArray { elements, .. } => { + elements.iter().all(|element| self.is_constant(*element)) + } + _ => false, } - _ => false, - }, + } + Value::Global(_) => { + unreachable!("The global value should have been indexed from the global space"); + } _ => true, } } @@ -547,6 +706,29 @@ impl DataFlowGraph { false } } + + pub(crate) fn is_global(&self, value: ValueId) -> bool { + matches!(self.values[value], Value::Global(_)) + } + + /// Uses value information to determine whether an instruction is from + /// this function's DFG or the global space's DFG. + pub(crate) fn get_local_or_global_instruction(&self, value: ValueId) -> Option<&Instruction> { + match &self[value] { + Value::Instruction { instruction, .. } => { + let instruction = if self.is_global(value) { + let instruction = &self.globals[*instruction]; + // We expect to only have MakeArray instructions in the global space + assert!(matches!(instruction, Instruction::MakeArray { .. })); + instruction + } else { + &self[*instruction] + }; + Some(instruction) + } + _ => None, + } + } } impl std::ops::Index for DataFlowGraph { @@ -565,7 +747,11 @@ impl std::ops::IndexMut for DataFlowGraph { impl std::ops::Index for DataFlowGraph { type Output = Value; fn index(&self, id: ValueId) -> &Self::Output { - &self.values[id] + let value = &self.values[id]; + if matches!(value, Value::Global(_)) { + return &self.globals[id]; + } + value } } @@ -583,6 +769,20 @@ impl std::ops::IndexMut for DataFlowGraph { } } +impl std::ops::Index for GlobalsGraph { + type Output = Value; + fn index(&self, id: ValueId) -> &Self::Output { + &self.values[id] + } +} + +impl std::ops::Index for GlobalsGraph { + type Output = Instruction; + fn index(&self, id: InstructionId) -> &Self::Output { + &self.instructions[id] + } +} + // The result of calling DataFlowGraph::insert_instruction can // be a list of results or a single ValueId if the instruction was simplified // to an existing value. diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/dom.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/dom.rs index ff54bf3b6ed..3dde6240e18 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/dom.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/dom.rs @@ -39,6 +39,7 @@ impl DominatorTreeNode { } /// The dominator tree for a single function. +#[derive(Default)] pub(crate) struct DominatorTree { /// The nodes of the dominator tree /// diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/function.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/function.rs index 6413107c04a..b21a84d16dc 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/function.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/function.rs @@ -1,17 +1,18 @@ use std::collections::BTreeSet; +use std::sync::Arc; use iter_extended::vecmap; use noirc_frontend::monomorphization::ast::InlineType; use serde::{Deserialize, Serialize}; use super::basic_block::BasicBlockId; -use super::dfg::DataFlowGraph; +use super::dfg::{DataFlowGraph, GlobalsGraph}; use super::instruction::TerminatorInstruction; use super::map::Id; use super::types::Type; use super::value::ValueId; -#[derive(Clone, Copy, PartialEq, Eq, Debug, Hash, Serialize, Deserialize)] +#[derive(Clone, Copy, PartialEq, Eq, Debug, Hash, Serialize, Deserialize, PartialOrd, Ord)] pub(crate) enum RuntimeType { // A noir function, to be compiled in ACIR and executed by ACVM Acir(InlineType), @@ -56,6 +57,12 @@ impl RuntimeType { } } +impl Default for RuntimeType { + fn default() -> Self { + RuntimeType::Acir(InlineType::default()) + } +} + /// A function holds a list of instructions. /// These instructions are further grouped into Basic blocks /// @@ -70,9 +77,7 @@ pub(crate) struct Function { /// Name of the function for debugging only name: String, - id: FunctionId, - - runtime: RuntimeType, + id: Option, /// The DataFlowGraph holds the majority of data pertaining to the function /// including its blocks, instructions, and values. @@ -86,20 +91,29 @@ impl Function { pub(crate) fn new(name: String, id: FunctionId) -> Self { let mut dfg = DataFlowGraph::default(); let entry_block = dfg.make_block(); - Self { name, id, entry_block, dfg, runtime: RuntimeType::Acir(InlineType::default()) } + Self { name, id: Some(id), entry_block, dfg } + } + + /// Globals are generated using the same codegen process as functions. + /// To avoid a recursive global context we should create a pseudo function to mock a globals context. + pub(crate) fn new_for_globals() -> Self { + let mut dfg = DataFlowGraph::default(); + let entry_block = dfg.make_block(); + Self { name: "globals".to_owned(), id: None, entry_block, dfg } } /// Creates a new function as a clone of the one passed in with the passed in id. pub(crate) fn clone_with_id(id: FunctionId, another: &Function) -> Self { let dfg = another.dfg.clone(); let entry_block = another.entry_block; - Self { name: another.name.clone(), id, entry_block, dfg, runtime: another.runtime } + Self { name: another.name.clone(), id: Some(id), entry_block, dfg } } /// Takes the signature (function name & runtime) from a function but does not copy the body. pub(crate) fn clone_signature(id: FunctionId, another: &Function) -> Self { let mut new_function = Function::new(another.name.clone(), id); - new_function.runtime = another.runtime; + new_function.set_runtime(another.runtime()); + new_function.set_globals(another.dfg.globals.clone()); new_function } @@ -111,17 +125,21 @@ impl Function { /// The id of the function. pub(crate) fn id(&self) -> FunctionId { - self.id + self.id.expect("FunctionId should be initialized") } /// Runtime type of the function. pub(crate) fn runtime(&self) -> RuntimeType { - self.runtime + self.dfg.runtime() } /// Set runtime type of the function. pub(crate) fn set_runtime(&mut self, runtime: RuntimeType) { - self.runtime = runtime; + self.dfg.set_runtime(runtime); + } + + pub(crate) fn set_globals(&mut self, globals: Arc) { + self.dfg.globals = globals; } pub(crate) fn is_no_predicates(&self) -> bool { @@ -224,12 +242,6 @@ pub(crate) struct Signature { pub(crate) returns: Vec, } -impl std::fmt::Display for Function { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - super::printer::display_function(self, f) - } -} - #[test] fn sign_smoke() { let mut signature = Signature::default(); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction.rs index 3072bb1d72d..5806e62bf95 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction.rs @@ -10,7 +10,7 @@ use fxhash::FxHasher64; use iter_extended::vecmap; use noirc_frontend::hir_def::types::Type as HirType; -use crate::ssa::{ir::function::RuntimeType, opt::flatten_cfg::value_merger::ValueMerger}; +use crate::ssa::opt::flatten_cfg::value_merger::ValueMerger; use super::{ basic_block::BasicBlockId, @@ -22,7 +22,7 @@ use super::{ value::{Value, ValueId}, }; -mod binary; +pub(crate) mod binary; mod call; mod cast; mod constrain; @@ -43,9 +43,9 @@ pub(crate) type InstructionId = Id; /// These are similar to built-ins in other languages. /// These can be classified under two categories: /// - Opcodes which the IR knows the target machine has -/// special support for. (LowLevel) +/// special support for. (LowLevel) /// - Opcodes which have no function definition in the -/// source code and must be processed by the IR. +/// source code and must be processed by the IR. #[derive(Debug, Copy, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)] pub(crate) enum Intrinsic { ArrayLen, @@ -65,8 +65,6 @@ pub(crate) enum Intrinsic { ToRadix(Endian), BlackBox(BlackBoxFunc), Hint(Hint), - FromField, - AsField, AsWitness, IsUnconstrained, DerivePedersenGenerators, @@ -97,8 +95,6 @@ impl std::fmt::Display for Intrinsic { Intrinsic::ToRadix(Endian::Little) => write!(f, "to_le_radix"), Intrinsic::BlackBox(function) => write!(f, "{function}"), Intrinsic::Hint(Hint::BlackBox) => write!(f, "black_box"), - Intrinsic::FromField => write!(f, "from_field"), - Intrinsic::AsField => write!(f, "as_field"), Intrinsic::AsWitness => write!(f, "as_witness"), Intrinsic::IsUnconstrained => write!(f, "is_unconstrained"), Intrinsic::DerivePedersenGenerators => write!(f, "derive_pedersen_generators"), @@ -140,8 +136,6 @@ impl Intrinsic { | Intrinsic::SlicePushFront | Intrinsic::SliceInsert | Intrinsic::StrAsBytes - | Intrinsic::FromField - | Intrinsic::AsField | Intrinsic::IsUnconstrained | Intrinsic::DerivePedersenGenerators | Intrinsic::FieldLessThan => false, @@ -213,8 +207,6 @@ impl Intrinsic { "to_be_radix" => Some(Intrinsic::ToRadix(Endian::Big)), "to_le_bits" => Some(Intrinsic::ToBits(Endian::Little)), "to_be_bits" => Some(Intrinsic::ToBits(Endian::Big)), - "from_field" => Some(Intrinsic::FromField), - "as_field" => Some(Intrinsic::AsField), "as_witness" => Some(Intrinsic::AsWitness), "is_unconstrained" => Some(Intrinsic::IsUnconstrained), "derive_pedersen_generators" => Some(Intrinsic::DerivePedersenGenerators), @@ -264,6 +256,9 @@ pub(crate) enum Instruction { /// Constrains two values to be equal to one another. Constrain(ValueId, ValueId, Option), + /// Constrains two values to not be equal to one another. + ConstrainNotEqual(ValueId, ValueId, Option), + /// Range constrain `value` to `max_bit_size` RangeCheck { value: ValueId, max_bit_size: u32, assert_message: Option }, @@ -343,6 +338,14 @@ pub(crate) enum Instruction { /// `typ` should be an array or slice type with an element type /// matching each of the `elements` values' types. MakeArray { elements: im::Vector, typ: Type }, + + /// A No-op instruction. These are intended to replace other instructions in a block's + /// instructions vector without having to move each instruction afterward. + /// + /// A No-op has no results and is always removed when Instruction::simplify is called. + /// When replacing another instruction, the instruction's results should always be mapped to a + /// new value since they will not be able to refer to their original instruction value any more. + Noop, } impl Instruction { @@ -364,10 +367,12 @@ impl Instruction { InstructionResultType::Operand(*value) } Instruction::Constrain(..) + | Instruction::ConstrainNotEqual(..) | Instruction::Store { .. } | Instruction::IncrementRc { .. } | Instruction::DecrementRc { .. } | Instruction::RangeCheck { .. } + | Instruction::Noop | Instruction::EnableSideEffectsIf { .. } => InstructionResultType::None, Instruction::Allocate { .. } | Instruction::Load { .. } @@ -404,17 +409,22 @@ impl Instruction { }, // These can fail. - Constrain(..) | RangeCheck { .. } => true, + Constrain(..) | ConstrainNotEqual(..) | RangeCheck { .. } => true, // This should never be side-effectful - MakeArray { .. } => false, + MakeArray { .. } | Noop => false, // Some binary math can overflow or underflow Binary(binary) => match binary.operator { - BinaryOp::Add | BinaryOp::Sub | BinaryOp::Mul | BinaryOp::Div | BinaryOp::Mod => { - true - } - BinaryOp::Eq + BinaryOp::Add { unchecked: false } + | BinaryOp::Sub { unchecked: false } + | BinaryOp::Mul { unchecked: false } + | BinaryOp::Div + | BinaryOp::Mod => true, + BinaryOp::Add { unchecked: true } + | BinaryOp::Sub { unchecked: true } + | BinaryOp::Mul { unchecked: true } + | BinaryOp::Eq | BinaryOp::Lt | BinaryOp::And | BinaryOp::Or @@ -466,7 +476,14 @@ impl Instruction { }, // We can deduplicate these instructions if we know the predicate is also the same. - Constrain(..) | RangeCheck { .. } => deduplicate_with_predicate, + Constrain(..) | ConstrainNotEqual(..) | RangeCheck { .. } => deduplicate_with_predicate, + + // Noop instructions can always be deduplicated, although they're more likely to be + // removed entirely. + Noop => true, + + // Cast instructions can always be deduplicated + Cast(_, _) => true, // Arrays can be mutated in unconstrained code so code that handles this case must // take care to track whether the array was possibly mutated or not before @@ -479,7 +496,6 @@ impl Instruction { // with one that was disabled. See // https://github.com/noir-lang/noir/pull/4716#issuecomment-2047846328. Binary(_) - | Cast(_, _) | Not(_) | Truncate { .. } | IfElse { .. } @@ -490,7 +506,7 @@ impl Instruction { } } - pub(crate) fn can_eliminate_if_unused(&self, function: &Function) -> bool { + pub(crate) fn can_eliminate_if_unused(&self, function: &Function, flattened: bool) -> bool { use Instruction::*; match self { Binary(binary) => { @@ -512,6 +528,7 @@ impl Instruction { | ArrayGet { .. } | IfElse { .. } | ArraySet { .. } + | Noop | MakeArray { .. } => true, // Store instructions must be removed by DIE in acir code, any load @@ -522,11 +539,11 @@ impl Instruction { // pass where this check is done, but does mean that we cannot perform mem2reg // after the DIE pass. Store { .. } => { - matches!(function.runtime(), RuntimeType::Acir(_)) - && function.reachable_blocks().len() == 1 + flattened && function.runtime().is_acir() && function.reachable_blocks().len() == 1 } Constrain(..) + | ConstrainNotEqual(..) | EnableSideEffectsIf { .. } | IncrementRc { .. } | DecrementRc { .. } @@ -556,10 +573,28 @@ impl Instruction { /// If true the instruction will depend on `enable_side_effects` context during acir-gen. pub(crate) fn requires_acir_gen_predicate(&self, dfg: &DataFlowGraph) -> bool { match self { - Instruction::Binary(binary) - if matches!(binary.operator, BinaryOp::Div | BinaryOp::Mod) => - { - true + Instruction::Binary(binary) => { + match binary.operator { + BinaryOp::Add { unchecked: false } + | BinaryOp::Sub { unchecked: false } + | BinaryOp::Mul { unchecked: false } + | BinaryOp::Div + | BinaryOp::Mod => { + // Some binary math can overflow or underflow, but this is only the case + // for unsigned types (here we assume the type of binary.lhs is the same) + dfg.type_of_value(binary.rhs).is_unsigned() + } + BinaryOp::Add { unchecked: true } + | BinaryOp::Sub { unchecked: true } + | BinaryOp::Mul { unchecked: true } + | BinaryOp::Eq + | BinaryOp::Lt + | BinaryOp::And + | BinaryOp::Or + | BinaryOp::Xor + | BinaryOp::Shl + | BinaryOp::Shr => false, + } } Instruction::ArrayGet { array, index } => { @@ -577,9 +612,9 @@ impl Instruction { _ => false, }, Instruction::Cast(_, _) - | Instruction::Binary(_) | Instruction::Not(_) | Instruction::Truncate { .. } + | Instruction::ConstrainNotEqual(..) | Instruction::Constrain(_, _, _) | Instruction::RangeCheck { .. } | Instruction::Allocate @@ -588,6 +623,7 @@ impl Instruction { | Instruction::IfElse { .. } | Instruction::IncrementRc { .. } | Instruction::DecrementRc { .. } + | Instruction::Noop | Instruction::MakeArray { .. } => false, } } @@ -625,6 +661,22 @@ impl Instruction { }); Instruction::Constrain(lhs, rhs, assert_message) } + Instruction::ConstrainNotEqual(lhs, rhs, assert_message) => { + // Must map the `lhs` and `rhs` first as the value `f` is moved with the closure + let lhs = f(*lhs); + let rhs = f(*rhs); + let assert_message = assert_message.as_ref().map(|error| match error { + ConstrainError::Dynamic(selector, is_string, payload_values) => { + ConstrainError::Dynamic( + *selector, + *is_string, + payload_values.iter().map(|&value| f(value)).collect(), + ) + } + _ => error.clone(), + }); + Instruction::ConstrainNotEqual(lhs, rhs, assert_message) + } Instruction::Call { func, arguments } => Instruction::Call { func: f(*func), arguments: vecmap(arguments.iter().copied(), f), @@ -667,6 +719,7 @@ impl Instruction { elements: elements.iter().copied().map(f).collect(), typ: typ.clone(), }, + Instruction::Noop => Instruction::Noop, } } @@ -682,7 +735,8 @@ impl Instruction { Instruction::Truncate { value, bit_size: _, max_bit_size: _ } => { *value = f(*value); } - Instruction::Constrain(lhs, rhs, assert_message) => { + Instruction::Constrain(lhs, rhs, assert_message) + | Instruction::ConstrainNotEqual(lhs, rhs, assert_message) => { *lhs = f(*lhs); *rhs = f(*rhs); if let Some(ConstrainError::Dynamic(_, _, payload_values)) = assert_message { @@ -731,6 +785,7 @@ impl Instruction { *element = f(*element); } } + Instruction::Noop => (), } } @@ -753,7 +808,8 @@ impl Instruction { | Instruction::Load { address: value } => { f(*value); } - Instruction::Constrain(lhs, rhs, assert_error) => { + Instruction::Constrain(lhs, rhs, assert_error) + | Instruction::ConstrainNotEqual(lhs, rhs, assert_error) => { f(*lhs); f(*rhs); if let Some(ConstrainError::Dynamic(_, _, values)) = assert_error.as_ref() { @@ -796,6 +852,7 @@ impl Instruction { f(*element); } } + Instruction::Noop => (), } } @@ -844,6 +901,7 @@ impl Instruction { SimplifiedToInstructionMultiple(constraints) } } + Instruction::ConstrainNotEqual(..) => None, Instruction::ArrayGet { array, index } => { if let Some(index) = dfg.get_numeric_constant(*index) { try_optimize_array_get_from_previous_set(dfg, *array, index) @@ -952,9 +1010,11 @@ impl Instruction { } } Instruction::IfElse { then_condition, then_value, else_condition, else_value } => { + let then_condition = dfg.resolve(*then_condition); + let else_condition = dfg.resolve(*else_condition); let typ = dfg.type_of_value(*then_value); - if let Some(constant) = dfg.get_numeric_constant(*then_condition) { + if let Some(constant) = dfg.get_numeric_constant(then_condition) { if constant.is_one() { return SimplifiedTo(*then_value); } else if constant.is_zero() { @@ -968,10 +1028,51 @@ impl Instruction { return SimplifiedTo(then_value); } - if matches!(&typ, Type::Numeric(_)) { - let then_condition = *then_condition; - let else_condition = *else_condition; + if let Value::Instruction { instruction, .. } = &dfg[then_value] { + if let Instruction::IfElse { + then_condition: inner_then_condition, + then_value: inner_then_value, + else_condition: inner_else_condition, + .. + } = dfg[*instruction] + { + if then_condition == inner_then_condition { + let instruction = Instruction::IfElse { + then_condition, + then_value: inner_then_value, + else_condition: inner_else_condition, + else_value, + }; + return SimplifiedToInstruction(instruction); + } + // TODO: We could check to see if `then_condition == inner_else_condition` + // but we run into issues with duplicate NOT instructions having distinct ValueIds. + } + }; + + if let Value::Instruction { instruction, .. } = &dfg[else_value] { + if let Instruction::IfElse { + then_condition: inner_then_condition, + else_condition: inner_else_condition, + else_value: inner_else_value, + .. + } = dfg[*instruction] + { + if then_condition == inner_then_condition { + let instruction = Instruction::IfElse { + then_condition, + then_value, + else_condition: inner_else_condition, + else_value: inner_else_value, + }; + return SimplifiedToInstruction(instruction); + } + // TODO: We could check to see if `then_condition == inner_else_condition` + // but we run into issues with duplicate NOT instructions having distinct ValueIds. + } + }; + if matches!(&typ, Type::Numeric(_)) { let result = ValueMerger::merge_numeric_values( dfg, block, @@ -986,6 +1087,7 @@ impl Instruction { } } Instruction::MakeArray { .. } => None, + Instruction::Noop => Remove, } } } @@ -1015,28 +1117,27 @@ fn try_optimize_array_get_from_previous_set( // Arbitrary number of maximum tries just to prevent this optimization from taking too long. let max_tries = 5; for _ in 0..max_tries { - match &dfg[array_id] { - Value::Instruction { instruction, .. } => { - match &dfg[*instruction] { - Instruction::ArraySet { array, index, value, .. } => { - if let Some(constant) = dfg.get_numeric_constant(*index) { - if constant == target_index { - return SimplifyResult::SimplifiedTo(*value); - } - - array_id = *array; // recur - } else { - return SimplifyResult::None; + if let Some(instruction) = dfg.get_local_or_global_instruction(array_id) { + match instruction { + Instruction::ArraySet { array, index, value, .. } => { + if let Some(constant) = dfg.get_numeric_constant(*index) { + if constant == target_index { + return SimplifyResult::SimplifiedTo(*value); } + + array_id = *array; // recur + } else { + return SimplifyResult::None; } - Instruction::MakeArray { elements: array, typ: _ } => { - elements = Some(array.clone()); - break; - } - _ => return SimplifyResult::None, } + Instruction::MakeArray { elements: array, typ: _ } => { + elements = Some(array.clone()); + break; + } + _ => return SimplifyResult::None, } - _ => return SimplifyResult::None, + } else { + return SimplifyResult::None; } } @@ -1232,31 +1333,6 @@ pub(crate) enum TerminatorInstruction { } impl TerminatorInstruction { - /// Map each ValueId in this terminator to a new value. - pub(crate) fn map_values( - &self, - mut f: impl FnMut(ValueId) -> ValueId, - ) -> TerminatorInstruction { - use TerminatorInstruction::*; - match self { - JmpIf { condition, then_destination, else_destination, call_stack } => JmpIf { - condition: f(*condition), - then_destination: *then_destination, - else_destination: *else_destination, - call_stack: *call_stack, - }, - Jmp { destination, arguments, call_stack } => Jmp { - destination: *destination, - arguments: vecmap(arguments, |value| f(*value)), - call_stack: *call_stack, - }, - Return { return_values, call_stack } => Return { - return_values: vecmap(return_values, |value| f(*value)), - call_stack: *call_stack, - }, - } - } - /// Mutate each ValueId to a new ValueId using the given mapping function pub(crate) fn map_values_mut(&mut self, mut f: impl FnMut(ValueId) -> ValueId) { use TerminatorInstruction::*; diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/binary.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/binary.rs index 81f2f3b1e01..df1e8f537da 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/binary.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/binary.rs @@ -15,11 +15,11 @@ use super::{ #[derive(Debug, PartialEq, Eq, Hash, Copy, Clone, Serialize, Deserialize)] pub(crate) enum BinaryOp { /// Addition of lhs + rhs. - Add, + Add { unchecked: bool }, /// Subtraction of lhs - rhs. - Sub, + Sub { unchecked: bool }, /// Multiplication of lhs * rhs. - Mul, + Mul { unchecked: bool }, /// Division of lhs / rhs. Div, /// Modulus of lhs % rhs. @@ -48,9 +48,12 @@ pub(crate) enum BinaryOp { impl std::fmt::Display for BinaryOp { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { - BinaryOp::Add => write!(f, "add"), - BinaryOp::Sub => write!(f, "sub"), - BinaryOp::Mul => write!(f, "mul"), + BinaryOp::Add { unchecked: false } => write!(f, "add"), + BinaryOp::Add { unchecked: true } => write!(f, "unchecked_add"), + BinaryOp::Sub { unchecked: false } => write!(f, "sub"), + BinaryOp::Sub { unchecked: true } => write!(f, "unchecked_sub"), + BinaryOp::Mul { unchecked: false } => write!(f, "mul"), + BinaryOp::Mul { unchecked: true } => write!(f, "unchecked_mul"), BinaryOp::Div => write!(f, "div"), BinaryOp::Eq => write!(f, "eq"), BinaryOp::Mod => write!(f, "mod"), @@ -86,28 +89,61 @@ impl Binary { /// Try to simplify this binary instruction, returning the new value if possible. pub(super) fn simplify(&self, dfg: &mut DataFlowGraph) -> SimplifyResult { - let lhs = dfg.get_numeric_constant(self.lhs); - let rhs = dfg.get_numeric_constant(self.rhs); - let operand_type = dfg.type_of_value(self.lhs).unwrap_numeric(); + let lhs_value = dfg.get_numeric_constant(self.lhs); + let rhs_value = dfg.get_numeric_constant(self.rhs); + + let lhs_type = dfg.type_of_value(self.lhs).unwrap_numeric(); + let rhs_type = dfg.type_of_value(self.rhs).unwrap_numeric(); + + let operator = self.operator; + if operator != BinaryOp::Shl && operator != BinaryOp::Shr { + assert_eq!( + lhs_type, rhs_type, + "ICE - Binary instruction operands must have the same type" + ); + } - if let (Some(lhs), Some(rhs)) = (lhs, rhs) { - return match eval_constant_binary_op(lhs, rhs, self.operator, operand_type) { + let operator = if lhs_type == NumericType::NativeField { + // Unchecked operations between fields or bools don't make sense, so we convert those to non-unchecked + // to reduce noise and confusion in the generated SSA. + match operator { + BinaryOp::Add { unchecked: true } => BinaryOp::Add { unchecked: false }, + BinaryOp::Sub { unchecked: true } => BinaryOp::Sub { unchecked: false }, + BinaryOp::Mul { unchecked: true } => BinaryOp::Mul { unchecked: false }, + _ => operator, + } + } else if lhs_type == NumericType::bool() { + // Unchecked mul between bools doesn't make sense, so we convert that to non-unchecked + if let BinaryOp::Mul { unchecked: true } = operator { + BinaryOp::Mul { unchecked: false } + } else { + operator + } + } else { + operator + }; + + // We never return `SimplifyResult::None` here because `operator` might have changed. + let simplified = Instruction::Binary(Binary { lhs: self.lhs, rhs: self.rhs, operator }); + + if let (Some(lhs), Some(rhs)) = (lhs_value, rhs_value) { + return match eval_constant_binary_op(lhs, rhs, operator, lhs_type) { Some((result, result_type)) => { let value = dfg.make_constant(result, result_type); SimplifyResult::SimplifiedTo(value) } - None => SimplifyResult::None, + None => SimplifyResult::SimplifiedToInstruction(simplified), }; } - let lhs_is_zero = lhs.map_or(false, |lhs| lhs.is_zero()); - let rhs_is_zero = rhs.map_or(false, |rhs| rhs.is_zero()); + let lhs_is_zero = lhs_value.map_or(false, |lhs| lhs.is_zero()); + let rhs_is_zero = rhs_value.map_or(false, |rhs| rhs.is_zero()); - let lhs_is_one = lhs.map_or(false, |lhs| lhs.is_one()); - let rhs_is_one = rhs.map_or(false, |rhs| rhs.is_one()); + let lhs_is_one = lhs_value.map_or(false, |lhs| lhs.is_one()); + let rhs_is_one = rhs_value.map_or(false, |rhs| rhs.is_one()); match self.operator { - BinaryOp::Add => { + BinaryOp::Add { .. } => { if lhs_is_zero { return SimplifyResult::SimplifiedTo(self.rhs); } @@ -115,12 +151,12 @@ impl Binary { return SimplifyResult::SimplifiedTo(self.lhs); } } - BinaryOp::Sub => { + BinaryOp::Sub { .. } => { if rhs_is_zero { return SimplifyResult::SimplifiedTo(self.lhs); } } - BinaryOp::Mul => { + BinaryOp::Mul { .. } => { if lhs_is_one { return SimplifyResult::SimplifiedTo(self.rhs); } @@ -128,14 +164,42 @@ impl Binary { return SimplifyResult::SimplifiedTo(self.lhs); } if lhs_is_zero || rhs_is_zero { - let zero = dfg.make_constant(FieldElement::zero(), operand_type); + let zero = dfg.make_constant(FieldElement::zero(), lhs_type); return SimplifyResult::SimplifiedTo(zero); } - if dfg.resolve(self.lhs) == dfg.resolve(self.rhs) - && dfg.get_value_max_num_bits(self.lhs) == 1 - { + if dfg.get_value_max_num_bits(self.lhs) == 1 { // Squaring a boolean value is a noop. - return SimplifyResult::SimplifiedTo(self.lhs); + if dfg.resolve(self.lhs) == dfg.resolve(self.rhs) { + return SimplifyResult::SimplifiedTo(self.lhs); + } + // b*(b*x) = b*x if b is boolean + if let super::Value::Instruction { instruction, .. } = &dfg[self.rhs] { + if let Instruction::Binary(Binary { lhs, rhs, operator }) = + dfg[*instruction] + { + if matches!(operator, BinaryOp::Mul { .. }) + && (dfg.resolve(self.lhs) == dfg.resolve(lhs) + || dfg.resolve(self.lhs) == dfg.resolve(rhs)) + { + return SimplifyResult::SimplifiedTo(self.rhs); + } + } + } + } + // (b*x)*b = b*x if b is boolean + if dfg.get_value_max_num_bits(self.rhs) == 1 { + if let super::Value::Instruction { instruction, .. } = &dfg[self.lhs] { + if let Instruction::Binary(Binary { lhs, rhs, operator }) = + dfg[*instruction] + { + if matches!(operator, BinaryOp::Mul { .. }) + && (dfg.resolve(self.rhs) == dfg.resolve(lhs) + || dfg.resolve(self.rhs) == dfg.resolve(rhs)) + { + return SimplifyResult::SimplifiedTo(self.lhs); + } + } + } } } BinaryOp::Div => { @@ -145,13 +209,13 @@ impl Binary { } BinaryOp::Mod => { if rhs_is_one { - let zero = dfg.make_constant(FieldElement::zero(), operand_type); + let zero = dfg.make_constant(FieldElement::zero(), lhs_type); return SimplifyResult::SimplifiedTo(zero); } - if operand_type.is_unsigned() { + if lhs_type.is_unsigned() { // lhs % 2**bit_size is equivalent to truncating `lhs` to `bit_size` bits. // We then convert to a truncation for consistency, allowing more optimizations. - if let Some(modulus) = rhs { + if let Some(modulus) = rhs_value { let modulus = modulus.to_u128(); if modulus.is_power_of_two() { let bit_size = modulus.ilog2(); @@ -159,7 +223,7 @@ impl Binary { Instruction::Truncate { value: self.lhs, bit_size, - max_bit_size: operand_type.bit_size(), + max_bit_size: lhs_type.bit_size(), }, ); } @@ -172,7 +236,7 @@ impl Binary { return SimplifyResult::SimplifiedTo(one); } - if operand_type == NumericType::bool() { + if lhs_type == NumericType::bool() { // Simplify forms of `(boolean == true)` into `boolean` if lhs_is_one { return SimplifyResult::SimplifiedTo(self.rhs); @@ -194,13 +258,13 @@ impl Binary { let zero = dfg.make_constant(FieldElement::zero(), NumericType::bool()); return SimplifyResult::SimplifiedTo(zero); } - if operand_type.is_unsigned() { + if lhs_type.is_unsigned() { if rhs_is_zero { // Unsigned values cannot be less than zero. let zero = dfg.make_constant(FieldElement::zero(), NumericType::bool()); return SimplifyResult::SimplifiedTo(zero); } else if rhs_is_one { - let zero = dfg.make_constant(FieldElement::zero(), operand_type); + let zero = dfg.make_constant(FieldElement::zero(), lhs_type); return SimplifyResult::SimplifiedToInstruction(Instruction::binary( BinaryOp::Eq, self.lhs, @@ -211,35 +275,37 @@ impl Binary { } BinaryOp::And => { if lhs_is_zero || rhs_is_zero { - let zero = dfg.make_constant(FieldElement::zero(), operand_type); + let zero = dfg.make_constant(FieldElement::zero(), lhs_type); return SimplifyResult::SimplifiedTo(zero); } if dfg.resolve(self.lhs) == dfg.resolve(self.rhs) { return SimplifyResult::SimplifiedTo(self.lhs); } - if operand_type == NumericType::bool() { + if lhs_type == NumericType::bool() { // Boolean AND is equivalent to multiplication, which is a cheaper operation. - let instruction = Instruction::binary(BinaryOp::Mul, self.lhs, self.rhs); + // (mul unchecked because these are bools so it doesn't matter really) + let instruction = + Instruction::binary(BinaryOp::Mul { unchecked: true }, self.lhs, self.rhs); return SimplifyResult::SimplifiedToInstruction(instruction); } - if operand_type.is_unsigned() { + if lhs_type.is_unsigned() { // It's common in other programming languages to truncate values to a certain bit size using // a bitwise AND with a bit mask. However this operation is quite inefficient inside a snark. // // We then replace this bitwise operation with an equivalent truncation instruction. - match (lhs, rhs) { + match (lhs_value, rhs_value) { (Some(bitmask), None) | (None, Some(bitmask)) => { // This substitution requires the bitmask to retain all of the lower bits. // The bitmask must then be one less than a power of 2. let bitmask_plus_one = bitmask.to_u128() + 1; if bitmask_plus_one.is_power_of_two() { - let value = if lhs.is_some() { self.rhs } else { self.lhs }; + let value = if lhs_value.is_some() { self.rhs } else { self.lhs }; let num_bits = bitmask_plus_one.ilog2(); return SimplifyResult::SimplifiedToInstruction( Instruction::Truncate { value, bit_size: num_bits, - max_bit_size: operand_type.bit_size(), + max_bit_size: lhs_type.bit_size(), }, ); } @@ -256,8 +322,8 @@ impl Binary { if rhs_is_zero { return SimplifyResult::SimplifiedTo(self.lhs); } - if operand_type == NumericType::bool() && (lhs_is_one || rhs_is_one) { - let one = dfg.make_constant(FieldElement::one(), operand_type); + if lhs_type == NumericType::bool() && (lhs_is_one || rhs_is_one) { + let one = dfg.make_constant(FieldElement::one(), lhs_type); return SimplifyResult::SimplifiedTo(one); } if dfg.resolve(self.lhs) == dfg.resolve(self.rhs) { @@ -272,29 +338,72 @@ impl Binary { return SimplifyResult::SimplifiedTo(self.lhs); } if dfg.resolve(self.lhs) == dfg.resolve(self.rhs) { - let zero = dfg.make_constant(FieldElement::zero(), operand_type); + let zero = dfg.make_constant(FieldElement::zero(), lhs_type); return SimplifyResult::SimplifiedTo(zero); } } - BinaryOp::Shl => return SimplifyResult::None, + BinaryOp::Shl => return SimplifyResult::SimplifiedToInstruction(simplified), BinaryOp::Shr => { // Bit shifts by constants can be treated as divisions. - if let Some(rhs_const) = rhs { - if rhs_const >= FieldElement::from(operand_type.bit_size() as u128) { + if let Some(rhs_const) = rhs_value { + if rhs_const >= FieldElement::from(lhs_type.bit_size() as u128) { // Shifting by the full width of the operand type, any `lhs` goes to zero. - let zero = dfg.make_constant(FieldElement::zero(), operand_type); + let zero = dfg.make_constant(FieldElement::zero(), lhs_type); return SimplifyResult::SimplifiedTo(zero); } - return SimplifyResult::None; + return SimplifyResult::SimplifiedToInstruction(simplified); + } + } + }; + SimplifyResult::SimplifiedToInstruction(simplified) + } + + /// Check if unsigned overflow is possible, and if so return some message to be used if it fails. + pub(crate) fn check_unsigned_overflow_msg( + &self, + dfg: &DataFlowGraph, + bit_size: u32, + ) -> Option<&'static str> { + // We try to optimize away operations that are guaranteed not to overflow + let max_lhs_bits = dfg.get_value_max_num_bits(self.lhs); + let max_rhs_bits = dfg.get_value_max_num_bits(self.rhs); + + let msg = match self.operator { + BinaryOp::Add { unchecked: false } => { + if std::cmp::max(max_lhs_bits, max_rhs_bits) < bit_size { + // `lhs` and `rhs` have both been casted up from smaller types and so cannot overflow. + return None; + } + "attempt to add with overflow" + } + BinaryOp::Sub { unchecked: false } => { + if dfg.is_constant(self.lhs) && max_lhs_bits > max_rhs_bits { + // `lhs` is a fixed constant and `rhs` is restricted such that `lhs - rhs > 0` + // Note strict inequality as `rhs > lhs` while `max_lhs_bits == max_rhs_bits` is possible. + return None; + } + "attempt to subtract with overflow" + } + BinaryOp::Mul { unchecked: false } => { + if bit_size == 1 + || max_lhs_bits + max_rhs_bits <= bit_size + || max_lhs_bits == 1 + || max_rhs_bits == 1 + { + // Either performing boolean multiplication (which cannot overflow), + // or `lhs` and `rhs` have both been casted up from smaller types and so cannot overflow. + return None; } + "attempt to multiply with overflow" } + _ => return None, }; - SimplifyResult::None + Some(msg) } } /// Evaluate a binary operation with constant arguments. -fn eval_constant_binary_op( +pub(crate) fn eval_constant_binary_op( lhs: FieldElement, rhs: FieldElement, operator: BinaryOp, @@ -399,9 +508,9 @@ fn truncate(int: u128, bit_size: u32) -> u128 { impl BinaryOp { fn get_field_function(self) -> Option FieldElement> { match self { - BinaryOp::Add => Some(std::ops::Add::add), - BinaryOp::Sub => Some(std::ops::Sub::sub), - BinaryOp::Mul => Some(std::ops::Mul::mul), + BinaryOp::Add { .. } => Some(std::ops::Add::add), + BinaryOp::Sub { .. } => Some(std::ops::Sub::sub), + BinaryOp::Mul { .. } => Some(std::ops::Mul::mul), BinaryOp::Div => Some(std::ops::Div::div), BinaryOp::Eq => Some(|x, y| (x == y).into()), BinaryOp::Lt => Some(|x, y| (x < y).into()), @@ -417,9 +526,9 @@ impl BinaryOp { fn get_u128_function(self) -> fn(u128, u128) -> Option { match self { - BinaryOp::Add => u128::checked_add, - BinaryOp::Sub => u128::checked_sub, - BinaryOp::Mul => u128::checked_mul, + BinaryOp::Add { .. } => u128::checked_add, + BinaryOp::Sub { .. } => u128::checked_sub, + BinaryOp::Mul { .. } => u128::checked_mul, BinaryOp::Div => u128::checked_div, BinaryOp::Mod => u128::checked_rem, BinaryOp::And => |x, y| Some(x & y), @@ -434,9 +543,9 @@ impl BinaryOp { fn get_i128_function(self) -> fn(i128, i128) -> Option { match self { - BinaryOp::Add => i128::checked_add, - BinaryOp::Sub => i128::checked_sub, - BinaryOp::Mul => i128::checked_mul, + BinaryOp::Add { .. } => i128::checked_add, + BinaryOp::Sub { .. } => i128::checked_sub, + BinaryOp::Mul { .. } => i128::checked_mul, BinaryOp::Div => i128::checked_div, BinaryOp::Mod => i128::checked_rem, BinaryOp::And => |x, y| Some(x & y), diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/call.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/call.rs index fa7d314ff33..992c633ffcd 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/call.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/call.rs @@ -3,7 +3,7 @@ use std::{collections::VecDeque, sync::Arc}; use acvm::{ acir::{AcirField, BlackBoxFunc}, - BlackBoxResolutionError, FieldElement, + FieldElement, }; use bn254_blackbox_solver::derive_generators; use iter_extended::vecmap; @@ -142,8 +142,7 @@ pub(super) fn simplify_call( slice.push_back(*elem); } - let new_slice_length = - update_slice_length(arguments[0], dfg, BinaryOp::Add, block); + let new_slice_length = increment_slice_length(arguments[0], dfg, block); let new_slice = make_array(dfg, slice, element_type, block, call_stack); return SimplifyResult::SimplifiedToMultiple(vec![new_slice_length, new_slice]); @@ -161,7 +160,7 @@ pub(super) fn simplify_call( slice.push_front(*elem); } - let new_slice_length = update_slice_length(arguments[0], dfg, BinaryOp::Add, block); + let new_slice_length = increment_slice_length(arguments[0], dfg, block); let new_slice = make_array(dfg, slice, element_type, block, call_stack); SimplifyResult::SimplifiedToMultiple(vec![new_slice_length, new_slice]) @@ -201,7 +200,7 @@ pub(super) fn simplify_call( slice.pop_front().expect("There are no elements in this slice to be removed") }); - let new_slice_length = update_slice_length(arguments[0], dfg, BinaryOp::Sub, block); + let new_slice_length = decrement_slice_length(arguments[0], dfg, block); results.push(new_slice_length); @@ -234,7 +233,7 @@ pub(super) fn simplify_call( index += 1; } - let new_slice_length = update_slice_length(arguments[0], dfg, BinaryOp::Add, block); + let new_slice_length = increment_slice_length(arguments[0], dfg, block); let new_slice = make_array(dfg, slice, typ, block, call_stack); SimplifyResult::SimplifiedToMultiple(vec![new_slice_length, new_slice]) @@ -272,7 +271,7 @@ pub(super) fn simplify_call( let new_slice = make_array(dfg, slice, typ, block, call_stack); results.insert(0, new_slice); - let new_slice_length = update_slice_length(arguments[0], dfg, BinaryOp::Sub, block); + let new_slice_length = decrement_slice_length(arguments[0], dfg, block); results.insert(0, new_slice_length); @@ -330,33 +329,11 @@ pub(super) fn simplify_call( Intrinsic::BlackBox(bb_func) => { simplify_black_box_func(bb_func, arguments, dfg, block, call_stack) } - Intrinsic::AsField => { - let instruction = Instruction::Cast(arguments[0], NumericType::NativeField); - SimplifyResult::SimplifiedToInstruction(instruction) - } - Intrinsic::FromField => { - let incoming_type = Type::field(); - let target_type = return_type.clone().unwrap(); - - let truncate = Instruction::Truncate { - value: arguments[0], - bit_size: target_type.bit_size(), - max_bit_size: incoming_type.bit_size(), - }; - let truncated_value = dfg - .insert_instruction_and_results( - truncate, - block, - Some(vec![incoming_type]), - call_stack, - ) - .first(); - - let instruction = Instruction::Cast(truncated_value, target_type.unwrap_numeric()); - SimplifyResult::SimplifiedToInstruction(instruction) - } Intrinsic::AsWitness => SimplifyResult::None, - Intrinsic::IsUnconstrained => SimplifyResult::None, + Intrinsic::IsUnconstrained => { + let result = dfg.runtime().is_brillig().into(); + SimplifyResult::SimplifiedTo(dfg.make_constant(result, NumericType::bool())) + } Intrinsic::DerivePedersenGenerators => { if let Some(Type::Array(_, len)) = return_type.clone() { simplify_derive_generators(dfg, arguments, len, block, call_stack) @@ -410,6 +387,22 @@ fn update_slice_length( dfg.insert_instruction_and_results(instruction, block, None, call_stack).first() } +fn increment_slice_length( + slice_len: ValueId, + dfg: &mut DataFlowGraph, + block: BasicBlockId, +) -> ValueId { + update_slice_length(slice_len, dfg, BinaryOp::Add { unchecked: false }, block) +} + +fn decrement_slice_length( + slice_len: ValueId, + dfg: &mut DataFlowGraph, + block: BasicBlockId, +) -> ValueId { + update_slice_length(slice_len, dfg, BinaryOp::Sub { unchecked: true }, block) +} + fn simplify_slice_push_back( mut slice: im::Vector, element_type: Type, @@ -430,7 +423,7 @@ fn simplify_slice_push_back( .insert_instruction_and_results(len_not_equals_capacity_instr, block, None, call_stack) .first(); - let new_slice_length = update_slice_length(arguments[0], dfg, BinaryOp::Add, block); + let new_slice_length = increment_slice_length(arguments[0], dfg, block); for elem in &arguments[2..] { slice.push_back(*elem); @@ -479,14 +472,17 @@ fn simplify_slice_pop_back( let element_count = element_types.len(); let mut results = VecDeque::with_capacity(element_count + 1); - let new_slice_length = update_slice_length(arguments[0], dfg, BinaryOp::Sub, block); + let new_slice_length = decrement_slice_length(arguments[0], dfg, block); let element_size = dfg.make_constant((element_count as u128).into(), NumericType::length_type()); - let flattened_len_instr = Instruction::binary(BinaryOp::Mul, arguments[0], element_size); + // Compute the flattened length doing an unchecked mul + // (it shouldn't overflow because it would have overflowed before when the slice was created) + let flattened_len_instr = + Instruction::binary(BinaryOp::Mul { unchecked: true }, arguments[0], element_size); let mut flattened_len = dfg.insert_instruction_and_results(flattened_len_instr, block, None, call_stack).first(); - flattened_len = update_slice_length(flattened_len, dfg, BinaryOp::Sub, block); + flattened_len = decrement_slice_length(flattened_len, dfg, block); // We must pop multiple elements in the case of a slice of tuples // Iterating through element types in reverse here since we're popping from the end @@ -500,7 +496,7 @@ fn simplify_slice_pop_back( .first(); results.push_front(get_last_elem); - flattened_len = update_slice_length(flattened_len, dfg, BinaryOp::Sub, block); + flattened_len = decrement_slice_length(flattened_len, dfg, block); } results.push_front(arguments[1]); @@ -518,20 +514,29 @@ fn simplify_black_box_func( block: BasicBlockId, call_stack: CallStackId, ) -> SimplifyResult { + let pedantic_solving = true; cfg_if::cfg_if! { if #[cfg(feature = "bn254")] { - let solver = bn254_blackbox_solver::Bn254BlackBoxSolver; + let solver = bn254_blackbox_solver::Bn254BlackBoxSolver(pedantic_solving); } else { - let solver = acvm::blackbox_solver::StubbedBlackBoxSolver; + let solver = acvm::blackbox_solver::StubbedBlackBoxSolver(pedantic_solving); } }; match bb_func { - BlackBoxFunc::Blake2s => { - simplify_hash(dfg, arguments, acvm::blackbox_solver::blake2s, block, call_stack) - } - BlackBoxFunc::Blake3 => { - simplify_hash(dfg, arguments, acvm::blackbox_solver::blake3, block, call_stack) - } + BlackBoxFunc::Blake2s => blackbox::simplify_hash( + dfg, + arguments, + acvm::blackbox_solver::blake2s, + block, + call_stack, + ), + BlackBoxFunc::Blake3 => blackbox::simplify_hash( + dfg, + arguments, + acvm::blackbox_solver::blake3, + block, + call_stack, + ), BlackBoxFunc::Keccakf1600 => { if let Some((array_input, _)) = dfg.get_array_constant(arguments[0]) { if array_is_constant(dfg, &array_input) { @@ -683,78 +688,6 @@ fn array_is_constant(dfg: &DataFlowGraph, values: &im::Vector>) -> boo values.iter().all(|value| dfg.get_numeric_constant(*value).is_some()) } -fn simplify_hash( - dfg: &mut DataFlowGraph, - arguments: &[ValueId], - hash_function: fn(&[u8]) -> Result<[u8; 32], BlackBoxResolutionError>, - block: BasicBlockId, - call_stack: CallStackId, -) -> SimplifyResult { - match dfg.get_array_constant(arguments[0]) { - Some((input, _)) if array_is_constant(dfg, &input) => { - let input_bytes: Vec = to_u8_vec(dfg, input); - - let hash = hash_function(&input_bytes) - .expect("Rust solvable black box function should not fail"); - - let hash_values = hash.iter().map(|byte| FieldElement::from_be_bytes_reduce(&[*byte])); - - let u8_type = NumericType::Unsigned { bit_size: 8 }; - let result_array = make_constant_array(dfg, hash_values, u8_type, block, call_stack); - SimplifyResult::SimplifiedTo(result_array) - } - _ => SimplifyResult::None, - } -} - -type ECDSASignatureVerifier = fn( - hashed_msg: &[u8], - public_key_x: &[u8; 32], - public_key_y: &[u8; 32], - signature: &[u8; 64], -) -> Result; -fn simplify_signature( - dfg: &mut DataFlowGraph, - arguments: &[ValueId], - signature_verifier: ECDSASignatureVerifier, -) -> SimplifyResult { - match ( - dfg.get_array_constant(arguments[0]), - dfg.get_array_constant(arguments[1]), - dfg.get_array_constant(arguments[2]), - dfg.get_array_constant(arguments[3]), - ) { - ( - Some((public_key_x, _)), - Some((public_key_y, _)), - Some((signature, _)), - Some((hashed_message, _)), - ) if array_is_constant(dfg, &public_key_x) - && array_is_constant(dfg, &public_key_y) - && array_is_constant(dfg, &signature) - && array_is_constant(dfg, &hashed_message) => - { - let public_key_x: [u8; 32] = to_u8_vec(dfg, public_key_x) - .try_into() - .expect("ECDSA public key fields are 32 bytes"); - let public_key_y: [u8; 32] = to_u8_vec(dfg, public_key_y) - .try_into() - .expect("ECDSA public key fields are 32 bytes"); - let signature: [u8; 64] = - to_u8_vec(dfg, signature).try_into().expect("ECDSA signatures are 64 bytes"); - let hashed_message: Vec = to_u8_vec(dfg, hashed_message); - - let valid_signature = - signature_verifier(&hashed_message, &public_key_x, &public_key_y, &signature) - .expect("Rust solvable black box function should not fail"); - - let valid_signature = dfg.make_constant(valid_signature.into(), NumericType::bool()); - SimplifyResult::SimplifiedTo(valid_signature) - } - _ => SimplifyResult::None, - } -} - fn simplify_derive_generators( dfg: &mut DataFlowGraph, arguments: &[ValueId], @@ -819,7 +752,7 @@ mod tests { return v2 } "#; - let ssa = Ssa::from_str(src).unwrap(); + let ssa = Ssa::from_str_simplifying(src).unwrap(); let expected = r#" brillig(inline) fn main f0 { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/call/blackbox.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/call/blackbox.rs index ffacf6fe8b5..fac2e8b4d5a 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/call/blackbox.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/call/blackbox.rs @@ -323,7 +323,7 @@ mod test { v2 = call multi_scalar_mul (v1, v0) -> [Field; 3] return v2 }"#; - let ssa = Ssa::from_str(src).unwrap(); + let ssa = Ssa::from_str_simplifying(src).unwrap(); let expected_src = r#" acir(inline) fn main f0 { @@ -351,7 +351,7 @@ mod test { return v4 }"#; - let ssa = Ssa::from_str(src).unwrap(); + let ssa = Ssa::from_str_simplifying(src).unwrap(); //First point is zero, second scalar is zero, so we should be left with the scalar mul of the last point. let expected_src = r#" acir(inline) fn main f0 { @@ -379,7 +379,7 @@ mod test { v4 = call multi_scalar_mul (v3, v2) -> [Field; 3] return v4 }"#; - let ssa = Ssa::from_str(src).unwrap(); + let ssa = Ssa::from_str_simplifying(src).unwrap(); //First and last scalar/point are constant, so we should be left with the msm of the middle point and the folded constant point let expected_src = r#" acir(inline) fn main f0 { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/constrain.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/constrain.rs index 5ae6a642a57..a3881419a83 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/constrain.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/instruction/constrain.rs @@ -44,7 +44,7 @@ pub(super) fn decompose_constrain( vec![Instruction::Constrain(lhs, rhs, msg.clone())] } - Instruction::Binary(Binary { lhs, rhs, operator: BinaryOp::Mul }) + Instruction::Binary(Binary { lhs, rhs, operator: BinaryOp::Mul { .. } }) if constant.is_one() && dfg.type_of_value(lhs) == Type::bool() => { // Replace an equality assertion on a boolean multiplication diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/list.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/list.rs deleted file mode 100644 index 9a84d304444..00000000000 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/list.rs +++ /dev/null @@ -1,187 +0,0 @@ -use serde::{Deserialize, Serialize}; -use std::sync::Arc; - -/// A shared linked list type intended to be cloned -#[derive(Clone, PartialEq, Eq, Hash, Serialize, Deserialize)] -pub struct List { - head: Arc>, - len: usize, -} - -#[derive(Default, Clone, PartialEq, Eq, Hash, Serialize, Deserialize)] -enum Node { - #[default] - Nil, - Cons(T, Arc>), -} - -impl Default for List { - fn default() -> Self { - List { head: Arc::new(Node::Nil), len: 0 } - } -} - -impl List { - pub fn new() -> Self { - Self::default() - } - - /// This is actually a push_front since we just create a new head for the - /// list. This is done so that the tail of the list can still be shared. - /// In the case of call stacks, the last node will be main, while the top - /// of the call stack will be the head of this list. - pub fn push_back(&mut self, value: T) { - self.len += 1; - self.head = Arc::new(Node::Cons(value, self.head.clone())); - } - - /// It is more efficient to iterate from the head of the list towards the tail. - /// For callstacks this means from the top of the call stack towards main. - fn iter_rev(&self) -> IterRev { - IterRev { head: &self.head, len: self.len } - } - - pub fn clear(&mut self) { - *self = Self::default(); - } - - pub fn append(&mut self, other: Self) - where - T: Copy + std::fmt::Debug, - { - let other = other.into_iter().collect::>(); - - for item in other { - self.push_back(item); - } - } - - pub fn len(&self) -> usize { - self.len - } - - pub fn is_empty(&self) -> bool { - self.len == 0 - } - - fn pop_front(&mut self) -> Option - where - T: Copy, - { - match self.head.as_ref() { - Node::Nil => None, - Node::Cons(value, rest) => { - let value = *value; - self.head = rest.clone(); - self.len -= 1; - Some(value) - } - } - } - - pub fn truncate(&mut self, len: usize) - where - T: Copy, - { - if self.len > len { - for _ in 0..self.len - len { - self.pop_front(); - } - } - } - - pub fn unit(item: T) -> Self { - let mut this = Self::default(); - this.push_back(item); - this - } - - pub fn back(&self) -> Option<&T> { - match self.head.as_ref() { - Node::Nil => None, - Node::Cons(item, _) => Some(item), - } - } -} - -pub struct IterRev<'a, T> { - head: &'a Node, - len: usize, -} - -impl IntoIterator for List -where - T: Copy + std::fmt::Debug, -{ - type Item = T; - - type IntoIter = std::iter::Rev>; - - fn into_iter(self) -> Self::IntoIter { - let items: Vec<_> = self.iter_rev().copied().collect(); - items.into_iter().rev() - } -} - -impl<'a, T> IntoIterator for &'a List { - type Item = &'a T; - - type IntoIter = std::iter::Rev< as IntoIterator>::IntoIter>; - - fn into_iter(self) -> Self::IntoIter { - let items: Vec<_> = self.iter_rev().collect(); - items.into_iter().rev() - } -} - -impl<'a, T> Iterator for IterRev<'a, T> { - type Item = &'a T; - - fn next(&mut self) -> Option { - match self.head { - Node::Nil => None, - Node::Cons(value, rest) => { - self.head = rest; - Some(value) - } - } - } - - fn size_hint(&self) -> (usize, Option) { - (0, Some(self.len)) - } -} - -impl<'a, T> ExactSizeIterator for IterRev<'a, T> {} - -impl std::fmt::Debug for List -where - T: std::fmt::Debug, -{ - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - write!(f, "[")?; - for (i, item) in self.iter_rev().enumerate() { - if i != 0 { - write!(f, ", ")?; - } - write!(f, "{item:?}")?; - } - write!(f, "]") - } -} - -impl std::fmt::Display for List -where - T: std::fmt::Display, -{ - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - write!(f, "[")?; - for (i, item) in self.iter_rev().enumerate() { - if i != 0 { - write!(f, ", ")?; - } - write!(f, "{item}")?; - } - write!(f, "]") - } -} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/map.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/map.rs index 0fb02f19b14..b6da107957c 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/map.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/map.rs @@ -1,4 +1,3 @@ -use fxhash::FxHashMap as HashMap; use serde::{Deserialize, Serialize}; use std::{ collections::BTreeMap, @@ -180,11 +179,6 @@ pub(crate) struct DenseMap { } impl DenseMap { - /// Returns the number of elements in the map. - pub(crate) fn len(&self) -> usize { - self.storage.len() - } - /// Adds an element to the map. /// Returns the identifier/reference to that element. pub(crate) fn insert(&mut self, element: T) -> Id { @@ -193,18 +187,10 @@ impl DenseMap { id } - /// Given the Id of the element being created, adds the element - /// returned by the given function to the map - pub(crate) fn insert_with_id(&mut self, f: impl FnOnce(Id) -> T) -> Id { - let id = Id::new(self.storage.len().try_into().unwrap()); - self.storage.push(f(id)); - id - } - /// Gets an iterator to a reference to each element in the dense map paired with its id. /// /// The id-element pairs are ordered by the numeric values of the ids. - pub(crate) fn iter(&self) -> impl ExactSizeIterator, &T)> { + pub(crate) fn iter(&self) -> impl DoubleEndedIterator, &T)> { let ids_iter = (0..self.storage.len() as u32).map(|idx| Id::new(idx)); ids_iter.zip(self.storage.iter()) } @@ -246,19 +232,6 @@ pub(crate) struct SparseMap { } impl SparseMap { - /// Returns the number of elements in the map. - pub(crate) fn len(&self) -> usize { - self.storage.len() - } - - /// Adds an element to the map. - /// Returns the identifier/reference to that element. - pub(crate) fn insert(&mut self, element: T) -> Id { - let id = Id::new(self.storage.len().try_into().unwrap()); - self.storage.insert(id, element); - id - } - /// Given the Id of the element being created, adds the element /// returned by the given function to the map pub(crate) fn insert_with_id(&mut self, f: impl FnOnce(Id) -> T) -> Id { @@ -267,13 +240,6 @@ impl SparseMap { id } - /// Remove an element from the map and return it. - /// This may return None if the element was already - /// previously removed from the map. - pub(crate) fn remove(&mut self, id: Id) -> Option { - self.storage.remove(&id) - } - /// Unwraps the inner storage of this map pub(crate) fn into_btree(self) -> BTreeMap, T> { self.storage @@ -300,65 +266,6 @@ impl std::ops::IndexMut> for SparseMap { } } -/// A TwoWayMap is a map from both key to value and value to key. -/// This is accomplished by keeping the map bijective - for every -/// value there is exactly one key and vice-versa. Any duplicate values -/// are prevented in the call to insert. -#[derive(Debug)] -pub(crate) struct TwoWayMap { - key_to_value: HashMap, - value_to_key: HashMap, -} - -impl TwoWayMap { - /// Returns the number of elements in the map. - pub(crate) fn len(&self) -> usize { - self.key_to_value.len() - } - - /// Adds an element to the map. - /// Returns the identifier/reference to that element. - pub(crate) fn insert(&mut self, key: K, element: V) -> K { - if let Some(existing) = self.value_to_key.get(&element) { - return existing.clone(); - } - - self.key_to_value.insert(key.clone(), element.clone()); - self.value_to_key.insert(element, key.clone()); - - key - } - - pub(crate) fn get(&self, key: &K) -> Option<&V> { - self.key_to_value.get(key) - } -} - -impl Default for TwoWayMap { - fn default() -> Self { - Self { key_to_value: HashMap::default(), value_to_key: HashMap::default() } - } -} - -// Note that there is no impl for IndexMut, -// if we allowed mutable access to map elements they may be -// mutated such that elements are no longer unique -impl std::ops::Index for TwoWayMap { - type Output = V; - - fn index(&self, id: K) -> &Self::Output { - &self.key_to_value[&id] - } -} - -impl std::ops::Index<&K> for TwoWayMap { - type Output = V; - - fn index(&self, id: &K) -> &Self::Output { - &self.key_to_value[id] - } -} - /// A simple counter to create fresh Ids without any storage. /// Useful for assigning ids before the storage is created or assigning ids /// for types that have no single owner. diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/mod.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/mod.rs index 88e0d8900db..686606452fb 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/mod.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/mod.rs @@ -6,7 +6,6 @@ pub(crate) mod dom; pub(crate) mod function; pub(crate) mod function_inserter; pub(crate) mod instruction; -pub mod list; pub(crate) mod map; pub(crate) mod post_order; pub(crate) mod printer; diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/post_order.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/post_order.rs index 398ce887b96..08f195e53d1 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/post_order.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/post_order.rs @@ -13,6 +13,7 @@ enum Visit { Last, } +#[derive(Default)] pub(crate) struct PostOrder(Vec); impl PostOrder { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/printer.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/printer.rs index 29e79728303..88bee0799a3 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/printer.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/printer.rs @@ -1,11 +1,14 @@ //! This file is for pretty-printing the SSA IR in a human-readable form for debugging. -use std::fmt::{Formatter, Result}; +use std::fmt::{Display, Formatter, Result}; use acvm::acir::AcirField; use im::Vector; use iter_extended::vecmap; -use crate::ssa::ir::types::{NumericType, Type}; +use crate::ssa::{ + ir::types::{NumericType, Type}, + Ssa, +}; use super::{ basic_block::BasicBlockId, @@ -15,72 +18,110 @@ use super::{ value::{Value, ValueId}, }; +impl Display for Ssa { + fn fmt(&self, f: &mut Formatter<'_>) -> Result { + for (id, global_value) in self.globals.dfg.values_iter() { + match global_value { + Value::NumericConstant { constant, typ } => { + writeln!(f, "g{} = {typ} {constant}", id.to_u32())?; + } + Value::Instruction { instruction, .. } => { + display_instruction(&self.globals.dfg, *instruction, true, f)?; + } + Value::Global(_) => { + panic!("Value::Global should only be in the function dfg"); + } + _ => panic!("Expected only numeric constant or instruction"), + }; + } + + if self.globals.dfg.values_iter().next().is_some() { + writeln!(f)?; + } + + for function in self.functions.values() { + writeln!(f, "{function}")?; + } + Ok(()) + } +} + +impl Display for Function { + fn fmt(&self, f: &mut Formatter<'_>) -> Result { + display_function(self, f) + } +} + /// Helper function for Function's Display impl to pretty-print the function with the given formatter. -pub(crate) fn display_function(function: &Function, f: &mut Formatter) -> Result { +fn display_function(function: &Function, f: &mut Formatter) -> Result { writeln!(f, "{} fn {} {} {{", function.runtime(), function.name(), function.id())?; for block_id in function.reachable_blocks() { - display_block(function, block_id, f)?; + display_block(&function.dfg, block_id, f)?; } write!(f, "}}") } /// Display a single block. This will not display the block's successors. -pub(crate) fn display_block( - function: &Function, - block_id: BasicBlockId, - f: &mut Formatter, -) -> Result { - let block = &function.dfg[block_id]; +fn display_block(dfg: &DataFlowGraph, block_id: BasicBlockId, f: &mut Formatter) -> Result { + let block = &dfg[block_id]; - writeln!(f, " {}({}):", block_id, value_list_with_types(function, block.parameters()))?; + writeln!(f, " {}({}):", block_id, value_list_with_types(dfg, block.parameters()))?; for instruction in block.instructions() { - display_instruction(function, *instruction, f)?; + display_instruction(dfg, *instruction, false, f)?; } - display_terminator(function, block.terminator(), f) + display_terminator(dfg, block.terminator(), f) } /// Specialize displaying value ids so that if they refer to a numeric /// constant or a function we print those directly. -fn value(function: &Function, id: ValueId) -> String { - let id = function.dfg.resolve(id); - match &function.dfg[id] { +fn value(dfg: &DataFlowGraph, id: ValueId) -> String { + let id = dfg.resolve(id); + match &dfg[id] { Value::NumericConstant { constant, typ } => { format!("{typ} {constant}") } Value::Function(id) => id.to_string(), Value::Intrinsic(intrinsic) => intrinsic.to_string(), - Value::Param { .. } | Value::Instruction { .. } | Value::ForeignFunction(_) => { - id.to_string() + Value::ForeignFunction(function) => function.clone(), + Value::Param { .. } | Value::Instruction { .. } => { + if dfg.is_global(id) { + format!("g{}", id.to_u32()) + } else { + id.to_string() + } + } + Value::Global(_) => { + format!("g{}", id.to_u32()) } } } /// Display each value along with its type. E.g. `v0: Field, v1: u64, v2: u1` -fn value_list_with_types(function: &Function, values: &[ValueId]) -> String { +fn value_list_with_types(dfg: &DataFlowGraph, values: &[ValueId]) -> String { vecmap(values, |id| { - let value = value(function, *id); - let typ = function.dfg.type_of_value(*id); + let value = value(dfg, *id); + let typ = dfg.type_of_value(*id); format!("{value}: {typ}") }) .join(", ") } /// Display each value separated by a comma -fn value_list(function: &Function, values: &[ValueId]) -> String { - vecmap(values, |id| value(function, *id)).join(", ") +fn value_list(dfg: &DataFlowGraph, values: &[ValueId]) -> String { + vecmap(values, |id| value(dfg, *id)).join(", ") } /// Display a terminator instruction -pub(crate) fn display_terminator( - function: &Function, +fn display_terminator( + dfg: &DataFlowGraph, terminator: Option<&TerminatorInstruction>, f: &mut Formatter, ) -> Result { match terminator { Some(TerminatorInstruction::Jmp { destination, arguments, call_stack: _ }) => { - writeln!(f, " jmp {}({})", destination, value_list(function, arguments)) + writeln!(f, " jmp {}({})", destination, value_list(dfg, arguments)) } Some(TerminatorInstruction::JmpIf { condition, @@ -91,7 +132,7 @@ pub(crate) fn display_terminator( writeln!( f, " jmpif {} then: {}, else: {}", - value(function, *condition), + value(dfg, *condition), then_destination, else_destination ) @@ -100,7 +141,7 @@ pub(crate) fn display_terminator( if return_values.is_empty() { writeln!(f, " return") } else { - writeln!(f, " return {}", value_list(function, return_values)) + writeln!(f, " return {}", value_list(dfg, return_values)) } } None => writeln!(f, " (no terminator instruction)"), @@ -108,29 +149,37 @@ pub(crate) fn display_terminator( } /// Display an arbitrary instruction -pub(crate) fn display_instruction( - function: &Function, +fn display_instruction( + dfg: &DataFlowGraph, instruction: InstructionId, + in_global_space: bool, f: &mut Formatter, ) -> Result { - // instructions are always indented within a function - write!(f, " ")?; + if !in_global_space { + // instructions are always indented within a function + write!(f, " ")?; + } - let results = function.dfg.instruction_results(instruction); + let results = dfg.instruction_results(instruction); if !results.is_empty() { - write!(f, "{} = ", value_list(function, results))?; + let mut value_list = value_list(dfg, results); + if in_global_space { + value_list = value_list.replace('v', "g"); + } + write!(f, "{} = ", value_list)?; } - display_instruction_inner(function, &function.dfg[instruction], results, f) + display_instruction_inner(dfg, &dfg[instruction], results, in_global_space, f) } fn display_instruction_inner( - function: &Function, + dfg: &DataFlowGraph, instruction: &Instruction, results: &[ValueId], + in_global_space: bool, f: &mut Formatter, ) -> Result { - let show = |id| value(function, id); + let show = |id| value(dfg, id); match instruction { Instruction::Binary(binary) => { @@ -145,20 +194,28 @@ fn display_instruction_inner( Instruction::Constrain(lhs, rhs, error) => { write!(f, "constrain {} == {}", show(*lhs), show(*rhs))?; if let Some(error) = error { - display_constrain_error(function, error, f) + display_constrain_error(dfg, error, f) + } else { + writeln!(f) + } + } + Instruction::ConstrainNotEqual(lhs, rhs, error) => { + write!(f, "constrain {} != {}", show(*lhs), show(*rhs))?; + if let Some(error) = error { + display_constrain_error(dfg, error, f) } else { writeln!(f) } } Instruction::Call { func, arguments } => { - let arguments = value_list(function, arguments); - writeln!(f, "call {}({}){}", show(*func), arguments, result_types(function, results)) + let arguments = value_list(dfg, arguments); + writeln!(f, "call {}({}){}", show(*func), arguments, result_types(dfg, results)) } Instruction::Allocate => { - writeln!(f, "allocate{}", result_types(function, results)) + writeln!(f, "allocate{}", result_types(dfg, results)) } Instruction::Load { address } => { - writeln!(f, "load {}{}", show(*address), result_types(function, results)) + writeln!(f, "load {}{}", show(*address), result_types(dfg, results)) } Instruction::Store { address, value } => { writeln!(f, "store {} at {}", show(*value), show(*address)) @@ -172,7 +229,7 @@ fn display_instruction_inner( "array_get {}, index {}{}", show(*array), show(*index), - result_types(function, results) + result_types(dfg, results) ) } Instruction::ArraySet { array, index, value, mutable } => { @@ -215,7 +272,7 @@ fn display_instruction_inner( if element_types.len() == 1 && element_types[0] == Type::Numeric(NumericType::Unsigned { bit_size: 8 }) { - if let Some(string) = try_byte_array_to_string(elements, function) { + if let Some(string) = try_byte_array_to_string(elements, dfg) { if is_slice { return writeln!(f, "make_array &b{:?}", string); } else { @@ -230,18 +287,23 @@ fn display_instruction_inner( if i != 0 { write!(f, ", ")?; } - write!(f, "{}", show(*element))?; + let mut value = show(*element); + if in_global_space { + value = value.replace('v', "g"); + } + write!(f, "{}", value)?; } writeln!(f, "] : {typ}") } + Instruction::Noop => writeln!(f, "no-op"), } } -fn try_byte_array_to_string(elements: &Vector, function: &Function) -> Option { +fn try_byte_array_to_string(elements: &Vector, dfg: &DataFlowGraph) -> Option { let mut string = String::new(); for element in elements { - let element = function.dfg.get_numeric_constant(*element)?; + let element = dfg.get_numeric_constant(*element)?; let element = element.try_to_u32()?; if element > 0xFF { return None; @@ -257,8 +319,8 @@ fn try_byte_array_to_string(elements: &Vector, function: &Function) -> Some(string) } -fn result_types(function: &Function, results: &[ValueId]) -> String { - let types = vecmap(results, |result| function.dfg.type_of_value(*result).to_string()); +fn result_types(dfg: &DataFlowGraph, results: &[ValueId]) -> String { + let types = vecmap(results, |result| dfg.type_of_value(*result).to_string()); if types.is_empty() { String::new() } else if types.len() == 1 { @@ -274,26 +336,15 @@ pub(crate) fn try_to_extract_string_from_error_payload( values: &[ValueId], dfg: &DataFlowGraph, ) -> Option { - (is_string_type && (values.len() == 1)) - .then_some(()) - .and_then(|()| { - let (values, _) = &dfg.get_array_constant(values[0])?; - let values = values.iter().map(|value_id| dfg.get_numeric_constant(*value_id)); - values.collect::>>() - }) - .map(|fields| { - fields - .iter() - .map(|field| { - let as_u8 = field.try_to_u64().unwrap_or_default() as u8; - as_u8 as char - }) - .collect() - }) + if is_string_type && values.len() == 1 { + dfg.get_string(values[0]) + } else { + None + } } fn display_constrain_error( - function: &Function, + dfg: &DataFlowGraph, error: &ConstrainError, f: &mut Formatter, ) -> Result { @@ -303,11 +354,11 @@ fn display_constrain_error( } ConstrainError::Dynamic(_, is_string, values) => { if let Some(constant_string) = - try_to_extract_string_from_error_payload(*is_string, values, &function.dfg) + try_to_extract_string_from_error_payload(*is_string, values, dfg) { writeln!(f, ", {constant_string:?}") } else { - writeln!(f, ", data {}", value_list(function, values)) + writeln!(f, ", data {}", value_list(dfg, values)) } } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/value.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/value.rs index ec7a8e25246..53f87a260c3 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/value.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ir/value.rs @@ -20,10 +20,10 @@ pub(crate) type ValueId = Id; pub(crate) enum Value { /// This value was created due to an instruction /// - /// instruction -- This is the instruction which defined it - /// typ -- This is the `Type` of the instruction - /// position -- Returns the position in the results - /// vector that this `Value` is located. + /// * `instruction`: This is the instruction which defined it + /// * `typ`: This is the `Type` of the instruction + /// * `position`: Returns the position in the results vector that this `Value` is located. + /// /// Example, if you add two numbers together, then the resulting /// value would have position `0`, the typ would be the type /// of the operands, and the instruction would map to an add instruction. @@ -53,6 +53,9 @@ pub(crate) enum Value { /// ForeignFunction's always have the type Type::Function and have similar semantics to Function, /// other than generating different backend operations and being only accessible through Brillig. ForeignFunction(String), + + /// This Value indicates we have a reserved slot that needs to be accessed in a separate global context + Global(Type), } impl Value { @@ -64,6 +67,7 @@ impl Value { Value::Function { .. } | Value::Intrinsic { .. } | Value::ForeignFunction { .. } => { Cow::Owned(Type::Function) } + Value::Global(typ) => Cow::Borrowed(typ), } } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/array_set.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/array_set.rs index 09339cf0797..05ceafcf450 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/array_set.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/array_set.rs @@ -62,7 +62,6 @@ struct Context<'f> { // Mapping of an array that comes from a load and whether the address // it was loaded from is a reference parameter passed to the block. arrays_from_load: HashMap, - inner_nested_arrays: HashMap, } impl<'f> Context<'f> { @@ -72,7 +71,6 @@ impl<'f> Context<'f> { array_to_last_use: HashMap::default(), instructions_that_can_be_made_mutable: HashSet::default(), arrays_from_load: HashMap::default(), - inner_nested_arrays: HashMap::default(), } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/assert_constant.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/assert_constant.rs index 6936c7ad542..192c0f59344 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/assert_constant.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/assert_constant.rs @@ -122,7 +122,11 @@ fn evaluate_static_assert( } else { let call_stack = function.dfg.get_instruction_call_stack(instruction); if function.dfg.is_constant(arguments[0]) { - Err(RuntimeError::StaticAssertFailed { call_stack }) + let message = function + .dfg + .get_string(arguments[1]) + .expect("Expected second argument to be a string"); + Err(RuntimeError::StaticAssertFailed { message, call_stack }) } else { Err(RuntimeError::StaticAssertDynamicPredicate { call_stack }) } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/constant_folding.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/constant_folding.rs index c81a557178b..e8cae7da5b5 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/constant_folding.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/constant_folding.rs @@ -307,6 +307,7 @@ impl<'brillig> Context<'brillig> { let old_results = dfg.instruction_results(id).to_vec(); // If a copy of this instruction exists earlier in the block, then reuse the previous results. + let runtime_is_brillig = dfg.runtime().is_brillig(); if let Some(cache_result) = self.get_cached(dfg, dom, &instruction, *side_effects_enabled_var, block) { @@ -314,7 +315,7 @@ impl<'brillig> Context<'brillig> { CacheResult::Cached(cached) => { // We track whether we may mutate MakeArray instructions before we deduplicate // them but we still need to issue an extra inc_rc in case they're mutated afterward. - if matches!(instruction, Instruction::MakeArray { .. }) { + if runtime_is_brillig && matches!(instruction, Instruction::MakeArray { .. }) { let value = *cached.last().unwrap(); let inc_rc = Instruction::IncrementRc { value }; let call_stack = dfg.get_instruction_call_stack_id(id); @@ -422,14 +423,18 @@ impl<'brillig> Context<'brillig> { .then(|| vecmap(old_results, |result| dfg.type_of_value(*result))); let call_stack = dfg.get_instruction_call_stack_id(id); - let new_results = - match dfg.insert_instruction_and_results(instruction, block, ctrl_typevars, call_stack) - { - InsertInstructionResult::SimplifiedTo(new_result) => vec![new_result], - InsertInstructionResult::SimplifiedToMultiple(new_results) => new_results, - InsertInstructionResult::Results(_, new_results) => new_results.to_vec(), - InsertInstructionResult::InstructionRemoved => vec![], - }; + let new_results = match dfg.insert_instruction_and_results_if_simplified( + instruction, + block, + ctrl_typevars, + call_stack, + Some(id), + ) { + InsertInstructionResult::SimplifiedTo(new_result) => vec![new_result], + InsertInstructionResult::SimplifiedToMultiple(new_results) => new_results, + InsertInstructionResult::Results(_, new_results) => new_results.to_vec(), + InsertInstructionResult::InstructionRemoved => vec![], + }; // Optimizations while inserting the instruction should not change the number of results. assert_eq!(old_results.len(), new_results.len()); @@ -560,7 +565,7 @@ impl<'brillig> Context<'brillig> { ); match evaluation_result { - EvaluationResult::NotABrilligCall | EvaluationResult::CannotEvaluate(_) => None, + EvaluationResult::NotABrilligCall | EvaluationResult::CannotEvaluate => None, EvaluationResult::Evaluated(memory_values) => { let mut memory_index = 0; let new_results = vecmap(old_results, |old_result| { @@ -601,14 +606,14 @@ impl<'brillig> Context<'brillig> { }; if !arguments.iter().all(|argument| dfg.is_constant(*argument)) { - return EvaluationResult::CannotEvaluate(*func_id); + return EvaluationResult::CannotEvaluate; } let mut brillig_arguments = Vec::new(); for argument in arguments { let typ = dfg.type_of_value(*argument); let Some(parameter) = type_to_brillig_parameter(&typ) else { - return EvaluationResult::CannotEvaluate(*func_id); + return EvaluationResult::CannotEvaluate; }; brillig_arguments.push(parameter); } @@ -617,12 +622,12 @@ impl<'brillig> Context<'brillig> { for return_id in func.returns().iter() { let typ = func.dfg.type_of_value(*return_id); if type_to_brillig_parameter(&typ).is_none() { - return EvaluationResult::CannotEvaluate(*func_id); + return EvaluationResult::CannotEvaluate; } } let Ok(generated_brillig) = gen_brillig_for(func, brillig_arguments, brillig) else { - return EvaluationResult::CannotEvaluate(*func_id); + return EvaluationResult::CannotEvaluate; }; let mut calldata = Vec::new(); @@ -632,13 +637,14 @@ impl<'brillig> Context<'brillig> { let bytecode = &generated_brillig.byte_code; let foreign_call_results = Vec::new(); - let black_box_solver = Bn254BlackBoxSolver; + let pedantic_solving = true; + let black_box_solver = Bn254BlackBoxSolver(pedantic_solving); let profiling_active = false; let mut vm = VM::new(calldata, bytecode, foreign_call_results, &black_box_solver, profiling_active); let vm_status: VMStatus<_> = vm.process_opcodes(); let VMStatus::Finished { return_data_offset, return_data_size } = vm_status else { - return EvaluationResult::CannotEvaluate(*func_id); + return EvaluationResult::CannotEvaluate; }; let memory = @@ -770,7 +776,7 @@ enum EvaluationResult { NotABrilligCall, /// The instruction was a call to a brillig function, but we couldn't evaluate it. /// This can occur in the situation where the brillig function reaches a "trap" or a foreign call opcode. - CannotEvaluate(FunctionId), + CannotEvaluate, /// The instruction was a call to a brillig function and we were able to evaluate it, /// returning evaluation memory values. Evaluated(Vec>), @@ -830,9 +836,12 @@ fn simplify(dfg: &DataFlowGraph, lhs: ValueId, rhs: ValueId) -> Option<(ValueId, mod test { use std::sync::Arc; + use noirc_frontend::monomorphization::ast::InlineType; + use crate::ssa::{ function_builder::FunctionBuilder, ir::{ + function::RuntimeType, map::Id, types::{NumericType, Type}, }, @@ -1045,7 +1054,7 @@ mod test { return } "; - let ssa = Ssa::from_str(src).unwrap(); + let ssa = Ssa::from_str_simplifying(src).unwrap(); let expected = " acir(inline) fn main f0 { @@ -1153,6 +1162,7 @@ mod test { // Compiling main let mut builder = FunctionBuilder::new("main".into(), main_id); + builder.set_runtime(RuntimeType::Brillig(InlineType::default())); let v0 = builder.add_parameter(Type::unsigned(64)); let zero = builder.numeric_constant(0u128, NumericType::unsigned(64)); let typ = Type::Array(Arc::new(vec![Type::unsigned(64)]), 25); @@ -1550,9 +1560,8 @@ mod test { fn deduplicates_side_effecting_intrinsics() { let src = " // After EnableSideEffectsIf removal: - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: Field, v1: Field, v2: u1): - v4 = call is_unconstrained() -> u1 v7 = call to_be_radix(v0, u32 256) -> [u8; 1] // `a.to_be_radix(256)`; inc_rc v7 v8 = call to_be_radix(v0, u32 256) -> [u8; 1] // duplicate load of `a` @@ -1561,23 +1570,20 @@ mod test { v10 = mul v0, v9 // attaching `c` to `a` v11 = call to_be_radix(v10, u32 256) -> [u8; 1] // calling `to_radix(c * a)` inc_rc v11 - enable_side_effects v2 // side effect var for `c` shifted down by removal return } "; let ssa = Ssa::from_str(src).unwrap(); let expected = " - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: Field, v1: Field, v2: u1): - v4 = call is_unconstrained() -> u1 - v7 = call to_be_radix(v0, u32 256) -> [u8; 1] - inc_rc v7 - inc_rc v7 - v8 = cast v2 as Field - v9 = mul v0, v8 - v10 = call to_be_radix(v9, u32 256) -> [u8; 1] - inc_rc v10 - enable_side_effects v2 + v5 = call to_be_radix(v0, u32 256) -> [u8; 1] + inc_rc v5 + inc_rc v5 + v6 = cast v2 as Field + v7 = mul v0, v6 + v8 = call to_be_radix(v7, u32 256) -> [u8; 1] + inc_rc v8 return } "; diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/defunctionalize.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/defunctionalize.rs index 7d7798fd30a..a6e04332c0a 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/defunctionalize.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/defunctionalize.rs @@ -8,12 +8,13 @@ use std::collections::{BTreeMap, BTreeSet, HashSet}; use acvm::FieldElement; use iter_extended::vecmap; +use noirc_frontend::monomorphization::ast::InlineType; use crate::ssa::{ function_builder::FunctionBuilder, ir::{ basic_block::BasicBlockId, - function::{Function, FunctionId, Signature}, + function::{Function, FunctionId, RuntimeType, Signature}, instruction::{BinaryOp, Instruction}, types::{NumericType, Type}, value::{Value, ValueId}, @@ -43,12 +44,15 @@ struct ApplyFunction { dispatches_to_multiple_functions: bool, } +type Variants = BTreeMap<(Signature, RuntimeType), Vec>; +type ApplyFunctions = HashMap<(Signature, RuntimeType), ApplyFunction>; + /// Performs defunctionalization on all functions /// This is done by changing all functions as value to be a number (FieldElement) /// And creating apply functions that dispatch to the correct target by runtime comparisons with constants #[derive(Debug, Clone)] struct DefunctionalizationContext { - apply_functions: HashMap, + apply_functions: ApplyFunctions, } impl Ssa { @@ -104,7 +108,7 @@ impl DefunctionalizationContext { }; // Find the correct apply function - let apply_function = self.get_apply_function(&signature); + let apply_function = self.get_apply_function(signature, func.runtime()); // Replace the instruction with a call to apply let apply_function_value_id = func.dfg.import_function(apply_function.id); @@ -152,19 +156,21 @@ impl DefunctionalizationContext { } /// Returns the apply function for the given signature - fn get_apply_function(&self, signature: &Signature) -> ApplyFunction { - *self.apply_functions.get(signature).expect("Could not find apply function") + fn get_apply_function(&self, signature: Signature, runtime: RuntimeType) -> ApplyFunction { + *self.apply_functions.get(&(signature, runtime)).expect("Could not find apply function") } } /// Collects all functions used as values that can be called by their signatures -fn find_variants(ssa: &Ssa) -> BTreeMap> { - let mut dynamic_dispatches: BTreeSet = BTreeSet::new(); +fn find_variants(ssa: &Ssa) -> Variants { + let mut dynamic_dispatches: BTreeSet<(Signature, RuntimeType)> = BTreeSet::new(); let mut functions_as_values: BTreeSet = BTreeSet::new(); for function in ssa.functions.values() { functions_as_values.extend(find_functions_as_values(function)); - dynamic_dispatches.extend(find_dynamic_dispatches(function)); + dynamic_dispatches.extend( + find_dynamic_dispatches(function).into_iter().map(|sig| (sig, function.runtime())), + ); } let mut signature_to_functions_as_value: BTreeMap> = BTreeMap::new(); @@ -174,16 +180,12 @@ fn find_variants(ssa: &Ssa) -> BTreeMap> { signature_to_functions_as_value.entry(signature).or_default().push(function_id); } - let mut variants = BTreeMap::new(); + let mut variants: Variants = BTreeMap::new(); - for dispatch_signature in dynamic_dispatches { - let mut target_fns = vec![]; - for (target_signature, functions) in &signature_to_functions_as_value { - if &dispatch_signature == target_signature { - target_fns.extend(functions); - } - } - variants.insert(dispatch_signature, target_fns); + for (dispatch_signature, caller_runtime) in dynamic_dispatches { + let target_fns = + signature_to_functions_as_value.get(&dispatch_signature).cloned().unwrap_or_default(); + variants.insert((dispatch_signature, caller_runtime), target_fns); } variants @@ -247,10 +249,10 @@ fn find_dynamic_dispatches(func: &Function) -> BTreeSet { fn create_apply_functions( ssa: &mut Ssa, - variants_map: BTreeMap>, -) -> HashMap { + variants_map: BTreeMap<(Signature, RuntimeType), Vec>, +) -> ApplyFunctions { let mut apply_functions = HashMap::default(); - for (signature, variants) in variants_map.into_iter() { + for ((signature, runtime), variants) in variants_map.into_iter() { assert!( !variants.is_empty(), "ICE: at least one variant should exist for a dynamic call {signature:?}" @@ -258,11 +260,12 @@ fn create_apply_functions( let dispatches_to_multiple_functions = variants.len() > 1; let id = if dispatches_to_multiple_functions { - create_apply_function(ssa, signature.clone(), variants) + create_apply_function(ssa, signature.clone(), runtime, variants) } else { variants[0] }; - apply_functions.insert(signature, ApplyFunction { id, dispatches_to_multiple_functions }); + apply_functions + .insert((signature, runtime), ApplyFunction { id, dispatches_to_multiple_functions }); } apply_functions } @@ -275,11 +278,21 @@ fn function_id_to_field(function_id: FunctionId) -> FieldElement { fn create_apply_function( ssa: &mut Ssa, signature: Signature, + caller_runtime: RuntimeType, function_ids: Vec, ) -> FunctionId { assert!(!function_ids.is_empty()); + let globals = ssa.functions[&function_ids[0]].dfg.globals.clone(); ssa.add_fn(|id| { let mut function_builder = FunctionBuilder::new("apply".to_string(), id); + function_builder.set_globals(globals); + + // We want to push for apply functions to be inlined more aggressively. + let runtime = match caller_runtime { + RuntimeType::Acir(_) => RuntimeType::Acir(InlineType::InlineAlways), + RuntimeType::Brillig(_) => RuntimeType::Brillig(InlineType::InlineAlways), + }; + function_builder.set_runtime(runtime); let target_id = function_builder.add_parameter(Type::field()); let params_ids = vecmap(signature.params, |typ| function_builder.add_parameter(typ)); @@ -337,22 +350,156 @@ fn create_apply_function( }) } -/// Crates a return block, if no previous return exists, it will create a final return -/// Else, it will create a bypass return block that points to the previous return block +/// If no previous return target exists, it will create a final return, +/// otherwise returns the existing return block to jump to. fn build_return_block( builder: &mut FunctionBuilder, previous_block: BasicBlockId, passed_types: &[Type], target: Option, ) -> BasicBlockId { + if let Some(return_block) = target { + return return_block; + } let return_block = builder.insert_block(); builder.switch_to_block(return_block); - let params = vecmap(passed_types, |typ| builder.add_block_parameter(return_block, typ.clone())); - match target { - None => builder.terminate_with_return(params), - Some(target) => builder.terminate_with_jmp(target, params), - } + builder.terminate_with_return(params); builder.switch_to_block(previous_block); return_block } + +#[cfg(test)] +mod tests { + use crate::ssa::opt::assert_normalized_ssa_equals; + + use super::Ssa; + + #[test] + fn apply_inherits_caller_runtime() { + // Extracted from `execution_success/brillig_fns_as_values` with `--force-brillig` + let src = " + brillig(inline) fn main f0 { + b0(v0: u32): + v3 = call f1(f2, v0) -> u32 + v5 = add v0, u32 1 + v6 = eq v3, v5 + constrain v3 == v5 + v9 = call f1(f3, v0) -> u32 + v10 = add v0, u32 1 + v11 = eq v9, v10 + constrain v9 == v10 + return + } + brillig(inline) fn wrapper f1 { + b0(v0: function, v1: u32): + v2 = call v0(v1) -> u32 + return v2 + } + brillig(inline) fn increment f2 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + brillig(inline) fn increment_acir f3 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + "; + + let ssa = Ssa::from_str(src).unwrap(); + let ssa = ssa.defunctionalize(); + + let expected = " + brillig(inline) fn main f0 { + b0(v0: u32): + v3 = call f1(Field 2, v0) -> u32 + v5 = add v0, u32 1 + v6 = eq v3, v5 + constrain v3 == v5 + v9 = call f1(Field 3, v0) -> u32 + v10 = add v0, u32 1 + v11 = eq v9, v10 + constrain v9 == v10 + return + } + brillig(inline) fn wrapper f1 { + b0(v0: Field, v1: u32): + v3 = call f4(v0, v1) -> u32 + return v3 + } + brillig(inline) fn increment f2 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + brillig(inline) fn increment_acir f3 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + brillig(inline_always) fn apply f4 { + b0(v0: Field, v1: u32): + v4 = eq v0, Field 2 + jmpif v4 then: b2, else: b1 + b1(): + constrain v0 == Field 3 + v7 = call f3(v1) -> u32 + jmp b3(v7) + b2(): + v9 = call f2(v1) -> u32 + jmp b3(v9) + b3(v2: u32): + return v2 + } + "; + assert_normalized_ssa_equals(ssa, expected); + } + + #[test] + fn apply_created_per_caller_runtime() { + let src = " + acir(inline) fn main f0 { + b0(v0: u32): + v3 = call f1(f2, v0) -> u32 + v5 = add v0, u32 1 + v6 = eq v3, v5 + constrain v3 == v5 + v9 = call f4(f3, v0) -> u32 + v10 = add v0, u32 1 + v11 = eq v9, v10 + constrain v9 == v10 + return + } + brillig(inline) fn wrapper f1 { + b0(v0: function, v1: u32): + v2 = call v0(v1) -> u32 + return v2 + } + acir(inline) fn wrapper_acir f4 { + b0(v0: function, v1: u32): + v2 = call v0(v1) -> u32 + return v2 + } + brillig(inline) fn increment f2 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + acir(inline) fn increment_acir f3 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + "; + + let ssa = Ssa::from_str(src).unwrap(); + let ssa = ssa.defunctionalize(); + + let applies = ssa.functions.values().filter(|f| f.name() == "apply").collect::>(); + assert_eq!(applies.len(), 2); + assert!(applies.iter().any(|f| f.runtime().is_acir())); + assert!(applies.iter().any(|f| f.runtime().is_brillig())); + } +} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/die.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/die.rs index 7b38b764eab..3b5537aceb4 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/die.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/die.rs @@ -22,9 +22,33 @@ use super::rc::{pop_rc_for, RcInstruction}; impl Ssa { /// Performs Dead Instruction Elimination (DIE) to remove any instructions with /// unused results. + /// + /// This step should come after the flattening of the CFG and mem2reg. #[tracing::instrument(level = "trace", skip(self))] - pub(crate) fn dead_instruction_elimination(mut self) -> Ssa { - self.functions.par_iter_mut().for_each(|(_, func)| func.dead_instruction_elimination(true)); + pub(crate) fn dead_instruction_elimination(self) -> Ssa { + self.dead_instruction_elimination_inner(true) + } + + fn dead_instruction_elimination_inner(mut self, flattened: bool) -> Ssa { + let mut used_global_values: HashSet<_> = self + .functions + .par_iter_mut() + .flat_map(|(_, func)| func.dead_instruction_elimination(true, flattened)) + .collect(); + + // Check which globals are used across all functions + for (id, value) in self.globals.dfg.values_iter().rev() { + if used_global_values.contains(&id) { + if let Value::Instruction { instruction, .. } = &value { + let instruction = &self.globals.dfg[*instruction]; + instruction.for_each_value(|value_id| { + used_global_values.insert(value_id); + }); + } + } + } + + self.used_global_values = used_global_values; self } @@ -37,8 +61,16 @@ impl Function { /// instructions that reference results from an instruction in another block are evaluated first. /// If we did not iterate blocks in this order we could not safely say whether or not the results /// of its instructions are needed elsewhere. - pub(crate) fn dead_instruction_elimination(&mut self, insert_out_of_bounds_checks: bool) { - let mut context = Context::default(); + /// + /// Returns the set of globals that were used in this function. + /// After processing all functions, the union of these sets enables determining the unused globals. + pub(crate) fn dead_instruction_elimination( + &mut self, + insert_out_of_bounds_checks: bool, + flattened: bool, + ) -> HashSet { + let mut context = Context { flattened, ..Default::default() }; + for call_data in &self.dfg.data_bus.call_data { context.mark_used_instruction_results(&self.dfg, call_data.array_id); } @@ -58,11 +90,12 @@ impl Function { // instructions (we don't want to remove those checks, or instructions that are // dependencies of those checks) if inserted_out_of_bounds_checks { - self.dead_instruction_elimination(false); - return; + return self.dead_instruction_elimination(false, flattened); } context.remove_rc_instructions(&mut self.dfg); + + context.used_values.into_iter().filter(|value| self.dfg.is_global(*value)).collect() } } @@ -76,6 +109,11 @@ struct Context { /// they technically contain side-effects but we still want to remove them if their /// `value` parameter is not used elsewhere. rc_instructions: Vec<(InstructionId, BasicBlockId)>, + + /// The elimination of certain unused instructions assumes that the DIE pass runs after + /// the flattening of the CFG, but if that's not the case then we should not eliminate + /// them just yet. + flattened: bool, } impl Context { @@ -172,7 +210,7 @@ impl Context { fn is_unused(&self, instruction_id: InstructionId, function: &Function) -> bool { let instruction = &function.dfg[instruction_id]; - if instruction.can_eliminate_if_unused(function) { + if instruction.can_eliminate_if_unused(function, self.flattened) { let results = function.dfg.instruction_results(instruction_id); results.iter().all(|result| !self.used_values.contains(result)) } else if let Instruction::Call { func, arguments } = instruction { @@ -195,15 +233,17 @@ impl Context { /// Inspects a value and marks all instruction results as used. fn mark_used_instruction_results(&mut self, dfg: &DataFlowGraph, value_id: ValueId) { let value_id = dfg.resolve(value_id); - if matches!(&dfg[value_id], Value::Instruction { .. } | Value::Param { .. }) { + if matches!(&dfg[value_id], Value::Instruction { .. } | Value::Param { .. }) + || dfg.is_global(value_id) + { self.used_values.insert(value_id); } } - fn remove_rc_instructions(self, dfg: &mut DataFlowGraph) { + fn remove_rc_instructions(&self, dfg: &mut DataFlowGraph) { let unused_rc_values_by_block: HashMap> = - self.rc_instructions.into_iter().fold(HashMap::default(), |mut acc, (rc, block)| { - let value = match &dfg[rc] { + self.rc_instructions.iter().fold(HashMap::default(), |mut acc, (rc, block)| { + let value = match &dfg[*rc] { Instruction::IncrementRc { value } => *value, Instruction::DecrementRc { value } => *value, other => { @@ -214,7 +254,7 @@ impl Context { }; if !self.used_values.contains(&value) { - acc.entry(block).or_default().insert(rc); + acc.entry(*block).or_default().insert(*rc); } acc }); @@ -356,15 +396,16 @@ impl Context { ) -> bool { use Instruction::*; if let IncrementRc { value } | DecrementRc { value } = instruction { - if let Value::Instruction { instruction, .. } = &dfg[*value] { - return match &dfg[*instruction] { - MakeArray { .. } => true, - Call { func, .. } => { - matches!(&dfg[*func], Value::Intrinsic(_) | Value::ForeignFunction(_)) - } - _ => false, - }; - } + let Some(instruction) = dfg.get_local_or_global_instruction(*value) else { + return false; + }; + return match instruction { + MakeArray { .. } => true, + Call { func, .. } => { + matches!(&dfg[*func], Value::Intrinsic(_) | Value::ForeignFunction(_)) + } + _ => false, + }; } false } @@ -507,16 +548,18 @@ fn apply_side_effects( let casted_condition = dfg.insert_instruction_and_results(cast, block_id, None, call_stack); let casted_condition = casted_condition.first(); + // Unchecked mul because the side effects var is always 0 or 1 let lhs = dfg.insert_instruction_and_results( - Instruction::binary(BinaryOp::Mul, lhs, casted_condition), + Instruction::binary(BinaryOp::Mul { unchecked: true }, lhs, casted_condition), block_id, None, call_stack, ); let lhs = lhs.first(); + // Unchecked mul because the side effects var is always 0 or 1 let rhs = dfg.insert_instruction_and_results( - Instruction::binary(BinaryOp::Mul, rhs, casted_condition), + Instruction::binary(BinaryOp::Mul { unchecked: true }, rhs, casted_condition), block_id, None, call_stack, @@ -637,10 +680,12 @@ mod test { use std::sync::Arc; use im::vector; + use noirc_frontend::monomorphization::ast::InlineType; use crate::ssa::{ function_builder::FunctionBuilder, ir::{ + function::RuntimeType, map::Id, types::{NumericType, Type}, }, @@ -742,7 +787,7 @@ mod test { #[test] fn keep_paired_rcs_with_array_set() { let src = " - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: [Field; 2]): inc_rc v0 v2 = array_set v0, index u32 0, value u32 0 @@ -759,7 +804,7 @@ mod test { #[test] fn keep_inc_rc_on_borrowed_array_store() { - // acir(inline) fn main f0 { + // brillig(inline) fn main f0 { // b0(): // v1 = make_array [u32 0, u32 0] // v2 = allocate @@ -776,6 +821,7 @@ mod test { // Compiling main let mut builder = FunctionBuilder::new("main".into(), main_id); + builder.set_runtime(RuntimeType::Brillig(InlineType::Inline)); let zero = builder.numeric_constant(0u128, NumericType::unsigned(32)); let array_type = Type::Array(Arc::new(vec![Type::unsigned(32)]), 2); let v1 = builder.insert_make_array(vector![zero, zero], array_type.clone()); @@ -810,7 +856,7 @@ mod test { #[test] fn keep_inc_rc_on_borrowed_array_set() { - // acir(inline) fn main f0 { + // brillig(inline) fn main f0 { // b0(v0: [u32; 2]): // inc_rc v0 // v3 = array_set v0, index u32 0, value u32 1 @@ -821,7 +867,7 @@ mod test { // return v4 // } let src = " - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: [u32; 2]): inc_rc v0 v3 = array_set v0, index u32 0, value u32 1 @@ -837,7 +883,7 @@ mod test { // We expect the output to be unchanged // Except for the repeated inc_rc instructions let expected = " - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: [u32; 2]): inc_rc v0 v3 = array_set v0, index u32 0, value u32 1 @@ -875,7 +921,7 @@ mod test { #[test] fn remove_inc_rcs_that_are_never_mutably_borrowed() { let src = " - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: [Field; 2]): inc_rc v0 inc_rc v0 @@ -893,7 +939,7 @@ mod test { assert_eq!(main.dfg[main.entry_block()].instructions().len(), 5); let expected = " - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: [Field; 2]): v2 = array_get v0, index u32 0 -> Field return v2 @@ -936,4 +982,53 @@ mod test { let ssa = ssa.dead_instruction_elimination(); assert_normalized_ssa_equals(ssa, src); } + + #[test] + fn do_not_remove_mutable_reference_params() { + let src = " + acir(inline) fn main f0 { + b0(v0: Field, v1: Field): + v2 = allocate -> &mut Field + store v0 at v2 + call f1(v2) + v4 = load v2 -> Field + v5 = eq v4, v1 + constrain v4 == v1 + return + } + acir(inline) fn Add10 f1 { + b0(v0: &mut Field): + v1 = load v0 -> Field + v2 = load v0 -> Field + v4 = add v2, Field 10 + store v4 at v0 + return + } + "; + + let ssa = Ssa::from_str(src).unwrap(); + + // Even though these ACIR functions only have 1 block, we have not inlined and flattened anything yet. + let ssa = ssa.dead_instruction_elimination_inner(false); + + let expected = " + acir(inline) fn main f0 { + b0(v0: Field, v1: Field): + v2 = allocate -> &mut Field + store v0 at v2 + call f1(v2) + v4 = load v2 -> Field + constrain v4 == v1 + return + } + acir(inline) fn Add10 f1 { + b0(v0: &mut Field): + v1 = load v0 -> Field + v3 = add v1, Field 10 + store v3 at v0 + return + } + "; + assert_normalized_ssa_equals(ssa, expected); + } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg.rs index 43420229257..76f8495c009 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg.rs @@ -34,7 +34,7 @@ //! the following transformations of certain instructions within the block are expected: //! //! 1. A constraint is multiplied by the condition and changes the constraint to -//! an equality with c: +//! an equality with c: //! //! constrain v0 //! ============ @@ -152,7 +152,6 @@ use crate::ssa::{ }; mod branch_analysis; -mod capacity_tracker; pub(crate) mod value_merger; impl Ssa { @@ -194,10 +193,6 @@ struct Context<'f> { /// the most recent condition combined with all previous conditions via `And` instructions. condition_stack: Vec, - /// Maps SSA array values with a slice type to their size. - /// This is maintained by appropriate calls to the `SliceCapacityTracker` and is used by the `ValueMerger`. - slice_sizes: HashMap, - /// Stack of block arguments /// When processing a block, we pop this stack to get its arguments /// and at the end we push the arguments for his successor @@ -211,6 +206,13 @@ struct Context<'f> { /// /// The `ValueId` here is that which is returned by the allocate instruction. local_allocations: HashSet, + + /// A map from `cond` to `Not(cond)` + /// + /// `Not` instructions are inserted constantly by this pass and this map helps keep + /// us from unnecessarily inserting extra instructions, and keeps ids unique which + /// helps simplifications. + not_instructions: HashMap, } #[derive(Clone)] @@ -252,10 +254,10 @@ fn flatten_function_cfg(function: &mut Function, no_predicates: &HashMap Context<'f> { let condition_call_stack = self.inserter.function.dfg.get_value_call_stack_id(cond_context.condition); - let else_condition = - self.insert_instruction(Instruction::Not(cond_context.condition), condition_call_stack); + + let else_condition = self.not_instruction(cond_context.condition, condition_call_stack); let else_condition = self.link_condition(else_condition); let old_allocations = std::mem::take(&mut self.local_allocations); @@ -464,6 +466,16 @@ impl<'f> Context<'f> { vec![self.cfg.successors(*block).next().unwrap()] } + fn not_instruction(&mut self, condition: ValueId, call_stack: CallStackId) -> ValueId { + if let Some(existing) = self.not_instructions.get(&condition) { + return *existing; + } + + let not = self.insert_instruction(Instruction::Not(condition), call_stack); + self.not_instructions.insert(condition, not); + not + } + /// Process the 'exit' block of a conditional statement fn else_stop(&mut self, block: &BasicBlockId) -> Vec { let mut cond_context = self.condition_stack.pop().unwrap(); @@ -642,20 +654,10 @@ impl<'f> Context<'f> { // Replace constraint `lhs == rhs` with `condition * lhs == condition * rhs`. // Condition needs to be cast to argument type in order to multiply them together. - let argument_type = self.inserter.function.dfg.type_of_value(lhs); - - let cast = Instruction::Cast(condition, argument_type.unwrap_numeric()); - let casted_condition = self.insert_instruction(cast, call_stack); - - let lhs = self.insert_instruction( - Instruction::binary(BinaryOp::Mul, lhs, casted_condition), - call_stack, - ); - let rhs = self.insert_instruction( - Instruction::binary(BinaryOp::Mul, rhs, casted_condition), - call_stack, - ); - + let casted_condition = + self.cast_condition_to_value_type(condition, lhs, call_stack); + let lhs = self.mul_by_condition(lhs, casted_condition, call_stack); + let rhs = self.mul_by_condition(rhs, casted_condition, call_stack); Instruction::Constrain(lhs, rhs, message) } Instruction::Store { address, value } => { @@ -671,8 +673,7 @@ impl<'f> Context<'f> { .insert_instruction_with_typevars(load, Some(vec![typ]), call_stack) .first(); - let else_condition = - self.insert_instruction(Instruction::Not(condition), call_stack); + let else_condition = self.not_instruction(condition, call_stack); let instruction = Instruction::IfElse { then_condition: condition, @@ -689,29 +690,18 @@ impl<'f> Context<'f> { // Replace value with `value * predicate` to zero out value when predicate is inactive. // Condition needs to be cast to argument type in order to multiply them together. - let argument_type = self.inserter.function.dfg.type_of_value(value); - let cast = Instruction::Cast(condition, argument_type.unwrap_numeric()); - let casted_condition = self.insert_instruction(cast, call_stack); - - let value = self.insert_instruction( - Instruction::binary(BinaryOp::Mul, value, casted_condition), - call_stack, - ); + let casted_condition = + self.cast_condition_to_value_type(condition, value, call_stack); + let value = self.mul_by_condition(value, casted_condition, call_stack); Instruction::RangeCheck { value, max_bit_size, assert_message } } Instruction::Call { func, mut arguments } => match self.inserter.function.dfg[func] { Value::Intrinsic(Intrinsic::ToBits(_) | Intrinsic::ToRadix(_)) => { let field = arguments[0]; - let argument_type = self.inserter.function.dfg.type_of_value(field); - - let cast = Instruction::Cast(condition, argument_type.unwrap_numeric()); - - let casted_condition = self.insert_instruction(cast, call_stack); - let field = self.insert_instruction( - Instruction::binary(BinaryOp::Mul, field, casted_condition), - call_stack, - ); + let casted_condition = + self.cast_condition_to_value_type(condition, field, call_stack); + let field = self.mul_by_condition(field, casted_condition, call_stack); arguments[0] = field; @@ -755,6 +745,30 @@ impl<'f> Context<'f> { } } + fn cast_condition_to_value_type( + &mut self, + condition: ValueId, + value: ValueId, + call_stack: CallStackId, + ) -> ValueId { + let argument_type = self.inserter.function.dfg.type_of_value(value); + let cast = Instruction::Cast(condition, argument_type.unwrap_numeric()); + self.insert_instruction(cast, call_stack) + } + + fn mul_by_condition( + &mut self, + value: ValueId, + condition: ValueId, + call_stack: CallStackId, + ) -> ValueId { + // Unchecked mul because the condition is always 0 or 1 + self.insert_instruction( + Instruction::binary(BinaryOp::Mul { unchecked: true }, value, condition), + call_stack, + ) + } + /// When a MSM is done under a predicate, we need to apply the predicate /// to the is_infinity property of the input points in order to ensure /// that the points will be on the curve no matter what. @@ -787,11 +801,11 @@ impl<'f> Context<'f> { // Computes: if condition { var } else { 1 } fn var_or_one(&mut self, var: ValueId, condition: ValueId, call_stack: CallStackId) -> ValueId { - let field = - self.insert_instruction(Instruction::binary(BinaryOp::Mul, var, condition), call_stack); - let not_condition = self.insert_instruction(Instruction::Not(condition), call_stack); + let field = self.mul_by_condition(var, condition, call_stack); + let not_condition = self.not_instruction(condition, call_stack); + // Unchecked add because of the values is guaranteed to be 0 self.insert_instruction( - Instruction::binary(BinaryOp::Add, field, not_condition), + Instruction::binary(BinaryOp::Add { unchecked: true }, field, not_condition), call_stack, ) } @@ -802,13 +816,9 @@ mod test { use acvm::acir::AcirField; use crate::ssa::{ - function_builder::FunctionBuilder, ir::{ dfg::DataFlowGraph, - function::Function, - instruction::{BinaryOp, Instruction, TerminatorInstruction}, - map::Id, - types::Type, + instruction::{Instruction, TerminatorInstruction}, value::{Value, ValueId}, }, opt::assert_normalized_ssa_equals, @@ -910,7 +920,6 @@ mod test { v8 = mul v5, v2 v9 = add v7, v8 store v9 at v1 - v10 = not v0 enable_side_effects u1 1 return } @@ -949,15 +958,14 @@ mod test { v8 = mul v5, v2 v9 = add v7, v8 store v9 at v1 - v10 = not v0 - enable_side_effects v10 - v11 = load v1 -> Field - v12 = cast v10 as Field - v13 = cast v0 as Field - v15 = mul v12, Field 6 - v16 = mul v13, v11 - v17 = add v15, v16 - store v17 at v1 + enable_side_effects v3 + v10 = load v1 -> Field + v11 = cast v3 as Field + v12 = cast v0 as Field + v14 = mul v11, Field 6 + v15 = mul v12, v10 + v16 = add v14, v15 + store v16 at v1 enable_side_effects u1 1 return } @@ -966,14 +974,6 @@ mod test { assert_normalized_ssa_equals(ssa, expected); } - fn count_instruction(function: &Function, f: impl Fn(&Instruction) -> bool) -> usize { - function.dfg[function.entry_block()] - .instructions() - .iter() - .filter(|id| f(&function.dfg[**id])) - .count() - } - #[test] fn nested_branch_stores() { // Here we build some SSA with control flow given by the following graph. @@ -1086,15 +1086,14 @@ mod test { v24 = mul v21, v16 v25 = add v23, v24 enable_side_effects v0 - v26 = not v0 - enable_side_effects v26 - v27 = cast v26 as Field - v28 = cast v0 as Field - v30 = mul v27, Field 3 - v31 = mul v28, v25 - v32 = add v30, v31 + enable_side_effects v3 + v26 = cast v3 as Field + v27 = cast v0 as Field + v29 = mul v26, Field 3 + v30 = mul v27, v25 + v31 = add v29, v30 enable_side_effects u1 1 - return v32 + return v31 }"; let main = ssa.main(); @@ -1284,17 +1283,16 @@ mod test { v12 = not v5 v13 = cast v4 as u8 v14 = cast v12 as u8 - v15 = mul v13, v10 - v16 = mul v14, v11 - v17 = add v15, v16 + v15 = unchecked_mul v13, v10 + v16 = unchecked_mul v14, v11 + v17 = unchecked_add v15, v16 store v17 at v6 - v18 = not v5 - enable_side_effects v18 - v19 = load v6 -> u8 - v20 = cast v18 as u8 - v21 = cast v4 as u8 - v22 = mul v21, v19 - store v22 at v6 + enable_side_effects v12 + v18 = load v6 -> u8 + v19 = cast v12 as u8 + v20 = cast v4 as u8 + v21 = unchecked_mul v20, v18 + store v21 at v6 enable_side_effects u1 1 constrain v5 == u1 1 return @@ -1326,104 +1324,50 @@ mod test { // Regression test for #1826. Ensures the `else` branch does not see the stores of the // `then` branch. // - // fn main f1 { - // b0(): - // v0 = allocate - // store Field 0 at v0 - // v2 = allocate - // store Field 2 at v2 - // v4 = load v2 - // v5 = lt v4, Field 2 - // jmpif v5 then: b1, else: b2 - // b1(): - // v24 = load v0 - // v25 = load v2 - // v26 = mul v25, Field 10 - // v27 = add v24, v26 - // store v27 at v0 - // v28 = load v2 - // v29 = add v28, Field 1 - // store v29 at v2 - // jmp b5() - // b5(): - // v14 = load v0 - // return v14 - // b2(): - // v6 = load v2 - // v8 = lt v6, Field 4 - // jmpif v8 then: b3, else: b4 - // b3(): - // v16 = load v0 - // v17 = load v2 - // v19 = mul v17, Field 100 - // v20 = add v16, v19 - // store v20 at v0 - // v21 = load v2 - // v23 = add v21, Field 1 - // store v23 at v2 - // jmp b4() - // b4(): - // jmp b5() - // } - let main_id = Id::test_new(0); - let mut builder = FunctionBuilder::new("main".into(), main_id); - - let b1 = builder.insert_block(); - let b2 = builder.insert_block(); - let b3 = builder.insert_block(); - let b4 = builder.insert_block(); - let b5 = builder.insert_block(); - - let zero = builder.field_constant(0u128); - let one = builder.field_constant(1u128); - let two = builder.field_constant(2u128); - let four = builder.field_constant(4u128); - let ten = builder.field_constant(10u128); - let one_hundred = builder.field_constant(100u128); - - let v0 = builder.insert_allocate(Type::field()); - builder.insert_store(v0, zero); - let v2 = builder.insert_allocate(Type::field()); - builder.insert_store(v2, two); - let v4 = builder.insert_load(v2, Type::field()); - let v5 = builder.insert_binary(v4, BinaryOp::Lt, two); - builder.terminate_with_jmpif(v5, b1, b2); - - builder.switch_to_block(b1); - let v24 = builder.insert_load(v0, Type::field()); - let v25 = builder.insert_load(v2, Type::field()); - let v26 = builder.insert_binary(v25, BinaryOp::Mul, ten); - let v27 = builder.insert_binary(v24, BinaryOp::Add, v26); - builder.insert_store(v0, v27); - let v28 = builder.insert_load(v2, Type::field()); - let v29 = builder.insert_binary(v28, BinaryOp::Add, one); - builder.insert_store(v2, v29); - builder.terminate_with_jmp(b5, vec![]); - - builder.switch_to_block(b5); - let v14 = builder.insert_load(v0, Type::field()); - builder.terminate_with_return(vec![v14]); - - builder.switch_to_block(b2); - let v6 = builder.insert_load(v2, Type::field()); - let v8 = builder.insert_binary(v6, BinaryOp::Lt, four); - builder.terminate_with_jmpif(v8, b3, b4); - - builder.switch_to_block(b3); - let v16 = builder.insert_load(v0, Type::field()); - let v17 = builder.insert_load(v2, Type::field()); - let v19 = builder.insert_binary(v17, BinaryOp::Mul, one_hundred); - let v20 = builder.insert_binary(v16, BinaryOp::Add, v19); - builder.insert_store(v0, v20); - let v21 = builder.insert_load(v2, Type::field()); - let v23 = builder.insert_binary(v21, BinaryOp::Add, one); - builder.insert_store(v2, v23); - builder.terminate_with_jmp(b4, vec![]); - - builder.switch_to_block(b4); - builder.terminate_with_jmp(b5, vec![]); - - let ssa = builder.finish().flatten_cfg().mem2reg().fold_constants(); + let src = " + acir(inline) fn main f0 { + b0(): + v0 = allocate -> &mut Field + store Field 0 at v0 + v2 = allocate -> &mut Field + store Field 2 at v2 + v4 = load v2 -> Field + v5 = lt v4, Field 2 + jmpif v5 then: b4, else: b1 + b1(): + v6 = load v2 -> Field + v8 = lt v6, Field 4 + jmpif v8 then: b2, else: b3 + b2(): + v9 = load v0 -> Field + v10 = load v2 -> Field + v12 = mul v10, Field 100 + v13 = add v9, v12 + store v13 at v0 + v14 = load v2 -> Field + v16 = add v14, Field 1 + store v16 at v2 + jmp b3() + b3(): + jmp b5() + b4(): + v17 = load v0 -> Field + v18 = load v2 -> Field + v20 = mul v18, Field 10 + v21 = add v17, v20 + store v21 at v0 + v22 = load v2 -> Field + v23 = add v22, Field 1 + store v23 at v2 + jmp b5() + b5(): + v24 = load v0 -> Field + return v24 + }"; + + let ssa = Ssa::from_str(src).unwrap(); + + let ssa = ssa.flatten_cfg().mem2reg().fold_constants(); let main = ssa.main(); @@ -1440,6 +1384,18 @@ mod test { } _ => unreachable!("Should have terminator instruction"), } + + let expected = " + acir(inline) fn main f0 { + b0(): + v0 = allocate -> &mut Field + v1 = allocate -> &mut Field + enable_side_effects u1 1 + return Field 200 + } + "; + + assert_normalized_ssa_equals(ssa, expected); } #[test] @@ -1460,4 +1416,88 @@ mod test { let merged_ssa = Ssa::from_str(src).unwrap(); let _ = merged_ssa.flatten_cfg(); } + + #[test] + fn eliminates_unnecessary_if_else_instructions_on_numeric_types() { + let src = " + acir(inline) fn main f0 { + b0(v0: bool): + v1 = allocate -> &mut [Field; 1] + store Field 0 at v1 + jmpif v0 then: b1, else: b2 + b1(): + store Field 1 at v1 + store Field 2 at v1 + jmp b2() + b2(): + v3 = load v1 -> Field + return v3 + }"; + + let ssa = Ssa::from_str(src).unwrap(); + + let ssa = ssa.flatten_cfg().mem2reg().fold_constants(); + + let expected = " + acir(inline) fn main f0 { + b0(v0: u1): + v1 = allocate -> &mut [Field; 1] + enable_side_effects v0 + v2 = not v0 + v3 = cast v0 as Field + v4 = cast v2 as Field + v6 = mul v3, Field 2 + v7 = mul v4, v3 + v8 = add v6, v7 + enable_side_effects u1 1 + return v8 + } + "; + + assert_normalized_ssa_equals(ssa, expected); + } + + #[test] + fn eliminates_unnecessary_if_else_instructions_on_array_types() { + let src = " + acir(inline) fn main f0 { + b0(v0: bool, v1: bool): + v2 = make_array [Field 0] : [Field; 1] + v3 = allocate -> &mut [Field; 1] + store v2 at v3 + jmpif v0 then: b1, else: b2 + b1(): + v4 = make_array [Field 1] : [Field; 1] + store v4 at v3 + v5 = make_array [Field 2] : [Field; 1] + store v5 at v3 + jmp b2() + b2(): + v24 = load v3 -> Field + return v24 + }"; + + let ssa = Ssa::from_str(src).unwrap(); + + let ssa = ssa + .flatten_cfg() + .mem2reg() + .remove_if_else() + .fold_constants() + .dead_instruction_elimination(); + + let expected = " + acir(inline) fn main f0 { + b0(v0: u1, v1: u1): + enable_side_effects v0 + v2 = cast v0 as Field + v4 = mul v2, Field 2 + v5 = make_array [v4] : [Field; 1] + enable_side_effects u1 1 + return v5 + } + "; + + assert_normalized_ssa_equals(ssa, expected); + } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/branch_analysis.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/branch_analysis.rs index ce54bb533f7..78091285208 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/branch_analysis.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/branch_analysis.rs @@ -2,14 +2,14 @@ //! //! The algorithm is split into two parts: //! 1. The outer part: -//! A. An (unrolled) CFG can be though of as a linear sequence of blocks where some nodes split -//! off, but eventually rejoin to a new node and continue the linear sequence. -//! B. Follow this sequence in order, and whenever a split is found call -//! `find_join_point_of_branches` and then recur from the join point it returns until the -//! return instruction is found. +//! A. An (unrolled) CFG can be though of as a linear sequence of blocks where some nodes split +//! off, but eventually rejoin to a new node and continue the linear sequence. +//! B. Follow this sequence in order, and whenever a split is found call +//! `find_join_point_of_branches` and then recur from the join point it returns until the +//! return instruction is found. //! //! 2. The inner part defined by `find_join_point_of_branches`: -//! A. For each of the two branches in a jmpif block: +//! A. For each of the two branches in a jmpif block: //! - Check if either has multiple predecessors. If so, it is a join point. //! - If not, continue to search the linear sequence of successor blocks from that block. //! - If another split point is found, recur in `find_join_point_of_branches` @@ -169,8 +169,8 @@ mod test { builder.switch_to_block(b9); builder.terminate_with_return(vec![]); - let mut ssa = builder.finish(); - let function = ssa.main_mut(); + let ssa = builder.finish(); + let function = ssa.main(); let cfg = ControlFlowGraph::with_function(function); let branch_ends = find_branch_ends(function, &cfg); assert_eq!(branch_ends.len(), 2); @@ -253,8 +253,8 @@ mod test { builder.switch_to_block(b15); builder.terminate_with_return(vec![]); - let mut ssa = builder.finish(); - let function = ssa.main_mut(); + let ssa = builder.finish(); + let function = ssa.main(); let cfg = ControlFlowGraph::with_function(function); find_branch_ends(function, &cfg); } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/capacity_tracker.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/capacity_tracker.rs deleted file mode 100644 index a01be691778..00000000000 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/capacity_tracker.rs +++ /dev/null @@ -1,171 +0,0 @@ -use crate::ssa::ir::{ - dfg::DataFlowGraph, - instruction::{Instruction, Intrinsic}, - types::Type, - value::{Value, ValueId}, -}; - -use acvm::{acir::AcirField, FieldElement}; -use fxhash::FxHashMap as HashMap; - -pub(crate) struct SliceCapacityTracker<'a> { - dfg: &'a DataFlowGraph, -} - -impl<'a> SliceCapacityTracker<'a> { - pub(crate) fn new(dfg: &'a DataFlowGraph) -> Self { - SliceCapacityTracker { dfg } - } - - /// Determine how the slice sizes map needs to be updated according to the provided instruction. - pub(crate) fn collect_slice_information( - &self, - instruction: &Instruction, - slice_sizes: &mut HashMap, - results: &[ValueId], - ) { - match instruction { - Instruction::ArrayGet { array, .. } => { - if let Some((_, array_type)) = self.dfg.get_array_constant(*array) { - if array_type.contains_slice_element() { - // Initial insertion into the slice sizes map - // Any other insertions should only occur if the value is already - // a part of the map. - self.compute_slice_capacity(*array, slice_sizes); - } - } - } - Instruction::ArraySet { array, value, .. } => { - if let Some((_, array_type)) = self.dfg.get_array_constant(*array) { - if array_type.contains_slice_element() { - // Initial insertion into the slice sizes map - // Any other insertions should only occur if the value is already - // a part of the map. - self.compute_slice_capacity(*array, slice_sizes); - } - } - - let value_typ = self.dfg.type_of_value(*value); - // Compiler sanity check - assert!(!value_typ.contains_slice_element(), "ICE: Nested slices are not allowed and should not have reached the flattening pass of SSA"); - - if let Some(capacity) = slice_sizes.get(array) { - slice_sizes.insert(results[0], *capacity); - } - } - Instruction::Call { func, arguments } => { - let func = &self.dfg[*func]; - if let Value::Intrinsic(intrinsic) = func { - let (argument_index, result_index) = match intrinsic { - Intrinsic::SlicePushBack - | Intrinsic::SlicePushFront - | Intrinsic::SlicePopBack - | Intrinsic::SliceInsert - | Intrinsic::SliceRemove => (1, 1), - // `pop_front` returns the popped element, and then the respective slice. - // This means in the case of a slice with structs, the result index of the popped slice - // will change depending on the number of elements in the struct. - // For example, a slice with four elements will look as such in SSA: - // v3, v4, v5, v6, v7, v8 = call slice_pop_front(v1, v2) - // where v7 is the slice length and v8 is the popped slice itself. - Intrinsic::SlicePopFront => (1, results.len() - 1), - Intrinsic::AsSlice => (0, 1), - _ => return, - }; - let result_slice = results[result_index]; - match intrinsic { - Intrinsic::SlicePushBack - | Intrinsic::SlicePushFront - | Intrinsic::SliceInsert => { - let slice_contents = arguments[argument_index]; - - for arg in &arguments[(argument_index + 1)..] { - let element_typ = self.dfg.type_of_value(*arg); - if element_typ.contains_slice_element() { - self.compute_slice_capacity(*arg, slice_sizes); - } - } - - if let Some(contents_capacity) = slice_sizes.get(&slice_contents) { - let new_capacity = *contents_capacity + 1; - slice_sizes.insert(result_slice, new_capacity); - } - } - Intrinsic::SlicePopBack - | Intrinsic::SliceRemove - | Intrinsic::SlicePopFront => { - let slice_contents = arguments[argument_index]; - - if let Some(contents_capacity) = slice_sizes.get(&slice_contents) { - // We use a saturating sub here as calling `pop_front` or `pop_back` - // on a zero-length slice would otherwise underflow. - let new_capacity = contents_capacity.saturating_sub(1); - slice_sizes.insert(result_slice, new_capacity); - } - } - Intrinsic::ToBits(_) => { - // Compiler sanity check - assert!(matches!(self.dfg.type_of_value(result_slice), Type::Slice(_))); - slice_sizes.insert(result_slice, FieldElement::max_num_bits()); - } - Intrinsic::ToRadix(_) => { - // Compiler sanity check - assert!(matches!(self.dfg.type_of_value(result_slice), Type::Slice(_))); - slice_sizes.insert(result_slice, FieldElement::max_num_bytes()); - } - Intrinsic::AsSlice => { - let array_size = self - .dfg - .try_get_array_length(arguments[argument_index]) - .expect("ICE: Should be have an array length for AsSlice input"); - slice_sizes.insert(result_slice, array_size); - } - _ => {} - } - } - } - Instruction::Store { address, value } => { - let value_typ = self.dfg.type_of_value(*value); - if value_typ.contains_slice_element() { - self.compute_slice_capacity(*value, slice_sizes); - - let value_capacity = slice_sizes.get(value).unwrap_or_else(|| { - panic!("ICE: should have slice capacity set for value {value} being stored at {address}") - }); - - slice_sizes.insert(*address, *value_capacity); - } - } - Instruction::Load { address } => { - let load_typ = self.dfg.type_of_value(*address); - if load_typ.contains_slice_element() { - let result = results[0]; - - let address_capacity = slice_sizes.get(address).unwrap_or_else(|| { - panic!("ICE: should have slice capacity set at address {address} being loaded into {result}") - }); - - slice_sizes.insert(result, *address_capacity); - } - } - _ => {} - } - } - - /// Computes the starting capacity of a slice which is still a `Value::Array` - pub(crate) fn compute_slice_capacity( - &self, - array_id: ValueId, - slice_sizes: &mut HashMap, - ) { - if let Some((array, typ)) = self.dfg.get_array_constant(array_id) { - // Compiler sanity check - assert!(!typ.is_nested_slice(), "ICE: Nested slices are not allowed and should not have reached the flattening pass of SSA"); - if let Type::Slice(_) = typ { - let element_size = typ.element_size(); - let len = array.len() / element_size; - slice_sizes.insert(array_id, len as u32); - } - } - } -} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/value_merger.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/value_merger.rs index df351d6c0cd..f4638cf85e4 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/value_merger.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/flatten_cfg/value_merger.rs @@ -121,13 +121,18 @@ impl<'a> ValueMerger<'a> { let else_condition = dfg.insert_instruction_and_results(cast, block, None, call_stack).first(); - let mul = Instruction::binary(BinaryOp::Mul, then_condition, then_value); + // Unchecked mul because `then_condition` will be 1 or 0 + let mul = + Instruction::binary(BinaryOp::Mul { unchecked: true }, then_condition, then_value); let then_value = dfg.insert_instruction_and_results(mul, block, None, call_stack).first(); - let mul = Instruction::binary(BinaryOp::Mul, else_condition, else_value); + // Unchecked mul because `else_condition` will be 1 or 0 + let mul = + Instruction::binary(BinaryOp::Mul { unchecked: true }, else_condition, else_value); let else_value = dfg.insert_instruction_and_results(mul, block, None, call_stack).first(); - let add = Instruction::binary(BinaryOp::Add, then_value, else_value); + // Unchecked add because one of the values will always be 0 + let add = Instruction::binary(BinaryOp::Add { unchecked: true }, then_value, else_value); dfg.insert_instruction_and_results(add, block, None, call_stack).first() } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/hint.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/hint.rs index 567a0795edc..1326c2cc010 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/hint.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/hint.rs @@ -14,7 +14,6 @@ mod tests { let options = &SsaEvaluatorOptions { ssa_logging: SsaLogging::None, enable_brillig_logging: false, - force_brillig_output: false, print_codegen_timings: false, expression_width: ExpressionWidth::default(), emit_ssa: None, diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/inlining.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/inlining.rs index 11201fc8f85..7554ad64a9c 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/inlining.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/inlining.rs @@ -46,29 +46,53 @@ impl Ssa { /// This step should run after runtime separation, since it relies on the runtime of the called functions being final. #[tracing::instrument(level = "trace", skip(self))] pub(crate) fn inline_functions(self, aggressiveness: i64) -> Ssa { - Self::inline_functions_inner(self, aggressiveness, false) + let inline_sources = get_functions_to_inline_into(&self, false, aggressiveness); + Self::inline_functions_inner(self, &inline_sources, false) } // Run the inlining pass where functions marked with `InlineType::NoPredicates` as not entry points pub(crate) fn inline_functions_with_no_predicates(self, aggressiveness: i64) -> Ssa { - Self::inline_functions_inner(self, aggressiveness, true) + let inline_sources = get_functions_to_inline_into(&self, true, aggressiveness); + Self::inline_functions_inner(self, &inline_sources, true) } fn inline_functions_inner( mut self, - aggressiveness: i64, + inline_sources: &BTreeSet, inline_no_predicates_functions: bool, ) -> Ssa { - let inline_sources = - get_functions_to_inline_into(&self, inline_no_predicates_functions, aggressiveness); - self.functions = btree_map(&inline_sources, |entry_point| { - let new_function = InlineContext::new( - &self, - *entry_point, - inline_no_predicates_functions, - inline_sources.clone(), - ) - .inline_all(&self); + // Note that we clear all functions other than those in `inline_sources`. + // If we decide to do partial inlining then we should change this to preserve those functions which still exist. + self.functions = btree_map(inline_sources, |entry_point| { + let should_inline_call = + |_context: &PerFunctionContext, ssa: &Ssa, called_func_id: FunctionId| -> bool { + let callee = &ssa.functions[&called_func_id]; + let caller_runtime = ssa.functions[entry_point].runtime(); + + match callee.runtime() { + RuntimeType::Acir(inline_type) => { + // If the called function is acir, we inline if it's not an entry point + + // If we have not already finished the flattening pass, functions marked + // to not have predicates should be preserved. + let preserve_function = + !inline_no_predicates_functions && callee.is_no_predicates(); + !inline_type.is_entry_point() && !preserve_function + } + RuntimeType::Brillig(_) => { + if caller_runtime.is_acir() { + // We never inline a brillig function into an ACIR function. + return false; + } + + // Avoid inlining recursive functions. + !inline_sources.contains(&called_func_id) + } + } + }; + + let new_function = + InlineContext::new(&self, *entry_point).inline_all(&self, &should_inline_call); (*entry_point, new_function) }); self @@ -88,16 +112,6 @@ struct InlineContext { // The FunctionId of the entry point function we're inlining into in the old, unmodified Ssa. entry_point: FunctionId, - - /// Whether the inlining pass should inline any functions marked with [`InlineType::NoPredicates`] - /// or whether these should be preserved as entrypoint functions. - /// - /// This is done as we delay inlining of functions with the attribute `#[no_predicates]` until after - /// the control flow graph has been flattened. - inline_no_predicates_functions: bool, - - // These are the functions of the program that we shouldn't inline. - functions_not_to_inline: BTreeSet, } /// The per-function inlining context contains information that is only valid for one function. @@ -129,6 +143,8 @@ struct PerFunctionContext<'function> { /// True if we're currently working on the entry point function. inlining_entry: bool, + + globals: &'function Function, } /// Utility function to find out the direct calls of a function. @@ -353,32 +369,30 @@ impl InlineContext { /// The function being inlined into will always be the main function, although it is /// actually a copy that is created in case the original main is still needed from a function /// that could not be inlined calling it. - fn new( - ssa: &Ssa, - entry_point: FunctionId, - inline_no_predicates_functions: bool, - functions_not_to_inline: BTreeSet, - ) -> InlineContext { + fn new(ssa: &Ssa, entry_point: FunctionId) -> Self { let source = &ssa.functions[&entry_point]; let mut builder = FunctionBuilder::new(source.name().to_owned(), entry_point); builder.set_runtime(source.runtime()); - Self { - builder, - recursion_level: 0, - entry_point, - call_stack: CallStackId::root(), - inline_no_predicates_functions, - functions_not_to_inline, - } + builder.current_function.set_globals(source.dfg.globals.clone()); + + Self { builder, recursion_level: 0, entry_point, call_stack: CallStackId::root() } } /// Start inlining the entry point function and all functions reachable from it. - fn inline_all(mut self, ssa: &Ssa) -> Function { + fn inline_all( + mut self, + ssa: &Ssa, + should_inline_call: &impl Fn(&PerFunctionContext, &Ssa, FunctionId) -> bool, + ) -> Function { let entry_point = &ssa.functions[&self.entry_point]; - let mut context = PerFunctionContext::new(&mut self, entry_point); + let mut context = PerFunctionContext::new(&mut self, entry_point, &ssa.globals); context.inlining_entry = true; + for (_, value) in entry_point.dfg.globals.values_iter() { + context.context.builder.current_function.dfg.make_global(value.get_type().into_owned()); + } + // The entry block is already inserted so we have to add it to context.blocks and add // its parameters here. Failing to do so would cause context.translate_block() to add // a fresh block for the entry block rather than use the existing one. @@ -392,7 +406,7 @@ impl InlineContext { } context.blocks.insert(context.source_function.entry_block(), entry_block); - context.inline_blocks(ssa); + context.inline_blocks(ssa, should_inline_call); // translate databus values let databus = entry_point.dfg.data_bus.map_values(|t| context.translate_value(t)); @@ -411,6 +425,7 @@ impl InlineContext { ssa: &Ssa, id: FunctionId, arguments: &[ValueId], + should_inline_call: &impl Fn(&PerFunctionContext, &Ssa, FunctionId) -> bool, ) -> Vec { self.recursion_level += 1; @@ -422,7 +437,7 @@ impl InlineContext { ); } - let mut context = PerFunctionContext::new(self, source_function); + let mut context = PerFunctionContext::new(self, source_function, &ssa.globals); let parameters = source_function.parameters(); assert_eq!(parameters.len(), arguments.len()); @@ -431,7 +446,7 @@ impl InlineContext { let current_block = context.context.builder.current_block(); context.blocks.insert(source_function.entry_block(), current_block); - let return_values = context.inline_blocks(ssa); + let return_values = context.inline_blocks(ssa, should_inline_call); self.recursion_level -= 1; return_values } @@ -442,13 +457,18 @@ impl<'function> PerFunctionContext<'function> { /// The value and block mappings for this context are initially empty except /// for containing the mapping between parameters in the source_function and /// the arguments of the destination function. - fn new(context: &'function mut InlineContext, source_function: &'function Function) -> Self { + fn new( + context: &'function mut InlineContext, + source_function: &'function Function, + globals: &'function Function, + ) -> Self { Self { context, source_function, blocks: HashMap::default(), values: HashMap::default(), inlining_entry: false, + globals, } } @@ -458,25 +478,54 @@ impl<'function> PerFunctionContext<'function> { /// and blocks respectively. If these assertions trigger it means a value is being used before /// the instruction or block that defines the value is inserted. fn translate_value(&mut self, id: ValueId) -> ValueId { + let id = self.source_function.dfg.resolve(id); if let Some(value) = self.values.get(&id) { return *value; } let new_value = match &self.source_function.dfg[id] { - value @ Value::Instruction { .. } => { + value @ Value::Instruction { instruction, .. } => { + if self.source_function.dfg.is_global(id) { + if self.context.builder.current_function.dfg.runtime().is_acir() { + let Instruction::MakeArray { elements, typ } = + &self.globals.dfg[*instruction] + else { + panic!("Only expect Instruction::MakeArray for a global"); + }; + let elements = elements + .iter() + .map(|element| self.translate_value(*element)) + .collect::>(); + return self.context.builder.insert_make_array(elements, typ.clone()); + } else { + return id; + } + } unreachable!("All Value::Instructions should already be known during inlining after creating the original inlined instruction. Unknown value {id} = {value:?}") } value @ Value::Param { .. } => { unreachable!("All Value::Params should already be known from previous calls to translate_block. Unknown value {id} = {value:?}") } Value::NumericConstant { constant, typ } => { - self.context.builder.numeric_constant(*constant, *typ) + // The dfg indexes a global's inner value directly, so we need to check here + // whether we have a global. + // We also only keep a global and do not inline it in a Brillig runtime. + if self.source_function.dfg.is_global(id) + && self.context.builder.current_function.dfg.runtime().is_brillig() + { + id + } else { + self.context.builder.numeric_constant(*constant, *typ) + } } Value::Function(function) => self.context.builder.import_function(*function), Value::Intrinsic(intrinsic) => self.context.builder.import_intrinsic_id(*intrinsic), Value::ForeignFunction(function) => { self.context.builder.import_foreign_function(function) } + Value::Global(_) => { + panic!("Expected a global to be resolved to its inner value"); + } }; self.values.insert(id, new_value); @@ -533,7 +582,11 @@ impl<'function> PerFunctionContext<'function> { } /// Inline all reachable blocks within the source_function into the destination function. - fn inline_blocks(&mut self, ssa: &Ssa) -> Vec { + fn inline_blocks( + &mut self, + ssa: &Ssa, + should_inline_call: &impl Fn(&PerFunctionContext, &Ssa, FunctionId) -> bool, + ) -> Vec { let mut seen_blocks = HashSet::new(); let mut block_queue = VecDeque::new(); block_queue.push_back(self.source_function.entry_block()); @@ -550,7 +603,7 @@ impl<'function> PerFunctionContext<'function> { self.context.builder.switch_to_block(translated_block_id); seen_blocks.insert(source_block_id); - self.inline_block_instructions(ssa, source_block_id); + self.inline_block_instructions(ssa, source_block_id, should_inline_call); if let Some((block, values)) = self.handle_terminator_instruction(source_block_id, &mut block_queue) @@ -595,7 +648,12 @@ impl<'function> PerFunctionContext<'function> { /// Inline each instruction in the given block into the function being inlined into. /// This may recurse if it finds another function to inline if a call instruction is within this block. - fn inline_block_instructions(&mut self, ssa: &Ssa, block_id: BasicBlockId) { + fn inline_block_instructions( + &mut self, + ssa: &Ssa, + block_id: BasicBlockId, + should_inline_call: &impl Fn(&PerFunctionContext, &Ssa, FunctionId) -> bool, + ) { let mut side_effects_enabled: Option = None; let block = &self.source_function.dfg[block_id]; @@ -603,8 +661,8 @@ impl<'function> PerFunctionContext<'function> { match &self.source_function.dfg[*id] { Instruction::Call { func, arguments } => match self.get_function(*func) { Some(func_id) => { - if self.should_inline_call(ssa, func_id) { - self.inline_function(ssa, *id, func_id, arguments); + if should_inline_call(self, ssa, func_id) { + self.inline_function(ssa, *id, func_id, arguments, should_inline_call); // This is only relevant during handling functions with `InlineType::NoPredicates` as these // can pollute the function they're being inlined into with `Instruction::EnabledSideEffects`, @@ -632,24 +690,6 @@ impl<'function> PerFunctionContext<'function> { } } - fn should_inline_call(&self, ssa: &Ssa, called_func_id: FunctionId) -> bool { - let function = &ssa.functions[&called_func_id]; - - if let RuntimeType::Acir(inline_type) = function.runtime() { - // If the called function is acir, we inline if it's not an entry point - - // If we have not already finished the flattening pass, functions marked - // to not have predicates should be preserved. - let preserve_function = - !self.context.inline_no_predicates_functions && function.is_no_predicates(); - !inline_type.is_entry_point() && !preserve_function - } else { - // If the called function is brillig, we inline only if it's into brillig and the function is not recursive - matches!(ssa.functions[&self.context.entry_point].runtime(), RuntimeType::Brillig(_)) - && !self.context.functions_not_to_inline.contains(&called_func_id) - } - } - /// Inline a function call and remember the inlined return values in the values map fn inline_function( &mut self, @@ -657,6 +697,7 @@ impl<'function> PerFunctionContext<'function> { call_id: InstructionId, function: FunctionId, arguments: &[ValueId], + should_inline_call: &impl Fn(&PerFunctionContext, &Ssa, FunctionId) -> bool, ) { let old_results = self.source_function.dfg.instruction_results(call_id); let arguments = vecmap(arguments, |arg| self.translate_value(*arg)); @@ -672,7 +713,8 @@ impl<'function> PerFunctionContext<'function> { .extend_call_stack(self.context.call_stack, &call_stack); self.context.call_stack = new_call_stack; - let new_results = self.context.inline_function(ssa, function, &arguments); + let new_results = + self.context.inline_function(ssa, function, &arguments, should_inline_call); self.context.call_stack = self .context .builder @@ -935,7 +977,8 @@ mod test { // Compiling square f1 builder.new_function("square".into(), square_id, InlineType::default()); let square_v0 = builder.add_parameter(Type::field()); - let square_v2 = builder.insert_binary(square_v0, BinaryOp::Mul, square_v0); + let square_v2 = + builder.insert_binary(square_v0, BinaryOp::Mul { unchecked: false }, square_v0); builder.terminate_with_return(vec![square_v2]); // Compiling id1 f2 @@ -1000,9 +1043,9 @@ mod test { builder.switch_to_block(b2); let factorial_id = builder.import_function(factorial_id); - let v2 = builder.insert_binary(v0, BinaryOp::Sub, one); + let v2 = builder.insert_binary(v0, BinaryOp::Sub { unchecked: false }, one); let v3 = builder.insert_call(factorial_id, vec![v2], vec![Type::field()])[0]; - let v4 = builder.insert_binary(v0, BinaryOp::Mul, v3); + let v4 = builder.insert_binary(v0, BinaryOp::Mul { unchecked: false }, v3); builder.terminate_with_return(vec![v4]); let ssa = builder.finish(); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/loop_invariant.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/loop_invariant.rs index 44c0eb380c2..125cf3a12ca 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/loop_invariant.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/loop_invariant.rs @@ -15,7 +15,7 @@ use crate::ssa::{ basic_block::BasicBlockId, function::Function, function_inserter::FunctionInserter, - instruction::{Instruction, InstructionId}, + instruction::{binary::eval_constant_binary_op, BinaryOp, Instruction, InstructionId}, types::Type, value::ValueId, }, @@ -112,10 +112,12 @@ impl<'f> LoopInvariantContext<'f> { // If we are hoisting a MakeArray instruction, // we need to issue an extra inc_rc in case they are mutated afterward. - if matches!( - self.inserter.function.dfg[instruction_id], - Instruction::MakeArray { .. } - ) { + if self.inserter.function.runtime().is_brillig() + && matches!( + self.inserter.function.dfg[instruction_id], + Instruction::MakeArray { .. } + ) + { let result = self.inserter.function.dfg.instruction_results(instruction_id)[0]; let inc_rc = Instruction::IncrementRc { value: result }; @@ -207,6 +209,7 @@ impl<'f> LoopInvariantContext<'f> { let can_be_deduplicated = instruction.can_be_deduplicated(self.inserter.function, false) || matches!(instruction, Instruction::MakeArray { .. }) + || matches!(instruction, Instruction::Binary(_)) || self.can_be_deduplicated_from_upper_bound(&instruction); is_loop_invariant && can_be_deduplicated @@ -231,6 +234,31 @@ impl<'f> LoopInvariantContext<'f> { false } } + Instruction::Binary(binary) => { + if !matches!(binary.operator, BinaryOp::Add { .. } | BinaryOp::Mul { .. }) { + return false; + } + + let operand_type = + self.inserter.function.dfg.type_of_value(binary.lhs).unwrap_numeric(); + + let lhs_const = + self.inserter.function.dfg.get_numeric_constant_with_type(binary.lhs); + let rhs_const = + self.inserter.function.dfg.get_numeric_constant_with_type(binary.rhs); + let (lhs, rhs) = match ( + lhs_const, + rhs_const, + self.outer_induction_variables.get(&binary.lhs), + self.outer_induction_variables.get(&binary.rhs), + ) { + (Some((lhs, _)), None, None, Some(upper_bound)) => (lhs, *upper_bound), + (None, Some((rhs, _)), Some(upper_bound), None) => (*upper_bound, rhs), + _ => return false, + }; + + eval_constant_binary_op(lhs, rhs, binary.operator, operand_type).is_some() + } _ => false, } } @@ -263,23 +291,23 @@ mod test { fn simple_loop_invariant_code_motion() { let src = " brillig(inline) fn main f0 { - b0(v0: u32, v1: u32): - jmp b1(u32 0) - b1(v2: u32): - v5 = lt v2, u32 4 + b0(v0: i32, v1: i32): + jmp b1(i32 0) + b1(v2: i32): + v5 = lt v2, i32 4 jmpif v5 then: b3, else: b2 b2(): return b3(): v6 = mul v0, v1 - constrain v6 == u32 6 - v8 = add v2, u32 1 + constrain v6 == i32 6 + v8 = add v2, i32 1 jmp b1(v8) } "; - let mut ssa = Ssa::from_str(src).unwrap(); - let main = ssa.main_mut(); + let ssa = Ssa::from_str(src).unwrap(); + let main = ssa.main(); let instructions = main.dfg[main.entry_block()].instructions(); assert_eq!(instructions.len(), 0); // The final return is not counted @@ -287,17 +315,17 @@ mod test { // `v6 = mul v0, v1` in b3 should now be `v3 = mul v0, v1` in b0 let expected = " brillig(inline) fn main f0 { - b0(v0: u32, v1: u32): + b0(v0: i32, v1: i32): v3 = mul v0, v1 - jmp b1(u32 0) - b1(v2: u32): - v6 = lt v2, u32 4 + jmp b1(i32 0) + b1(v2: i32): + v6 = lt v2, i32 4 jmpif v6 then: b3, else: b2 b2(): return b3(): - constrain v3 == u32 6 - v9 = add v2, u32 1 + constrain v3 == i32 6 + v9 = add v2, i32 1 jmp b1(v9) } "; @@ -312,31 +340,31 @@ mod test { // is hoisted to the parent loop's pre-header block. let src = " brillig(inline) fn main f0 { - b0(v0: u32, v1: u32): - jmp b1(u32 0) - b1(v2: u32): - v6 = lt v2, u32 4 + b0(v0: i32, v1: i32): + jmp b1(i32 0) + b1(v2: i32): + v6 = lt v2, i32 4 jmpif v6 then: b3, else: b2 b2(): return b3(): - jmp b4(u32 0) - b4(v3: u32): - v7 = lt v3, u32 4 + jmp b4(i32 0) + b4(v3: i32): + v7 = lt v3, i32 4 jmpif v7 then: b6, else: b5 b5(): - v9 = add v2, u32 1 + v9 = add v2, i32 1 jmp b1(v9) b6(): v10 = mul v0, v1 - constrain v10 == u32 6 - v12 = add v3, u32 1 + constrain v10 == i32 6 + v12 = add v3, i32 1 jmp b4(v12) } "; - let mut ssa = Ssa::from_str(src).unwrap(); - let main = ssa.main_mut(); + let ssa = Ssa::from_str(src).unwrap(); + let main = ssa.main(); let instructions = main.dfg[main.entry_block()].instructions(); assert_eq!(instructions.len(), 0); // The final return is not counted @@ -344,25 +372,25 @@ mod test { // `v10 = mul v0, v1` in b6 should now be `v4 = mul v0, v1` in b0 let expected = " brillig(inline) fn main f0 { - b0(v0: u32, v1: u32): + b0(v0: i32, v1: i32): v4 = mul v0, v1 - jmp b1(u32 0) - b1(v2: u32): - v7 = lt v2, u32 4 + jmp b1(i32 0) + b1(v2: i32): + v7 = lt v2, i32 4 jmpif v7 then: b3, else: b2 b2(): return b3(): - jmp b4(u32 0) - b4(v3: u32): - v8 = lt v3, u32 4 + jmp b4(i32 0) + b4(v3: i32): + v8 = lt v3, i32 4 jmpif v8 then: b6, else: b5 b5(): - v10 = add v2, u32 1 + v10 = add v2, i32 1 jmp b1(v10) b6(): - constrain v4 == u32 6 - v12 = add v3, u32 1 + constrain v4 == i32 6 + v12 = add v3, i32 1 jmp b4(v12) } "; @@ -386,44 +414,44 @@ mod test { // hoist `v7 = mul v6, v0`. let src = " brillig(inline) fn main f0 { - b0(v0: u32, v1: u32): - jmp b1(u32 0) - b1(v2: u32): - v5 = lt v2, u32 4 + b0(v0: i32, v1: i32): + jmp b1(i32 0) + b1(v2: i32): + v5 = lt v2, i32 4 jmpif v5 then: b3, else: b2 b2(): return b3(): v6 = mul v0, v1 v7 = mul v6, v0 - v8 = eq v7, u32 12 - constrain v7 == u32 12 - v9 = add v2, u32 1 + v8 = eq v7, i32 12 + constrain v7 == i32 12 + v9 = add v2, i32 1 jmp b1(v9) } "; - let mut ssa = Ssa::from_str(src).unwrap(); - let main = ssa.main_mut(); + let ssa = Ssa::from_str(src).unwrap(); + let main = ssa.main(); let instructions = main.dfg[main.entry_block()].instructions(); assert_eq!(instructions.len(), 0); // The final return is not counted let expected = " brillig(inline) fn main f0 { - b0(v0: u32, v1: u32): + b0(v0: i32, v1: i32): v3 = mul v0, v1 v4 = mul v3, v0 - v6 = eq v4, u32 12 - jmp b1(u32 0) - b1(v2: u32): - v9 = lt v2, u32 4 + v6 = eq v4, i32 12 + jmp b1(i32 0) + b1(v2: i32): + v9 = lt v2, i32 4 jmpif v9 then: b3, else: b2 b2(): return b3(): - constrain v4 == u32 12 - v11 = add v2, u32 1 + constrain v4 == i32 12 + v11 = add v2, i32 1 jmp b1(v11) } "; @@ -462,8 +490,8 @@ mod test { } "; - let mut ssa = Ssa::from_str(src).unwrap(); - let main = ssa.main_mut(); + let ssa = Ssa::from_str(src).unwrap(); + let main = ssa.main(); let instructions = main.dfg[main.entry_block()].instructions(); assert_eq!(instructions.len(), 4); // The final return is not counted diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/make_constrain_not_equal.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/make_constrain_not_equal.rs new file mode 100644 index 00000000000..21f536eba2d --- /dev/null +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/make_constrain_not_equal.rs @@ -0,0 +1,72 @@ +use acvm::AcirField; + +use crate::ssa::{ + ir::{ + function::Function, + instruction::{Binary, BinaryOp, Instruction}, + value::Value, + }, + ssa_gen::Ssa, +}; + +impl Ssa { + /// A simple SSA pass to go through each [`Instruction::Constrain`], determine whether it's asserting + /// two values are not equal, and if so replace it with a [`Instruction::ConstrainNotEqual`]. + /// + /// Note that this pass must be placed after CFG flattening as the flattening pass cannot + /// handle this instruction. + #[tracing::instrument(level = "trace", skip(self))] + pub(crate) fn make_constrain_not_equal_instructions(mut self) -> Ssa { + for function in self.functions.values_mut() { + function.make_constrain_not_equal(); + } + self + } +} + +impl Function { + pub(crate) fn make_constrain_not_equal(&mut self) { + if !self.runtime().is_acir() { + return; + } + + for block in self.reachable_blocks() { + let instructions = self.dfg[block].instructions().to_vec(); + + for instruction in instructions { + let constrain_ne: Instruction = match &self.dfg[instruction] { + Instruction::Constrain(lhs, rhs, msg) => { + if self + .dfg + .get_numeric_constant(*rhs) + .map_or(false, |constant| constant.is_zero()) + { + if let Value::Instruction { instruction, .. } = + &self.dfg[self.dfg.resolve(*lhs)] + { + if let Instruction::Binary(Binary { + lhs, + rhs, + operator: BinaryOp::Eq, + .. + }) = self.dfg[*instruction] + { + Instruction::ConstrainNotEqual(lhs, rhs, msg.clone()) + } else { + continue; + } + } else { + continue; + } + } else { + continue; + } + } + _ => continue, + }; + + self.dfg[instruction] = constrain_ne; + } + } + } +} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg.rs index 1e5cd8bdfbd..ce76825877a 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg.rs @@ -32,9 +32,11 @@ //! - We also track the last instance of a load instruction to each address in a block. //! If we see that the last load instruction was from the same address as the current load instruction, //! we move to replace the result of the current load with the result of the previous load. +//! //! This removal requires a couple conditions: -//! - No store occurs to that address before the next load, -//! - The address is not used as an argument to a call +//! - No store occurs to that address before the next load, +//! - The address is not used as an argument to a call +//! //! This optimization helps us remove repeated loads for which there are not known values. //! - On `Instruction::Store { address, value }`: //! - If the address of the store is known: @@ -77,6 +79,7 @@ mod block; use std::collections::{BTreeMap, BTreeSet}; use fxhash::{FxHashMap as HashMap, FxHashSet as HashSet}; +use vec_collections::VecSet; use crate::ssa::{ ir::{ @@ -200,7 +203,7 @@ impl<'f> PerFunctionContext<'f> { .get(store_address) .map_or(false, |expression| matches!(expression, Expression::Dereference(_))); - if self.last_loads.get(store_address).is_none() + if !self.last_loads.contains_key(store_address) && !store_alias_used && !is_dereference { @@ -617,7 +620,7 @@ impl<'f> PerFunctionContext<'f> { // then those parameters also alias each other. // We save parameters with repeat arguments to later mark those // parameters as aliasing one another. - let mut arg_set: HashMap> = HashMap::default(); + let mut arg_set = HashMap::default(); // Add an alias for each reference parameter for (parameter, argument) in destination_parameters.iter().zip(arguments) { @@ -630,7 +633,8 @@ impl<'f> PerFunctionContext<'f> { aliases.insert(*parameter); // Check if we have seen the same argument - let seen_parameters = arg_set.entry(argument).or_default(); + let seen_parameters = + arg_set.entry(argument).or_insert_with(VecSet::empty); // Add the current parameter to the parameters we have seen for this argument. // The previous parameters and the current one alias one another. seen_parameters.insert(*parameter); @@ -907,8 +911,8 @@ mod tests { } "; - let mut ssa = Ssa::from_str(src).unwrap(); - let main = ssa.main_mut(); + let ssa = Ssa::from_str(src).unwrap(); + let main = ssa.main(); let instructions = main.dfg[main.entry_block()].instructions(); assert_eq!(instructions.len(), 6); // The final return is not counted @@ -1000,7 +1004,7 @@ mod tests { let two = builder.field_constant(2u128); builder.insert_store(v5, two); let one = builder.field_constant(1u128); - let v3_plus_one = builder.insert_binary(v3, BinaryOp::Add, one); + let v3_plus_one = builder.insert_binary(v3, BinaryOp::Add { unchecked: false }, one); builder.terminate_with_jmp(b1, vec![v3_plus_one]); builder.switch_to_block(b3); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg/alias_set.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg/alias_set.rs index e32eaa70186..443b21cfd15 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg/alias_set.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg/alias_set.rs @@ -1,4 +1,4 @@ -use std::collections::BTreeSet; +use vec_collections::{AbstractVecSet, VecSet}; use crate::ssa::ir::value::ValueId; @@ -10,7 +10,7 @@ use crate::ssa::ir::value::ValueId; /// "unknown which aliases this may refer to" - `None`. #[derive(Debug, Default, Clone)] pub(super) struct AliasSet { - aliases: Option>, + aliases: Option>, } impl AliasSet { @@ -19,12 +19,10 @@ impl AliasSet { } pub(super) fn known(value: ValueId) -> AliasSet { - let mut aliases = BTreeSet::new(); - aliases.insert(value); - Self { aliases: Some(aliases) } + Self { aliases: Some(VecSet::single(value)) } } - pub(super) fn known_multiple(values: BTreeSet) -> AliasSet { + pub(super) fn known_multiple(values: VecSet<[ValueId; 1]>) -> AliasSet { Self { aliases: Some(values) } } @@ -32,7 +30,7 @@ impl AliasSet { /// particular value will be known to be zero, which is distinct from being unknown and /// possibly referring to any alias. pub(super) fn known_empty() -> AliasSet { - Self { aliases: Some(BTreeSet::new()) } + Self { aliases: Some(VecSet::empty()) } } pub(super) fn is_unknown(&self) -> bool { @@ -44,19 +42,33 @@ impl AliasSet { pub(super) fn single_alias(&self) -> Option { self.aliases .as_ref() - .and_then(|aliases| (aliases.len() == 1).then(|| *aliases.first().unwrap())) + .and_then(|aliases| (aliases.len() == 1).then(|| *aliases.iter().next().unwrap())) } /// Unify this alias set with another. The result of this set is empty if either set is empty. /// Otherwise, it is the union of both alias sets. pub(super) fn unify(&mut self, other: &Self) { if let (Some(self_aliases), Some(other_aliases)) = (&mut self.aliases, &other.aliases) { - self_aliases.extend(other_aliases); + self_aliases.extend(other_aliases.iter().cloned()); } else { self.aliases = None; } } + /// Returns true if calling `unify` would change something in this alias set. + /// + /// This is an optimization to avoid having to look up an entry ready to be modified in the [Block](crate::ssa::opt::mem2reg::block::Block), + /// because doing so would involve calling `Arc::make_mut` which clones the entry, ready for modification. + pub(super) fn should_unify(&self, other: &Self) -> bool { + if let (Some(self_aliases), Some(other_aliases)) = (&self.aliases, &other.aliases) { + // `unify` would extend `self_aliases` with `other_aliases`, so if `other_aliases` is a subset, then nothing would happen. + !other_aliases.is_subset(self_aliases) + } else { + // `unify` would set `aliases` to `None`, so if it's not `Some`, then nothing would happen. + self.aliases.is_some() + } + } + /// Inserts a new alias into this set if it is not unknown pub(super) fn insert(&mut self, new_alias: ValueId) { if let Some(aliases) = &mut self.aliases { @@ -82,6 +94,6 @@ impl AliasSet { /// The ordering is arbitrary (by lowest ValueId) so this method should only be /// used when you need an arbitrary ValueId from the alias set. pub(super) fn first(&self) -> Option { - self.aliases.as_ref().and_then(|aliases| aliases.first().copied()) + self.aliases.as_ref().and_then(|aliases| aliases.iter().next().copied()) } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg/block.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg/block.rs index f4265b2466d..91e27f07b8e 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg/block.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mem2reg/block.rs @@ -128,6 +128,12 @@ impl Block { } for (expression, new_aliases) in &other.aliases { + // If nothing would change, then don't call `.entry(...).and_modify(...)` as it involves creating more `Arc`s. + if let Some(aliases) = self.aliases.get(expression) { + if !aliases.should_unify(new_aliases) { + continue; + } + } let expression = expression.clone(); self.aliases diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mod.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mod.rs index bd0c86570e2..f97d36f0844 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mod.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/mod.rs @@ -14,14 +14,14 @@ pub(crate) mod flatten_cfg; mod hint; mod inlining; mod loop_invariant; +mod make_constrain_not_equal; mod mem2reg; mod normalize_value_ids; mod rc; mod remove_bit_shifts; mod remove_enable_side_effects; mod remove_if_else; -mod resolve_is_unconstrained; -mod runtime_separation; +mod remove_unreachable; mod simplify_cfg; mod unrolling; diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/normalize_value_ids.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/normalize_value_ids.rs index 63ca523bd57..b248f6734a9 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/normalize_value_ids.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/normalize_value_ids.rs @@ -72,6 +72,10 @@ impl Context { let new_function_id = self.new_ids.function_ids[&old_function.id()]; let new_function = &mut self.functions[new_function_id]; + for (_, value) in old_function.dfg.globals.values_iter() { + new_function.dfg.make_global(value.get_type().into_owned()); + } + let mut reachable_blocks = PostOrder::with_function(old_function).into_vec(); reachable_blocks.reverse(); @@ -96,12 +100,13 @@ impl Context { .requires_ctrl_typevars() .then(|| vecmap(old_results, |result| old_function.dfg.type_of_value(*result))); - let new_results = new_function.dfg.insert_instruction_and_results( - instruction, - new_block_id, - ctrl_typevars, - new_call_stack, - ); + let new_results = + new_function.dfg.insert_instruction_and_results_without_simplification( + instruction, + new_block_id, + ctrl_typevars, + new_call_stack, + ); assert_eq!(old_results.len(), new_results.len()); for (old_result, new_result) in old_results.iter().zip(new_results.results().iter()) @@ -165,6 +170,11 @@ impl IdMaps { old_value: ValueId, ) -> ValueId { let old_value = old_function.dfg.resolve(old_value); + if old_function.dfg.is_global(old_value) { + // Globals are computed at compile-time and thus are expected to be remain normalized + // between SSA passes + return old_value; + } match &old_function.dfg[old_value] { value @ Value::Instruction { instruction, .. } => { *self.values.get(&old_value).unwrap_or_else(|| { @@ -180,7 +190,9 @@ impl IdMaps { } Value::Function(id) => { - let new_id = self.function_ids[id]; + let new_id = *self.function_ids.get(id).unwrap_or_else(|| { + unreachable!("Unmapped function with id {id}") + }); new_function.dfg.import_function(new_id) } @@ -189,6 +201,9 @@ impl IdMaps { } Value::Intrinsic(intrinsic) => new_function.dfg.import_intrinsic(*intrinsic), Value::ForeignFunction(name) => new_function.dfg.import_foreign_function(name), + Value::Global(_) => { + unreachable!("Should have handled the global case already"); + }, } } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/rc.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/rc.rs index 64f6e2ddfea..e25ad350145 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/rc.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/rc.rs @@ -246,6 +246,7 @@ mod test { // } let main_id = Id::test_new(0); let mut builder = FunctionBuilder::new("mutator".into(), main_id); + builder.set_runtime(RuntimeType::Brillig(InlineType::default())); let array_type = Type::Array(Arc::new(vec![Type::field()]), 2); let v0 = builder.add_parameter(array_type.clone()); @@ -295,6 +296,7 @@ mod test { // } let main_id = Id::test_new(0); let mut builder = FunctionBuilder::new("mutator2".into(), main_id); + builder.set_runtime(RuntimeType::Brillig(InlineType::default())); let array_type = Type::Array(Arc::new(vec![Type::field()]), 2); let reference_type = Type::Reference(Arc::new(array_type.clone())); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_bit_shifts.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_bit_shifts.rs index 4c5189d8c91..e36be71aeea 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_bit_shifts.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_bit_shifts.rs @@ -7,7 +7,7 @@ use crate::ssa::{ basic_block::BasicBlockId, call_stack::CallStackId, dfg::InsertInstructionResult, - function::{Function, RuntimeType}, + function::Function, instruction::{Binary, BinaryOp, Endian, Instruction, InstructionId, Intrinsic}, types::{NumericType, Type}, value::ValueId, @@ -32,7 +32,7 @@ impl Function { /// The structure of this pass is simple: /// Go through each block and re-insert all instructions. pub(crate) fn remove_bit_shifts(&mut self) { - if matches!(self.runtime(), RuntimeType::Brillig(_)) { + if self.runtime().is_brillig() { return; } @@ -120,27 +120,36 @@ impl Context<'_> { let pow = self.numeric_constant(FieldElement::from(rhs_bit_size_pow_2), typ); let max_lhs_bits = self.function.dfg.get_value_max_num_bits(lhs); - - (max_lhs_bits + bit_shift_size, pow) + let max_bit_size = max_lhs_bits + bit_shift_size; + // There is no point trying to truncate to more than the Field size. + // A higher `max_lhs_bits` input can come from trying to left-shift a Field. + let max_bit_size = max_bit_size.min(NumericType::NativeField.bit_size()); + (max_bit_size, pow) } else { // we use a predicate to nullify the result in case of overflow let u8_type = NumericType::unsigned(8); let bit_size_var = self.numeric_constant(FieldElement::from(bit_size as u128), u8_type); let overflow = self.insert_binary(rhs, BinaryOp::Lt, bit_size_var); let predicate = self.insert_cast(overflow, typ); - // we can safely cast to unsigned because overflow_checks prevent bit-shift with a negative value - let rhs_unsigned = self.insert_cast(rhs, NumericType::unsigned(bit_size)); - let pow = self.pow(base, rhs_unsigned); + let pow = self.pow(base, rhs); let pow = self.insert_cast(pow, typ); - (FieldElement::max_num_bits(), self.insert_binary(predicate, BinaryOp::Mul, pow)) + + // Unchecked mul because `predicate` will be 1 or 0 + ( + FieldElement::max_num_bits(), + self.insert_binary(predicate, BinaryOp::Mul { unchecked: true }, pow), + ) }; if max_bit <= bit_size { - self.insert_binary(lhs, BinaryOp::Mul, pow) + // Unchecked mul as it can't overflow + self.insert_binary(lhs, BinaryOp::Mul { unchecked: true }, pow) } else { let lhs_field = self.insert_cast(lhs, NumericType::NativeField); let pow_field = self.insert_cast(pow, NumericType::NativeField); - let result = self.insert_binary(lhs_field, BinaryOp::Mul, pow_field); + // Unchecked mul as this is a wrapping operation that we later truncate + let result = + self.insert_binary(lhs_field, BinaryOp::Mul { unchecked: true }, pow_field); let result = self.insert_truncate(result, bit_size, max_bit); self.insert_cast(result, typ) } @@ -158,6 +167,7 @@ impl Context<'_> { let lhs_typ = self.function.dfg.type_of_value(lhs).unwrap_numeric(); let base = self.field_constant(FieldElement::from(2_u128)); let pow = self.pow(base, rhs); + let pow = self.insert_cast(pow, lhs_typ); if lhs_typ.is_unsigned() { // unsigned right bit shift is just a normal division self.insert_binary(lhs, BinaryOp::Div, pow) @@ -168,14 +178,29 @@ impl Context<'_> { let lhs_sign_as_field = self.insert_cast(lhs_sign, NumericType::NativeField); let lhs_as_field = self.insert_cast(lhs, NumericType::NativeField); // For negative numbers, convert to 1-complement using wrapping addition of a + 1 - let one_complement = self.insert_binary(lhs_sign_as_field, BinaryOp::Add, lhs_as_field); + // Unchecked add as these are fields + let one_complement = self.insert_binary( + lhs_sign_as_field, + BinaryOp::Add { unchecked: true }, + lhs_as_field, + ); let one_complement = self.insert_truncate(one_complement, bit_size, bit_size + 1); let one_complement = self.insert_cast(one_complement, NumericType::signed(bit_size)); // Performs the division on the 1-complement (or the operand if positive) let shifted_complement = self.insert_binary(one_complement, BinaryOp::Div, pow); // Convert back to 2-complement representation if operand is negative let lhs_sign_as_int = self.insert_cast(lhs_sign, lhs_typ); - let shifted = self.insert_binary(shifted_complement, BinaryOp::Sub, lhs_sign_as_int); + + // The requirements for this to underflow are all of these: + // - lhs < 0 + // - ones_complement(lhs) / (2^rhs) == 0 + // As the upper bit is set for the ones complement of negative numbers we'd need 2^rhs + // to be larger than the lhs bitsize for this to overflow. + let shifted = self.insert_binary( + shifted_complement, + BinaryOp::Sub { unchecked: true }, + lhs_sign_as_int, + ); self.insert_truncate(shifted, bit_size, bit_size + 1) } } @@ -199,17 +224,18 @@ impl Context<'_> { let rhs_bits = rhs_bits[0]; let one = self.field_constant(FieldElement::one()); let mut r = one; + // All operations are unchecked as we're acting on Field types (which are always unchecked) for i in 1..bit_size + 1 { - let r_squared = self.insert_binary(r, BinaryOp::Mul, r); - let a = self.insert_binary(r_squared, BinaryOp::Mul, lhs); + let r_squared = self.insert_binary(r, BinaryOp::Mul { unchecked: true }, r); + let a = self.insert_binary(r_squared, BinaryOp::Mul { unchecked: true }, lhs); let idx = self.field_constant(FieldElement::from((bit_size - i) as i128)); let b = self.insert_array_get(rhs_bits, idx, Type::bool()); let not_b = self.insert_not(b); let b = self.insert_cast(b, NumericType::NativeField); let not_b = self.insert_cast(not_b, NumericType::NativeField); - let r1 = self.insert_binary(a, BinaryOp::Mul, b); - let r2 = self.insert_binary(r_squared, BinaryOp::Mul, not_b); - r = self.insert_binary(r1, BinaryOp::Add, r2); + let r1 = self.insert_binary(a, BinaryOp::Mul { unchecked: true }, b); + let r2 = self.insert_binary(r_squared, BinaryOp::Mul { unchecked: true }, not_b); + r = self.insert_binary(r1, BinaryOp::Add { unchecked: true }, r2); } r } else { diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_enable_side_effects.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_enable_side_effects.rs index e85e2c4a441..942fe67b5d5 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_enable_side_effects.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_enable_side_effects.rs @@ -126,7 +126,7 @@ impl Context { use Instruction::*; match instruction { Binary(binary) => match binary.operator { - BinaryOp::Add | BinaryOp::Sub | BinaryOp::Mul => { + BinaryOp::Add { .. } | BinaryOp::Sub { .. } | BinaryOp::Mul { .. } => { dfg.type_of_value(binary.lhs).is_unsigned() } BinaryOp::Div | BinaryOp::Mod => { @@ -143,10 +143,12 @@ impl Context { | Not(_) | Truncate { .. } | Constrain(..) + | ConstrainNotEqual(..) | RangeCheck { .. } | IfElse { .. } | IncrementRc { .. } | DecrementRc { .. } + | Noop | MakeArray { .. } => false, EnableSideEffectsIf { .. } @@ -176,8 +178,6 @@ impl Context { | Intrinsic::ToRadix(_) | Intrinsic::BlackBox(_) | Intrinsic::Hint(Hint::BlackBox) - | Intrinsic::FromField - | Intrinsic::AsField | Intrinsic::AsSlice | Intrinsic::AsWitness | Intrinsic::IsUnconstrained @@ -240,13 +240,13 @@ mod test { let one = builder.numeric_constant(1u128, NumericType::bool()); builder.insert_enable_side_effects_if(one); - builder.insert_binary(v0, BinaryOp::Mul, two); + builder.insert_binary(v0, BinaryOp::Mul { unchecked: false }, two); builder.insert_enable_side_effects_if(one); - builder.insert_binary(v0, BinaryOp::Mul, two); + builder.insert_binary(v0, BinaryOp::Mul { unchecked: false }, two); builder.insert_enable_side_effects_if(one); - builder.insert_binary(v0, BinaryOp::Mul, two); + builder.insert_binary(v0, BinaryOp::Mul { unchecked: false }, two); builder.insert_enable_side_effects_if(one); - builder.insert_binary(v0, BinaryOp::Mul, two); + builder.insert_binary(v0, BinaryOp::Mul { unchecked: false }, two); builder.insert_enable_side_effects_if(one); let ssa = builder.finish(); @@ -276,7 +276,10 @@ mod test { assert_eq!(instructions.len(), 4); for instruction in instructions.iter().take(4) { - assert_eq!(&main.dfg[*instruction], &Instruction::binary(BinaryOp::Mul, v0, two)); + assert_eq!( + &main.dfg[*instruction], + &Instruction::binary(BinaryOp::Mul { unchecked: false }, v0, two) + ); } } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_if_else.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_if_else.rs index 79f2354aff6..8dde79a3c60 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_if_else.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_if_else.rs @@ -52,10 +52,6 @@ impl Function { struct Context { slice_sizes: HashMap, - // Maps array_set result -> element that was overwritten by that instruction. - // Used to undo array_sets while merging values - prev_array_set_elem_values: HashMap, - // Maps array_set result -> enable_side_effects_if value which was active during it. array_set_conditionals: HashMap, } @@ -235,8 +231,6 @@ fn slice_capacity_change( | Intrinsic::StrAsBytes | Intrinsic::BlackBox(_) | Intrinsic::Hint(Hint::BlackBox) - | Intrinsic::FromField - | Intrinsic::AsField | Intrinsic::AsWitness | Intrinsic::IsUnconstrained | Intrinsic::DerivePedersenGenerators diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_unreachable.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_unreachable.rs new file mode 100644 index 00000000000..9b80b3a4d23 --- /dev/null +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/remove_unreachable.rs @@ -0,0 +1,135 @@ +use std::collections::BTreeSet; + +use fxhash::FxHashSet as HashSet; + +use crate::ssa::{ + ir::{ + function::{Function, FunctionId}, + instruction::Instruction, + value::Value, + }, + ssa_gen::Ssa, +}; + +impl Ssa { + /// Removes any unreachable functions from the code. These can result from + /// optimizations making existing functions unreachable, e.g. `if false { foo() }`, + /// or even from monomorphizing an unconstrained version of a constrained function + /// where the original constrained version ends up never being used. + pub(crate) fn remove_unreachable_functions(mut self) -> Self { + let mut used_functions = HashSet::default(); + + for (id, function) in self.functions.iter() { + // XXX: `self.is_entry_point(*id)` could leave Brillig functions that nobody calls in the SSA. + let is_entry_point = function.id() == self.main_id + || function.runtime().is_acir() && function.runtime().is_entry_point(); + + if is_entry_point { + collect_reachable_functions(&self, *id, &mut used_functions); + } + } + + self.functions.retain(|id, _| used_functions.contains(id)); + self + } +} + +fn collect_reachable_functions( + ssa: &Ssa, + current_func_id: FunctionId, + reachable_functions: &mut HashSet, +) { + if reachable_functions.contains(¤t_func_id) { + return; + } + reachable_functions.insert(current_func_id); + + // If the debugger is used, its possible for function inlining + // to remove functions that the debugger still references + let Some(func) = ssa.functions.get(¤t_func_id) else { + return; + }; + + let used_functions = used_functions(func); + + for called_func_id in used_functions.iter() { + collect_reachable_functions(ssa, *called_func_id, reachable_functions); + } +} + +fn used_functions(func: &Function) -> BTreeSet { + let mut used_function_ids = BTreeSet::default(); + + let mut find_functions = |value| { + if let Value::Function(function) = func.dfg[func.dfg.resolve(value)] { + used_function_ids.insert(function); + } + }; + + for block_id in func.reachable_blocks() { + let block = &func.dfg[block_id]; + + for instruction_id in block.instructions() { + let instruction = &func.dfg[*instruction_id]; + + if matches!(instruction, Instruction::Store { .. } | Instruction::Call { .. }) { + instruction.for_each_value(&mut find_functions); + } + } + + block.unwrap_terminator().for_each_value(&mut find_functions); + } + + used_function_ids +} + +#[cfg(test)] +mod tests { + use crate::ssa::opt::assert_normalized_ssa_equals; + + use super::Ssa; + + #[test] + fn remove_unused_brillig() { + let src = " + brillig(inline) fn main f0 { + b0(v0: u32): + v2 = call f1(v0) -> u32 + v4 = add v0, u32 1 + v5 = eq v2, v4 + constrain v2 == v4 + return + } + brillig(inline) fn increment f1 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + brillig(inline) fn increment_acir f2 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + "; + + let ssa = Ssa::from_str(src).unwrap(); + let ssa = ssa.remove_unreachable_functions(); + + let expected = " + brillig(inline) fn main f0 { + b0(v0: u32): + v2 = call f1(v0) -> u32 + v4 = add v0, u32 1 + v5 = eq v2, v4 + constrain v2 == v4 + return + } + brillig(inline) fn increment f1 { + b0(v0: u32): + v2 = add v0, u32 1 + return v2 + } + "; + assert_normalized_ssa_equals(ssa, expected); + } +} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/resolve_is_unconstrained.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/resolve_is_unconstrained.rs deleted file mode 100644 index 87e680932c6..00000000000 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/resolve_is_unconstrained.rs +++ /dev/null @@ -1,56 +0,0 @@ -use crate::ssa::{ - ir::{ - function::{Function, RuntimeType}, - instruction::{Instruction, Intrinsic}, - types::NumericType, - value::Value, - }, - ssa_gen::Ssa, -}; -use fxhash::FxHashSet as HashSet; - -impl Ssa { - /// An SSA pass to find any calls to `Intrinsic::IsUnconstrained` and replacing any uses of the result of the intrinsic - /// with the resolved boolean value. - /// Note that this pass must run after the pass that does runtime separation, since in SSA generation an ACIR function can end up targeting brillig. - #[tracing::instrument(level = "trace", skip(self))] - pub(crate) fn resolve_is_unconstrained(mut self) -> Self { - for func in self.functions.values_mut() { - func.replace_is_unconstrained_result(); - } - self - } -} - -impl Function { - pub(crate) fn replace_is_unconstrained_result(&mut self) { - let mut is_unconstrained_calls = HashSet::default(); - // Collect all calls to is_unconstrained - for block_id in self.reachable_blocks() { - for &instruction_id in self.dfg[block_id].instructions() { - let target_func = match &self.dfg[instruction_id] { - Instruction::Call { func, .. } => *func, - _ => continue, - }; - - if let Value::Intrinsic(Intrinsic::IsUnconstrained) = &self.dfg[target_func] { - is_unconstrained_calls.insert(instruction_id); - } - } - } - - for instruction_id in is_unconstrained_calls { - let call_returns = self.dfg.instruction_results(instruction_id); - let original_return_id = call_returns[0]; - - // We replace the result with a fresh id. This will be unused, so the DIE pass will remove the leftover intrinsic call. - self.dfg.replace_result(instruction_id, original_return_id); - - let is_unconstrained = matches!(self.runtime(), RuntimeType::Brillig(_)).into(); - let is_within_unconstrained = - self.dfg.make_constant(is_unconstrained, NumericType::bool()); - // Replace all uses of the original return value with the constant - self.dfg.set_value_from_id(original_return_id, is_within_unconstrained); - } - } -} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/runtime_separation.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/runtime_separation.rs deleted file mode 100644 index 5628e12b9ae..00000000000 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/runtime_separation.rs +++ /dev/null @@ -1,351 +0,0 @@ -use std::collections::BTreeSet; - -use fxhash::{FxHashMap as HashMap, FxHashSet as HashSet}; - -use crate::ssa::{ - ir::{ - function::{Function, FunctionId, RuntimeType}, - instruction::Instruction, - value::{Value, ValueId}, - }, - ssa_gen::Ssa, -}; - -impl Ssa { - /// This optimization step separates the runtime of the functions in the SSA. - /// After this step, all functions with runtime `Acir` will be converted to Acir and - /// the functions with runtime `Brillig` will be converted to Brillig. - /// It does so by cloning all ACIR functions called from a Brillig context - /// and changing the runtime of the cloned functions to Brillig. - /// This pass needs to run after functions as values have been resolved (defunctionalization). - #[tracing::instrument(level = "trace", skip(self))] - pub(crate) fn separate_runtime(mut self) -> Self { - RuntimeSeparatorContext::separate_runtime(&mut self); - self - } -} - -#[derive(Debug, Default)] -struct RuntimeSeparatorContext { - // Original functions to clone to brillig - acir_functions_called_from_brillig: BTreeSet, - // Tracks the original => cloned version - mapped_functions: HashMap, -} - -impl RuntimeSeparatorContext { - pub(crate) fn separate_runtime(ssa: &mut Ssa) { - let mut runtime_separator = RuntimeSeparatorContext::default(); - - // We first collect all the acir functions called from a brillig context by exploring the SSA recursively - let mut processed_functions = HashSet::default(); - runtime_separator.collect_acir_functions_called_from_brillig( - ssa, - ssa.main_id, - false, - &mut processed_functions, - ); - - // Now we clone the relevant acir functions and change their runtime to brillig - runtime_separator.convert_acir_functions_called_from_brillig_to_brillig(ssa); - - // Now we update any calls within a brillig context to the mapped functions - runtime_separator.replace_calls_to_mapped_functions(ssa); - - // Some functions might be unreachable now (for example an acir function only called from brillig) - prune_unreachable_functions(ssa); - } - - fn collect_acir_functions_called_from_brillig( - &mut self, - ssa: &Ssa, - current_func_id: FunctionId, - mut within_brillig: bool, - processed_functions: &mut HashSet<(/* within_brillig */ bool, FunctionId)>, - ) { - // Processed functions needs the within brillig flag, since it is possible to call the same function from both brillig and acir - if processed_functions.contains(&(within_brillig, current_func_id)) { - return; - } - processed_functions.insert((within_brillig, current_func_id)); - - let func = &ssa.functions[¤t_func_id]; - if matches!(func.runtime(), RuntimeType::Brillig(_)) { - within_brillig = true; - } - - let called_functions = called_functions(func); - - if within_brillig { - for called_func_id in called_functions.iter() { - let called_func = &ssa.functions[&called_func_id]; - if matches!(called_func.runtime(), RuntimeType::Acir(_)) { - self.acir_functions_called_from_brillig.insert(*called_func_id); - } - } - } - - for called_func_id in called_functions.into_iter() { - self.collect_acir_functions_called_from_brillig( - ssa, - called_func_id, - within_brillig, - processed_functions, - ); - } - } - - fn convert_acir_functions_called_from_brillig_to_brillig(&mut self, ssa: &mut Ssa) { - for acir_func_id in self.acir_functions_called_from_brillig.iter() { - let RuntimeType::Acir(inline_type) = ssa.functions[acir_func_id].runtime() else { - unreachable!("Function to transform to brillig should be ACIR") - }; - let cloned_id = ssa.clone_fn(*acir_func_id); - let new_func = - ssa.functions.get_mut(&cloned_id).expect("Cloned function should exist in SSA"); - new_func.set_runtime(RuntimeType::Brillig(inline_type)); - self.mapped_functions.insert(*acir_func_id, cloned_id); - } - } - - fn replace_calls_to_mapped_functions(&self, ssa: &mut Ssa) { - for (_function_id, func) in ssa.functions.iter_mut() { - if matches!(func.runtime(), RuntimeType::Brillig(_)) { - for called_func_value_id in called_functions_values(func).iter() { - let Value::Function(called_func_id) = &func.dfg[*called_func_value_id] else { - unreachable!("Value should be a function") - }; - if let Some(mapped_func_id) = self.mapped_functions.get(called_func_id) { - let mapped_value_id = func.dfg.import_function(*mapped_func_id); - func.dfg.set_value_from_id(*called_func_value_id, mapped_value_id); - } - } - } - } - } -} - -// We only consider direct calls to functions since functions as values should have been resolved -fn called_functions_values(func: &Function) -> BTreeSet { - let mut called_function_ids = BTreeSet::default(); - for block_id in func.reachable_blocks() { - for instruction_id in func.dfg[block_id].instructions() { - let Instruction::Call { func: called_value_id, .. } = &func.dfg[*instruction_id] else { - continue; - }; - - if let Value::Function(_) = func.dfg[*called_value_id] { - called_function_ids.insert(*called_value_id); - } - } - } - - called_function_ids -} - -fn called_functions(func: &Function) -> BTreeSet { - called_functions_values(func) - .into_iter() - .map(|value_id| { - let Value::Function(func_id) = func.dfg[value_id] else { - unreachable!("Value should be a function") - }; - func_id - }) - .collect() -} - -fn collect_reachable_functions( - ssa: &Ssa, - current_func_id: FunctionId, - reachable_functions: &mut HashSet, -) { - if reachable_functions.contains(¤t_func_id) { - return; - } - reachable_functions.insert(current_func_id); - - let func = &ssa.functions[¤t_func_id]; - let called_functions = called_functions(func); - - for called_func_id in called_functions.iter() { - collect_reachable_functions(ssa, *called_func_id, reachable_functions); - } -} - -fn prune_unreachable_functions(ssa: &mut Ssa) { - let mut reachable_functions = HashSet::default(); - collect_reachable_functions(ssa, ssa.main_id, &mut reachable_functions); - - ssa.functions.retain(|id, _value| reachable_functions.contains(id)); -} - -#[cfg(test)] -mod test { - use std::collections::BTreeSet; - - use noirc_frontend::monomorphization::ast::InlineType; - - use crate::ssa::{ - function_builder::FunctionBuilder, - ir::{ - function::{Function, FunctionId, RuntimeType}, - map::Id, - types::Type, - }, - opt::runtime_separation::called_functions, - ssa_gen::Ssa, - }; - - #[test] - fn basic_runtime_separation() { - // brillig fn foo { - // b0(): - // v0 = call bar() - // return v0 - // } - // acir fn bar { - // b0(): - // return 72 - // } - let foo_id = Id::test_new(0); - let mut builder = FunctionBuilder::new("foo".into(), foo_id); - builder.current_function.set_runtime(RuntimeType::Brillig(InlineType::default())); - - let bar_id = Id::test_new(1); - let bar = builder.import_function(bar_id); - let results = builder.insert_call(bar, Vec::new(), vec![Type::field()]).to_vec(); - builder.terminate_with_return(results); - - builder.new_function("bar".into(), bar_id, InlineType::default()); - let expected_return = 72u128; - let seventy_two = builder.field_constant(expected_return); - builder.terminate_with_return(vec![seventy_two]); - - let ssa = builder.finish(); - assert_eq!(ssa.functions.len(), 2); - - // Expected result - // brillig fn foo { - // b0(): - // v0 = call bar() - // return v0 - // } - // brillig fn bar { - // b0(): - // return 72 - // } - let separated = ssa.separate_runtime(); - - // The original bar function must have been pruned - assert_eq!(separated.functions.len(), 2); - - // All functions should be brillig now - for func in separated.functions.values() { - assert_eq!(func.runtime(), RuntimeType::Brillig(InlineType::default())); - } - } - - fn find_func_by_name<'ssa>( - ssa: &'ssa Ssa, - funcs: &BTreeSet, - name: &str, - ) -> &'ssa Function { - funcs - .iter() - .find_map(|id| { - let func = ssa.functions.get(id).unwrap(); - if func.name() == name { - Some(func) - } else { - None - } - }) - .unwrap() - } - - #[test] - fn same_function_shared_acir_brillig() { - // acir fn foo { - // b0(): - // v0 = call bar() - // v1 = call baz() - // return v0, v1 - // } - // brillig fn bar { - // b0(): - // v0 = call baz() - // return v0 - // } - // acir fn baz { - // b0(): - // return 72 - // } - let foo_id = Id::test_new(0); - let mut builder = FunctionBuilder::new("foo".into(), foo_id); - - let bar_id = Id::test_new(1); - let baz_id = Id::test_new(2); - let bar = builder.import_function(bar_id); - let baz = builder.import_function(baz_id); - let v0 = builder.insert_call(bar, Vec::new(), vec![Type::field()]).to_vec(); - let v1 = builder.insert_call(baz, Vec::new(), vec![Type::field()]).to_vec(); - builder.terminate_with_return(vec![v0[0], v1[0]]); - - builder.new_brillig_function("bar".into(), bar_id, InlineType::default()); - let baz = builder.import_function(baz_id); - let v0 = builder.insert_call(baz, Vec::new(), vec![Type::field()]).to_vec(); - builder.terminate_with_return(v0); - - builder.new_function("baz".into(), baz_id, InlineType::default()); - let expected_return = 72u128; - let seventy_two = builder.field_constant(expected_return); - builder.terminate_with_return(vec![seventy_two]); - - let ssa = builder.finish(); - assert_eq!(ssa.functions.len(), 3); - - // Expected result - // acir fn foo { - // b0(): - // v0 = call bar() - // v1 = call baz() <- baz_acir - // return v0, v1 - // } - // brillig fn bar { - // b0(): - // v0 = call baz() <- baz_brillig - // return v0 - // } - // acir fn baz { - // b0(): - // return 72 - // } - // brillig fn baz { - // b0(): - // return 72 - // } - let separated = ssa.separate_runtime(); - - // The original baz function must have been duplicated - assert_eq!(separated.functions.len(), 4); - - let main_function = separated.functions.get(&separated.main_id).unwrap(); - assert_eq!(main_function.runtime(), RuntimeType::Acir(InlineType::Inline)); - - let main_calls = called_functions(main_function); - assert_eq!(main_calls.len(), 2); - - let bar = find_func_by_name(&separated, &main_calls, "bar"); - let baz_acir = find_func_by_name(&separated, &main_calls, "baz"); - - assert_eq!(baz_acir.runtime(), RuntimeType::Acir(InlineType::Inline)); - assert_eq!(bar.runtime(), RuntimeType::Brillig(InlineType::default())); - - let bar_calls = called_functions(bar); - assert_eq!(bar_calls.len(), 1); - - let baz_brillig = find_func_by_name(&separated, &bar_calls, "baz"); - assert_eq!(baz_brillig.runtime(), RuntimeType::Brillig(InlineType::default())); - } -} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/unrolling.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/unrolling.rs index 2a272236195..cae7735b2c3 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/unrolling.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/opt/unrolling.rs @@ -18,11 +18,16 @@ //! //! When unrolling ACIR code, we remove reference count instructions because they are //! only used by Brillig bytecode. +use std::collections::BTreeSet; + use acvm::{acir::AcirField, FieldElement}; use im::HashSet; use crate::{ - brillig::brillig_gen::convert_ssa_function, + brillig::{ + brillig_gen::{brillig_globals::convert_ssa_globals, convert_ssa_function}, + brillig_ir::brillig_variable::BrilligVariable, + }, errors::RuntimeError, ssa::{ ir::{ @@ -85,8 +90,13 @@ impl Ssa { if has_unrolled { if let Some((orig_function, max_incr_pct)) = orig_func_and_max_incr_pct { - let new_size = brillig_bytecode_size(function); - let orig_size = brillig_bytecode_size(&orig_function); + // DIE is run at the end of our SSA optimizations, so we mark all globals as in use here. + let used_globals = &self.globals.dfg.values_iter().map(|(id, _)| id).collect(); + let (_, brillig_globals) = + convert_ssa_globals(false, &self.globals, used_globals); + + let new_size = brillig_bytecode_size(function, &brillig_globals); + let orig_size = brillig_bytecode_size(&orig_function, &brillig_globals); if !is_new_size_ok(orig_size, new_size, max_incr_pct) { *function = orig_function; } @@ -117,7 +127,7 @@ pub(super) struct Loop { back_edge_start: BasicBlockId, /// All the blocks contained within the loop, including `header` and `back_edge_start`. - pub(super) blocks: HashSet, + pub(super) blocks: BTreeSet, } pub(super) struct Loops { @@ -238,7 +248,7 @@ impl Loop { back_edge_start: BasicBlockId, cfg: &ControlFlowGraph, ) -> Self { - let mut blocks = HashSet::default(); + let mut blocks = BTreeSet::default(); blocks.insert(header); let mut insert = |block, stack: &mut Vec| { @@ -308,11 +318,13 @@ impl Loop { // simplified to a simple jump. return None; } - assert_eq!( - instructions.len(), - 1, - "The header should just compare the induction variable and jump" - ); + + if instructions.len() != 1 { + // The header should just compare the induction variable and jump. + // If that's not the case, this might be a `loop` and not a `for` loop. + return None; + } + match &function.dfg[instructions[0]] { Instruction::Binary(Binary { lhs: _, operator: BinaryOp::Lt, rhs }) => { function.dfg.get_numeric_constant(*rhs) @@ -617,10 +629,16 @@ impl Loop { let header = &function.dfg[self.header]; let induction_var = header.parameters()[0]; - back.instructions().iter().filter(|instruction| { - let instruction = &function.dfg[**instruction]; - matches!(instruction, Instruction::Binary(Binary { lhs, operator: BinaryOp::Add, rhs: _ }) if *lhs == induction_var) - }).count() + back.instructions() + .iter() + .filter(|instruction| { + let instruction = &function.dfg[**instruction]; + matches!(instruction, + Instruction::Binary(Binary { lhs, operator: BinaryOp::Add { .. }, rhs: _ }) + if *lhs == induction_var + ) + }) + .count() } /// Decide if this loop is small enough that it can be inlined in a way that the number @@ -742,7 +760,13 @@ fn get_induction_variable(function: &Function, block: BasicBlockId) -> Result usize { +fn brillig_bytecode_size( + function: &Function, + globals: &HashMap, +) -> usize { // We need to do some SSA passes in order for the conversion to be able to go ahead, // otherwise we can hit `unreachable!()` instructions in `convert_ssa_instruction`. // Creating a clone so as not to modify the originals. @@ -982,9 +1009,9 @@ fn brillig_bytecode_size(function: &Function) -> usize { simplify_between_unrolls(&mut temp); // This is to try to prevent hitting ICE. - temp.dead_instruction_elimination(false); + temp.dead_instruction_elimination(false, true); - convert_ssa_function(&temp, false).byte_code.len() + convert_ssa_function(&temp, false, globals).byte_code.len() } /// Decide if the new bytecode size is acceptable, compared to the original. diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/into_ssa.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/into_ssa.rs index 7c7e977c6ce..e2eea234dc7 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/into_ssa.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/into_ssa.rs @@ -5,7 +5,9 @@ use acvm::acir::circuit::ErrorSelector; use crate::ssa::{ function_builder::FunctionBuilder, ir::{ - basic_block::BasicBlockId, function::FunctionId, instruction::ConstrainError, + basic_block::BasicBlockId, + function::{Function, FunctionId}, + instruction::ConstrainError, value::ValueId, }, }; @@ -16,15 +18,15 @@ use super::{ }; impl ParsedSsa { - pub(crate) fn into_ssa(self) -> Result { - Translator::translate(self) + pub(crate) fn into_ssa(self, simplify: bool) -> Result { + Translator::translate(self, simplify) } } struct Translator { builder: FunctionBuilder, - /// Maps function names to their IDs + /// Maps internal function names (e.g. "f1") to their IDs functions: HashMap, /// Maps block names to their IDs @@ -41,8 +43,8 @@ struct Translator { } impl Translator { - fn translate(mut parsed_ssa: ParsedSsa) -> Result { - let mut translator = Self::new(&mut parsed_ssa)?; + fn translate(mut parsed_ssa: ParsedSsa, simplify: bool) -> Result { + let mut translator = Self::new(&mut parsed_ssa, simplify)?; // Note that the `new` call above removed the main function, // so all we are left with are non-main functions. @@ -53,13 +55,14 @@ impl Translator { Ok(translator.finish()) } - fn new(parsed_ssa: &mut ParsedSsa) -> Result { + fn new(parsed_ssa: &mut ParsedSsa, simplify: bool) -> Result { // A FunctionBuilder must be created with a main Function, so here wer remove it // from the parsed SSA to avoid adding it twice later on. let main_function = parsed_ssa.functions.remove(0); let main_id = FunctionId::test_new(0); let mut builder = FunctionBuilder::new(main_function.external_name.clone(), main_id); builder.set_runtime(main_function.runtime_type); + builder.simplify = simplify; // Map function names to their IDs so calls can be resolved let mut function_id_counter = 1; @@ -134,14 +137,14 @@ impl Translator { match block.terminator { ParsedTerminator::Jmp { destination, arguments } => { - let block_id = self.lookup_block(destination)?; + let block_id = self.lookup_block(&destination)?; let arguments = self.translate_values(arguments)?; self.builder.terminate_with_jmp(block_id, arguments); } ParsedTerminator::Jmpif { condition, then_block, else_block } => { let condition = self.translate_value(condition)?; - let then_destination = self.lookup_block(then_block)?; - let else_destination = self.lookup_block(else_block)?; + let then_destination = self.lookup_block(&then_block)?; + let else_destination = self.lookup_block(&else_block)?; self.builder.terminate_with_jmpif(condition, then_destination, else_destination); } ParsedTerminator::Return(values) => { @@ -186,8 +189,13 @@ impl Translator { let function_id = if let Some(id) = self.builder.import_intrinsic(&function.name) { id } else { - let function_id = self.lookup_function(function)?; - self.builder.import_function(function_id) + let maybe_func = + self.lookup_function(&function).map(|f| self.builder.import_function(f)); + + maybe_func.or_else(|e| { + // e.g. `v2 = call v0(v1) -> u32`, a lambda passed as a parameter + self.lookup_variable(&function).map_err(|_| e) + })? }; let arguments = self.translate_values(arguments)?; @@ -292,7 +300,14 @@ impl Translator { ParsedValue::NumericConstant { constant, typ } => { Ok(self.builder.numeric_constant(constant, typ.unwrap_numeric())) } - ParsedValue::Variable(identifier) => self.lookup_variable(identifier), + ParsedValue::Variable(identifier) => self.lookup_variable(&identifier).or_else(|e| { + self.lookup_function(&identifier) + .map(|f| { + // e.g. `v3 = call f1(f2, v0) -> u32` + self.builder.import_function(f) + }) + .map_err(|_| e) + }), } } @@ -313,27 +328,27 @@ impl Translator { Ok(()) } - fn lookup_variable(&mut self, identifier: Identifier) -> Result { + fn lookup_variable(&mut self, identifier: &Identifier) -> Result { if let Some(value_id) = self.variables[&self.current_function_id()].get(&identifier.name) { Ok(*value_id) } else { - Err(SsaError::UnknownVariable(identifier)) + Err(SsaError::UnknownVariable(identifier.clone())) } } - fn lookup_block(&mut self, identifier: Identifier) -> Result { + fn lookup_block(&mut self, identifier: &Identifier) -> Result { if let Some(block_id) = self.blocks[&self.current_function_id()].get(&identifier.name) { Ok(*block_id) } else { - Err(SsaError::UnknownBlock(identifier)) + Err(SsaError::UnknownBlock(identifier.clone())) } } - fn lookup_function(&mut self, identifier: Identifier) -> Result { + fn lookup_function(&mut self, identifier: &Identifier) -> Result { if let Some(function_id) = self.functions.get(&identifier.name) { Ok(*function_id) } else { - Err(SsaError::UnknownFunction(identifier)) + Err(SsaError::UnknownFunction(identifier.clone())) } } @@ -344,6 +359,8 @@ impl Translator { // that the SSA we parsed was printed by the `SsaBuilder`, which normalizes // before each print. ssa.normalize_ids(); + // Does not matter what ID we use here. + ssa.globals = Function::new("globals".to_owned(), ssa.main_id); ssa } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/lexer.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/lexer.rs index 5b66810c641..e22b6a661de 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/lexer.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/lexer.rs @@ -275,10 +275,6 @@ impl<'a> Lexer<'a> { self.chars.clone().next().map(|(_, ch)| ch) } - fn is_code_whitespace(c: char) -> bool { - c.is_ascii_whitespace() - } - pub(crate) fn newline_follows(&self) -> bool { let chars = self.chars.clone(); chars.take_while(|(_, char)| char.is_ascii_whitespace()).any(|(_, char)| char == '\n') diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/mod.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/mod.rs index 24a5ff43071..143ba511879 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/mod.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/mod.rs @@ -32,16 +32,24 @@ mod token; impl Ssa { /// Creates an Ssa object from the given string. - /// - /// Note that the resulting Ssa might not be exactly the same as the given string. - /// This is because, internally, the Ssa is built using a `FunctionBuilder`, so - /// some instructions might be simplified while they are inserted. pub(crate) fn from_str(src: &str) -> Result { + Self::from_str_impl(src, false) + } + + /// Creates an Ssa object from the given string but trying to simplify + /// each parsed instruction as it's inserted into the final SSA. + pub(crate) fn from_str_simplifying(src: &str) -> Result { + Self::from_str_impl(src, true) + } + + fn from_str_impl(src: &str, simplify: bool) -> Result { let mut parser = Parser::new(src).map_err(|err| SsaErrorWithSource::parse_error(err, src))?; let parsed_ssa = parser.parse_ssa().map_err(|err| SsaErrorWithSource::parse_error(err, src))?; - parsed_ssa.into_ssa().map_err(|error| SsaErrorWithSource { src: src.to_string(), error }) + parsed_ssa + .into_ssa(simplify) + .map_err(|error| SsaErrorWithSource { src: src.to_string(), error }) } } @@ -278,9 +286,9 @@ impl<'a> Parser<'a> { fn eat_binary_op(&mut self) -> ParseResult> { let op = match self.token.token() { - Token::Keyword(Keyword::Add) => BinaryOp::Add, - Token::Keyword(Keyword::Sub) => BinaryOp::Sub, - Token::Keyword(Keyword::Mul) => BinaryOp::Mul, + Token::Keyword(Keyword::Add) => BinaryOp::Add { unchecked: false }, + Token::Keyword(Keyword::Sub) => BinaryOp::Sub { unchecked: false }, + Token::Keyword(Keyword::Mul) => BinaryOp::Mul { unchecked: false }, Token::Keyword(Keyword::Div) => BinaryOp::Div, Token::Keyword(Keyword::Eq) => BinaryOp::Eq, Token::Keyword(Keyword::Mod) => BinaryOp::Mod, @@ -290,6 +298,9 @@ impl<'a> Parser<'a> { Token::Keyword(Keyword::Xor) => BinaryOp::Xor, Token::Keyword(Keyword::Shl) => BinaryOp::Shl, Token::Keyword(Keyword::Shr) => BinaryOp::Shr, + Token::Keyword(Keyword::UncheckedAdd) => BinaryOp::Add { unchecked: true }, + Token::Keyword(Keyword::UncheckedSub) => BinaryOp::Sub { unchecked: true }, + Token::Keyword(Keyword::UncheckedMul) => BinaryOp::Mul { unchecked: true }, _ => return Ok(None), }; @@ -859,10 +870,6 @@ impl<'a> Parser<'a> { self.token.token() == &token } - fn at_keyword(&self, keyword: Keyword) -> bool { - self.at(Token::Keyword(keyword)) - } - fn newline_follows(&self) -> bool { self.lexer.newline_follows() } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/tests.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/tests.rs index dab96dfa04f..8c24b2ec458 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/tests.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/tests.rs @@ -326,11 +326,27 @@ fn test_array_get_set_bug() { #[test] fn test_binary() { - for op in ["add", "sub", "mul", "div", "eq", "mod", "lt", "and", "or", "xor", "shl", "shr"] { + for op in [ + "add", + "sub", + "mul", + "div", + "eq", + "mod", + "lt", + "and", + "or", + "xor", + "shl", + "shr", + "unchecked_add", + "unchecked_sub", + "unchecked_mul", + ] { let src = format!( " acir(inline) fn main f0 {{ - b0(v0: Field, v1: Field): + b0(v0: u32, v1: u32): v2 = {op} v0, v1 return }} @@ -415,7 +431,7 @@ fn test_store() { #[test] fn test_inc_rc() { let src = " - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: [Field; 3]): inc_rc v0 return @@ -427,7 +443,7 @@ fn test_inc_rc() { #[test] fn test_dec_rc() { let src = " - acir(inline) fn main f0 { + brillig(inline) fn main f0 { b0(v0: [Field; 3]): dec_rc v0 return @@ -502,3 +518,15 @@ fn test_function_type() { "; assert_ssa_roundtrip(src); } + +#[test] +fn test_does_not_simplify() { + let src = " + acir(inline) fn main f0 { + b0(): + v2 = add Field 1, Field 2 + return v2 + } + "; + assert_ssa_roundtrip(src); +} diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/token.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/token.rs index 83a2a1d1ed2..eb09209466d 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/token.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/parser/token.rs @@ -160,6 +160,9 @@ pub(crate) enum Keyword { Then, To, Truncate, + UncheckedAdd, + UncheckedSub, + UncheckedMul, Value, Xor, } @@ -217,6 +220,9 @@ impl Keyword { "then" => Keyword::Then, "to" => Keyword::To, "truncate" => Keyword::Truncate, + "unchecked_add" => Keyword::UncheckedAdd, + "unchecked_sub" => Keyword::UncheckedSub, + "unchecked_mul" => Keyword::UncheckedMul, "value" => Keyword::Value, "xor" => Keyword::Xor, _ => return None, @@ -278,6 +284,9 @@ impl Display for Keyword { Keyword::Then => write!(f, "then"), Keyword::To => write!(f, "to"), Keyword::Truncate => write!(f, "truncate"), + Keyword::UncheckedAdd => write!(f, "unchecked_add"), + Keyword::UncheckedSub => write!(f, "unchecked_sub"), + Keyword::UncheckedMul => write!(f, "unchecked_mul"), Keyword::Value => write!(f, "value"), Keyword::Xor => write!(f, "xor"), } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/context.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/context.rs index 7807658dabb..a845c5654b2 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/context.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/context.rs @@ -1,10 +1,11 @@ +use std::collections::BTreeMap; use std::sync::{Arc, Mutex, RwLock}; use acvm::{acir::AcirField, FieldElement}; use iter_extended::vecmap; use noirc_errors::Location; use noirc_frontend::ast::{BinaryOpKind, Signedness}; -use noirc_frontend::monomorphization::ast::{self, InlineType, LocalId, Parameters}; +use noirc_frontend::monomorphization::ast::{self, GlobalId, InlineType, LocalId, Parameters}; use noirc_frontend::monomorphization::ast::{FuncId, Program}; use crate::errors::RuntimeError; @@ -19,6 +20,7 @@ use crate::ssa::ir::types::{NumericType, Type}; use crate::ssa::ir::value::ValueId; use super::value::{Tree, Value, Values}; +use super::GlobalsGraph; use fxhash::{FxHashMap as HashMap, FxHashSet as HashSet}; /// The FunctionContext is the main context object for translating a @@ -71,6 +73,14 @@ pub(super) struct SharedContext { /// Shared counter used to assign the ID of the next function function_counter: AtomicCounter, + /// A pseudo function that represents global values. + /// Globals are only concerned with the values and instructions (due to Instruction::MakeArray) + /// in a function's DataFlowGraph. However, in order to re-use various codegen methods + /// we need to use the same `Function` type. + pub(super) globals_context: Function, + + pub(super) globals: BTreeMap, + /// The entire monomorphized source program pub(super) program: Program, } @@ -78,7 +88,8 @@ pub(super) struct SharedContext { #[derive(Copy, Clone)] pub(super) struct Loop { pub(super) loop_entry: BasicBlockId, - pub(super) loop_index: ValueId, + /// The loop index will be `Some` for a `for` and `None` for a `loop` + pub(super) loop_index: Option, pub(super) loop_end: BasicBlockId, } @@ -100,6 +111,7 @@ impl<'a> FunctionContext<'a> { parameters: &Parameters, runtime: RuntimeType, shared_context: &'a SharedContext, + globals: GlobalsGraph, ) -> Self { let function_id = shared_context .pop_next_function_in_queue() @@ -108,6 +120,8 @@ impl<'a> FunctionContext<'a> { let mut builder = FunctionBuilder::new(function_name, function_id); builder.set_runtime(runtime); + builder.set_globals(Arc::new(globals)); + let definitions = HashMap::default(); let mut this = Self { definitions, builder, shared_context, loops: Vec::new() }; this.add_parameters_to_scope(parameters); @@ -119,18 +133,17 @@ impl<'a> FunctionContext<'a> { /// /// Note that the previous function cannot be resumed after calling this. Developers should /// avoid calling new_function until the previous function is completely finished with ssa-gen. - pub(super) fn new_function( - &mut self, - id: IrFunctionId, - func: &ast::Function, - force_brillig_runtime: bool, - ) { + pub(super) fn new_function(&mut self, id: IrFunctionId, func: &ast::Function) { self.definitions.clear(); - if func.unconstrained || (force_brillig_runtime && func.inline_type != InlineType::Inline) { + + let globals = self.builder.current_function.dfg.globals.clone(); + if func.unconstrained { self.builder.new_brillig_function(func.name.clone(), id, func.inline_type); } else { self.builder.new_function(func.name.clone(), id, func.inline_type); } + self.builder.set_globals(globals); + self.add_parameters_to_scope(&func.parameters); } @@ -322,12 +335,24 @@ impl<'a> FunctionContext<'a> { // We use unsafe casts here, this is fine as we're casting to a `field` type. let as_field = self.builder.insert_cast(input, NumericType::NativeField); let sign_field = self.builder.insert_cast(sign, NumericType::NativeField); - let positive_predicate = self.builder.insert_binary(sign_field, BinaryOp::Mul, as_field); - let two_complement = self.builder.insert_binary(bit_width, BinaryOp::Sub, as_field); + + // All of these operations are unchecked because they deal with fields + let positive_predicate = + self.builder.insert_binary(sign_field, BinaryOp::Mul { unchecked: true }, as_field); + let two_complement = + self.builder.insert_binary(bit_width, BinaryOp::Sub { unchecked: true }, as_field); let sign_not_field = self.builder.insert_cast(sign_not, NumericType::NativeField); - let negative_predicate = - self.builder.insert_binary(sign_not_field, BinaryOp::Mul, two_complement); - self.builder.insert_binary(positive_predicate, BinaryOp::Add, negative_predicate) + let negative_predicate = self.builder.insert_binary( + sign_not_field, + BinaryOp::Mul { unchecked: true }, + two_complement, + ); + // Unchecked addition because either `positive_predicate` or `negative_predicate` will be 0 + self.builder.insert_binary( + positive_predicate, + BinaryOp::Add { unchecked: true }, + negative_predicate, + ) } /// Insert constraints ensuring that the operation does not overflow the bit size of the result @@ -482,8 +507,12 @@ impl<'a> FunctionContext<'a> { //Check the result has the same sign as its inputs let result_sign = self.builder.insert_binary(result, BinaryOp::Lt, half_width); let sign_diff = self.builder.insert_binary(result_sign, BinaryOp::Eq, lhs_sign); - let sign_diff_with_predicate = - self.builder.insert_binary(sign_diff, BinaryOp::Mul, same_sign); + // Unchecked multiplication because boolean inputs + let sign_diff_with_predicate = self.builder.insert_binary( + sign_diff, + BinaryOp::Mul { unchecked: true }, + same_sign, + ); let overflow_check = Instruction::Constrain( sign_diff_with_predicate, same_sign, @@ -496,7 +525,9 @@ impl<'a> FunctionContext<'a> { // First we compute the absolute value of operands, and their product let lhs_abs = self.absolute_value_helper(lhs, lhs_sign, bit_size); let rhs_abs = self.absolute_value_helper(rhs, rhs_sign, bit_size); - let product_field = self.builder.insert_binary(lhs_abs, BinaryOp::Mul, rhs_abs); + // Unchecked mul because these are fields + let product_field = + self.builder.insert_binary(lhs_abs, BinaryOp::Mul { unchecked: true }, rhs_abs); // It must not already overflow the bit_size self.builder.set_location(location).insert_range_check( product_field, @@ -510,8 +541,12 @@ impl<'a> FunctionContext<'a> { let not_same = self.builder.insert_not(same_sign); let not_same_sign_field = self.insert_safe_cast(not_same, NumericType::unsigned(bit_size), location); - let positive_maximum_with_offset = - self.builder.insert_binary(half_width, BinaryOp::Add, not_same_sign_field); + // Unchecked add because adding 1 to half_width can't overflow + let positive_maximum_with_offset = self.builder.insert_binary( + half_width, + BinaryOp::Add { unchecked: true }, + not_same_sign_field, + ); let product_overflow_check = self.builder.insert_binary(product, BinaryOp::Lt, positive_maximum_with_offset); @@ -615,7 +650,8 @@ impl<'a> FunctionContext<'a> { if offset != 0 { let typ = self.builder.type_of_value(address).unwrap_numeric(); let offset = self.builder.numeric_constant(offset, typ); - address = self.builder.insert_binary(address, BinaryOp::Add, offset); + address = + self.builder.insert_binary(address, BinaryOp::Add { unchecked: true }, offset); } address } @@ -638,6 +674,10 @@ impl<'a> FunctionContext<'a> { self.definitions.get(&id).expect("lookup: variable not defined").clone() } + pub(super) fn lookup_global(&self, id: GlobalId) -> Values { + self.shared_context.globals.get(&id).expect("lookup_global: variable not defined").clone() + } + /// Extract the given field of the tuple. Panics if the given Values is not /// a Tree::Branch or does not have enough fields. pub(super) fn get_field(tuple: Values, field_index: usize) -> Values { @@ -873,14 +913,20 @@ impl<'a> FunctionContext<'a> { self.builder.numeric_constant(self.element_size(array), NumericType::length_type()); // The actual base index is the user's index * the array element type's size - let mut index = - self.builder.set_location(location).insert_binary(index, BinaryOp::Mul, element_size); + // Unchecked mul because we are reaching for an array element: if it overflows here + // it would have overflowed when creating the array. + let mut index = self.builder.set_location(location).insert_binary( + index, + BinaryOp::Mul { unchecked: true }, + element_size, + ); let one = self.builder.numeric_constant(FieldElement::one(), NumericType::length_type()); new_value.for_each(|value| { let value = value.eval(self); array = self.builder.insert_array_set(array, index, value); - index = self.builder.insert_binary(index, BinaryOp::Add, one); + // Unchecked add because this can't overflow (it would have overflowed when creating the array) + index = self.builder.insert_binary(index, BinaryOp::Add { unchecked: true }, one); }); array } @@ -965,13 +1011,8 @@ impl<'a> FunctionContext<'a> { } } - pub(crate) fn enter_loop( - &mut self, - loop_entry: BasicBlockId, - loop_index: ValueId, - loop_end: BasicBlockId, - ) { - self.loops.push(Loop { loop_entry, loop_index, loop_end }); + pub(crate) fn enter_loop(&mut self, loop_: Loop) { + self.loops.push(loop_); } pub(crate) fn exit_loop(&mut self) { @@ -1005,9 +1046,9 @@ fn operator_requires_swapped_operands(op: BinaryOpKind) -> bool { /// to represent the full operation correctly. fn convert_operator(op: BinaryOpKind) -> BinaryOp { match op { - BinaryOpKind::Add => BinaryOp::Add, - BinaryOpKind::Subtract => BinaryOp::Sub, - BinaryOpKind::Multiply => BinaryOp::Mul, + BinaryOpKind::Add => BinaryOp::Add { unchecked: false }, + BinaryOpKind::Subtract => BinaryOp::Sub { unchecked: false }, + BinaryOpKind::Multiply => BinaryOp::Mul { unchecked: false }, BinaryOpKind::Divide => BinaryOp::Div, BinaryOpKind::Modulo => BinaryOp::Mod, BinaryOpKind::Equal => BinaryOp::Eq, @@ -1027,11 +1068,46 @@ fn convert_operator(op: BinaryOpKind) -> BinaryOp { impl SharedContext { /// Create a new SharedContext for the given monomorphized program. pub(super) fn new(program: Program) -> Self { + let globals_shared_context = SharedContext::new_for_globals(); + + let globals_id = Program::global_space_id(); + + // Queue the function representing the globals space for compilation + globals_shared_context.get_or_queue_function(globals_id); + + let mut context = FunctionContext::new( + "globals".to_owned(), + &vec![], + RuntimeType::Brillig(InlineType::default()), + &globals_shared_context, + GlobalsGraph::default(), + ); + let mut globals = BTreeMap::default(); + for (id, global) in program.globals.iter() { + let values = context.codegen_expression(global).unwrap(); + globals.insert(*id, values); + } + Self { functions: Default::default(), function_queue: Default::default(), function_counter: Default::default(), program, + globals_context: context.builder.current_function, + globals, + } + } + + pub(super) fn new_for_globals() -> Self { + let globals_context = Function::new_for_globals(); + + Self { + functions: Default::default(), + function_queue: Default::default(), + function_counter: Default::default(), + program: Default::default(), + globals_context, + globals: Default::default(), } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/mod.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/mod.rs index d3821158b80..fbb2b306bdf 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/mod.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/mod.rs @@ -2,10 +2,11 @@ pub(crate) mod context; mod program; mod value; +use acvm::AcirField; use noirc_frontend::token::FmtStrFragment; pub(crate) use program::Ssa; -use context::SharedContext; +use context::{Loop, SharedContext}; use iter_extended::{try_vecmap, vecmap}; use noirc_errors::Location; use noirc_frontend::ast::{UnaryOp, Visibility}; @@ -22,6 +23,7 @@ use self::{ value::{Tree, Values}, }; +use super::ir::dfg::GlobalsGraph; use super::ir::instruction::ErrorType; use super::ir::types::NumericType; use super::{ @@ -39,10 +41,7 @@ pub(crate) const SSA_WORD_SIZE: u32 = 32; /// Generates SSA for the given monomorphized program. /// /// This function will generate the SSA but does not perform any optimizations on it. -pub(crate) fn generate_ssa( - program: Program, - force_brillig_runtime: bool, -) -> Result { +pub(crate) fn generate_ssa(program: Program) -> Result { // see which parameter has call_data/return_data attribute let is_databus = DataBusBuilder::is_databus(&program.main_function_signature); @@ -51,21 +50,20 @@ pub(crate) fn generate_ssa( let return_location = program.return_location; let context = SharedContext::new(program); + let globals = GlobalsGraph::from_dfg(context.globals_context.dfg.clone()); + let main_id = Program::main_id(); let main = context.program.main(); // Queue the main function for compilation context.get_or_queue_function(main_id); - let mut function_context = FunctionContext::new( - main.name.clone(), - &main.parameters, - if force_brillig_runtime || main.unconstrained { - RuntimeType::Brillig(main.inline_type) - } else { - RuntimeType::Acir(main.inline_type) - }, - &context, - ); + let main_runtime = if main.unconstrained { + RuntimeType::Brillig(main.inline_type) + } else { + RuntimeType::Acir(main.inline_type) + }; + let mut function_context = + FunctionContext::new(main.name.clone(), &main.parameters, main_runtime, &context, globals); // Generate the call_data bus from the relevant parameters. We create it *before* processing the function body let call_data = function_context.builder.call_data_bus(is_databus); @@ -122,11 +120,13 @@ pub(crate) fn generate_ssa( // to generate SSA for each function used within the program. while let Some((src_function_id, dest_id)) = context.pop_next_function_in_queue() { let function = &context.program[src_function_id]; - function_context.new_function(dest_id, function, force_brillig_runtime); + function_context.new_function(dest_id, function); function_context.codegen_function_body(&function.body)?; } - Ok(function_context.builder.finish()) + let mut ssa = function_context.builder.finish(); + ssa.globals = context.globals_context; + Ok(ssa) } impl<'a> FunctionContext<'a> { @@ -152,6 +152,7 @@ impl<'a> FunctionContext<'a> { Expression::Index(index) => self.codegen_index(index), Expression::Cast(cast) => self.codegen_cast(cast), Expression::For(for_expr) => self.codegen_for(for_expr), + Expression::Loop(block) => self.codegen_loop(block), Expression::If(if_expr) => self.codegen_if(if_expr), Expression::Tuple(tuple) => self.codegen_tuple(tuple), Expression::ExtractTupleField(tuple, index) => { @@ -184,6 +185,7 @@ impl<'a> FunctionContext<'a> { fn codegen_ident_reference(&mut self, ident: &ast::Ident) -> Values { match &ident.definition { ast::Definition::Local(id) => self.lookup(*id), + ast::Definition::Global(id) => self.lookup_global(*id), ast::Definition::Function(id) => self.get_or_queue_function(*id), ast::Definition::Oracle(name) => self.builder.import_foreign_function(name).into(), ast::Definition::Builtin(name) | ast::Definition::LowLevel(name) => { @@ -451,8 +453,13 @@ impl<'a> FunctionContext<'a> { let type_size = Self::convert_type(element_type).size_of_type(); let type_size = self.builder.numeric_constant(type_size as u128, NumericType::length_type()); - let base_index = - self.builder.set_location(location).insert_binary(index, BinaryOp::Mul, type_size); + // This shouldn't overflow as we are reaching for an initial array offset + // (otherwise it would have overflowed when creating the array) + let base_index = self.builder.set_location(location).insert_binary( + index, + BinaryOp::Mul { unchecked: true }, + type_size, + ); let mut field_index = 0u128; Ok(Self::map_type(element_type, |typ| { @@ -525,6 +532,22 @@ impl<'a> FunctionContext<'a> { /// ... This is the current insert point after codegen_for finishes ... /// ``` fn codegen_for(&mut self, for_expr: &ast::For) -> Result { + self.builder.set_location(for_expr.start_range_location); + let start_index = self.codegen_non_tuple_expression(&for_expr.start_range)?; + + self.builder.set_location(for_expr.end_range_location); + let end_index = self.codegen_non_tuple_expression(&for_expr.end_range)?; + + if let (Some(start_constant), Some(end_constant)) = ( + self.builder.current_function.dfg.get_numeric_constant(start_index), + self.builder.current_function.dfg.get_numeric_constant(end_index), + ) { + // If we can determine that the loop contains zero iterations then there's no need to codegen the loop. + if start_constant >= end_constant { + return Ok(Self::unit_value()); + } + } + let loop_entry = self.builder.insert_block(); let loop_body = self.builder.insert_block(); let loop_end = self.builder.insert_block(); @@ -535,13 +558,7 @@ impl<'a> FunctionContext<'a> { // Remember the blocks and variable used in case there are break/continue instructions // within the loop which need to jump to them. - self.enter_loop(loop_entry, loop_index, loop_end); - - self.builder.set_location(for_expr.start_range_location); - let start_index = self.codegen_non_tuple_expression(&for_expr.start_range)?; - - self.builder.set_location(for_expr.end_range_location); - let end_index = self.codegen_non_tuple_expression(&for_expr.end_range)?; + self.enter_loop(Loop { loop_entry, loop_index: Some(loop_index), loop_end }); // Set the location of the initial jmp instruction to the start range. This is the location // used to issue an error if the start range cannot be determined at compile-time. @@ -571,6 +588,38 @@ impl<'a> FunctionContext<'a> { Ok(Self::unit_value()) } + /// Codegens a loop, creating three new blocks in the process. + /// The return value of a loop is always a unit literal. + /// + /// For example, the loop `loop { body }` is codegen'd as: + /// + /// ```text + /// br loop_body() + /// loop_body(): + /// v3 = ... codegen body ... + /// br loop_body() + /// loop_end(): + /// ... This is the current insert point after codegen_for finishes ... + /// ``` + fn codegen_loop(&mut self, block: &Expression) -> Result { + let loop_body = self.builder.insert_block(); + let loop_end = self.builder.insert_block(); + + self.enter_loop(Loop { loop_entry: loop_body, loop_index: None, loop_end }); + + self.builder.terminate_with_jmp(loop_body, vec![]); + + // Compile the loop body + self.builder.switch_to_block(loop_body); + self.codegen_expression(block)?; + self.builder.terminate_with_jmp(loop_body, vec![]); + + // Finish by switching to the end of the loop + self.builder.switch_to_block(loop_end); + self.exit_loop(); + Ok(Self::unit_value()) + } + /// Codegens an if expression, handling the case of what to do if there is no 'else'. /// /// For example, the expression `if cond { a } else { b }` is codegen'd as: @@ -601,6 +650,9 @@ impl<'a> FunctionContext<'a> { /// ``` fn codegen_if(&mut self, if_expr: &ast::If) -> Result { let condition = self.codegen_non_tuple_expression(&if_expr.condition)?; + if let Some(result) = self.try_codegen_constant_if(condition, if_expr) { + return result; + } let then_block = self.builder.insert_block(); let else_block = self.builder.insert_block(); @@ -639,6 +691,25 @@ impl<'a> FunctionContext<'a> { Ok(result) } + /// If the condition is known, skip codegen for the then/else branch and only compile the + /// relevant branch. + fn try_codegen_constant_if( + &mut self, + condition: ValueId, + if_expr: &ast::If, + ) -> Option> { + let condition = self.builder.current_function.dfg.get_numeric_constant(condition)?; + + Some(if condition.is_zero() { + match if_expr.alternative.as_ref() { + Some(alternative) => self.codegen_expression(alternative), + None => Ok(Self::unit_value()), + } + } else { + self.codegen_expression(&if_expr.consequence) + }) + } + fn codegen_tuple(&mut self, tuple: &[Expression]) -> Result { Ok(Tree::Branch(try_vecmap(tuple, |expr| self.codegen_expression(expr))?)) } @@ -685,7 +756,12 @@ impl<'a> FunctionContext<'a> { // We add one here in the case of a slice insert as a slice insert at the length of the slice // can be converted to a slice push back - let len_plus_one = self.builder.insert_binary(arguments[0], BinaryOp::Add, one); + // This is unchecked as the slice length could be u32::max + let len_plus_one = self.builder.insert_binary( + arguments[0], + BinaryOp::Add { unchecked: false }, + one, + ); self.codegen_slice_access_check(arguments[2], Some(len_plus_one)); } @@ -709,9 +785,7 @@ impl<'a> FunctionContext<'a> { // Don't mutate the reference count if we're assigning an array literal to a Let: // `let mut foo = [1, 2, 3];` // we consider the array to be moved, so we should have an initial rc of just 1. - // - // TODO: this exception breaks #6763 - let should_inc_rc = true; // !let_expr.expression.is_array_or_slice_literal(); + let should_inc_rc = !let_expr.expression.is_array_or_slice_literal(); values = values.map(|value| { let value = value.eval(self); @@ -811,8 +885,12 @@ impl<'a> FunctionContext<'a> { let loop_ = self.current_loop(); // Must remember to increment i before jumping - let new_loop_index = self.make_offset(loop_.loop_index, 1); - self.builder.terminate_with_jmp(loop_.loop_entry, vec![new_loop_index]); + if let Some(loop_index) = loop_.loop_index { + let new_loop_index = self.make_offset(loop_index, 1); + self.builder.terminate_with_jmp(loop_.loop_entry, vec![new_loop_index]); + } else { + self.builder.terminate_with_jmp(loop_.loop_entry, vec![]); + } Self::unit_value() } } diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/program.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/program.rs index de01a4596ad..7cd5c5c3990 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/program.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/program.rs @@ -1,22 +1,27 @@ -use std::{collections::BTreeMap, fmt::Display}; +use std::collections::BTreeMap; use acvm::acir::circuit::ErrorSelector; +use fxhash::FxHashSet as HashSet; use iter_extended::btree_map; use serde::{Deserialize, Serialize}; use serde_with::serde_as; use crate::ssa::ir::{ - function::{Function, FunctionId, RuntimeType}, + function::{Function, FunctionId}, map::AtomicCounter, }; use noirc_frontend::hir_def::types::Type as HirType; +use super::ValueId; + /// Contains the entire SSA representation of the program. #[serde_as] #[derive(Serialize, Deserialize)] pub(crate) struct Ssa { #[serde_as(as = "Vec<(_, _)>")] pub(crate) functions: BTreeMap, + pub(crate) globals: Function, + pub(crate) used_global_values: HashSet, pub(crate) main_id: FunctionId, #[serde(skip)] pub(crate) next_id: AtomicCounter, @@ -53,6 +58,12 @@ impl Ssa { next_id: AtomicCounter::starting_after(max_id), entry_point_to_generated_index: BTreeMap::new(), error_selector_to_type: error_types, + // These fields should be set afterwards as globals are generated + // outside of the FunctionBuilder, which is where the `Ssa` is instantiated. + globals: Function::new_for_globals(), + // This field is set only after running DIE and is utilized + // for optimizing implementation of globals post-SSA. + used_global_values: HashSet::default(), } } @@ -62,6 +73,7 @@ impl Ssa { } /// Returns the entry-point function of the program as a mutable reference + #[cfg(test)] pub(crate) fn main_mut(&mut self) -> &mut Function { self.functions.get_mut(&self.main_id).expect("ICE: Ssa should have a main function") } @@ -77,29 +89,10 @@ impl Ssa { new_id } - /// Clones an already existing function with a fresh id - pub(crate) fn clone_fn(&mut self, existing_function_id: FunctionId) -> FunctionId { - let new_id = self.next_id.next(); - let function = Function::clone_with_id(new_id, &self.functions[&existing_function_id]); - self.functions.insert(new_id, function); - new_id - } pub(crate) fn generate_entry_point_index(mut self) -> Self { - self.entry_point_to_generated_index = btree_map( - self.functions - .iter() - .filter(|(_, func)| { - let runtime = func.runtime(); - match func.runtime() { - RuntimeType::Acir(_) => { - runtime.is_entry_point() || func.id() == self.main_id - } - RuntimeType::Brillig(_) => false, - } - }) - .enumerate(), - |(i, (id, _))| (*id, i as u32), - ); + let entry_points = + self.functions.keys().filter(|function| self.is_entry_point(**function)).enumerate(); + self.entry_point_to_generated_index = btree_map(entry_points, |(i, id)| (*id, i as u32)); self } @@ -111,14 +104,9 @@ impl Ssa { ); self.entry_point_to_generated_index.get(func_id).copied() } -} -impl Display for Ssa { - fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - for function in self.functions.values() { - writeln!(f, "{function}")?; - } - Ok(()) + pub(crate) fn is_entry_point(&self, function: FunctionId) -> bool { + function == self.main_id || self.functions[&function].runtime().is_entry_point() } } @@ -143,8 +131,8 @@ mod test { let one = builder.field_constant(1u128); let three = builder.field_constant(3u128); - let v1 = builder.insert_binary(v0, BinaryOp::Add, one); - let v2 = builder.insert_binary(v1, BinaryOp::Mul, three); + let v1 = builder.insert_binary(v0, BinaryOp::Add { unchecked: false }, one); + let v2 = builder.insert_binary(v1, BinaryOp::Mul { unchecked: false }, three); builder.terminate_with_return(vec![v2]); let ssa = builder.finish(); diff --git a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/value.rs b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/value.rs index d71d4e5604e..5d148dd181c 100644 --- a/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/value.rs +++ b/noir/noir-repo/compiler/noirc_evaluator/src/ssa/ssa_gen/value.rs @@ -96,18 +96,6 @@ impl Tree { } } - /// Map mutably over this tree, mutating each leaf value within using the given function - pub(super) fn map_mut(&mut self, mut f: impl FnMut(&T) -> Tree) { - self.map_mut_helper(&mut f); - } - - fn map_mut_helper(&mut self, f: &mut impl FnMut(&T) -> Tree) { - match self { - Tree::Branch(trees) => trees.iter_mut().for_each(|tree| tree.map_mut_helper(f)), - Tree::Leaf(value) => *self = f(value), - } - } - /// Calls the given function on each leaf node, mapping this tree into a new one. /// /// Because the given function returns a Tree rather than a U, it is possible diff --git a/noir/noir-repo/compiler/noirc_frontend/Cargo.toml b/noir/noir-repo/compiler/noirc_frontend/Cargo.toml index 5f8f02689c8..041c1b1e015 100644 --- a/noir/noir-repo/compiler/noirc_frontend/Cargo.toml +++ b/noir/noir-repo/compiler/noirc_frontend/Cargo.toml @@ -31,6 +31,7 @@ petgraph = "0.6" rangemap = "1.4.0" strum.workspace = true strum_macros.workspace = true +fxhash.workspace = true [dev-dependencies] diff --git a/noir/noir-repo/compiler/noirc_frontend/src/ast/enumeration.rs b/noir/noir-repo/compiler/noirc_frontend/src/ast/enumeration.rs new file mode 100644 index 00000000000..eeeb823b9fc --- /dev/null +++ b/noir/noir-repo/compiler/noirc_frontend/src/ast/enumeration.rs @@ -0,0 +1,50 @@ +use std::fmt::Display; + +use crate::ast::{Ident, UnresolvedGenerics, UnresolvedType}; +use crate::token::SecondaryAttribute; + +use iter_extended::vecmap; +use noirc_errors::Span; + +use super::{Documented, ItemVisibility}; + +/// Ast node for an enum +#[derive(Clone, Debug, PartialEq, Eq)] +pub struct NoirEnumeration { + pub name: Ident, + pub attributes: Vec, + pub visibility: ItemVisibility, + pub generics: UnresolvedGenerics, + pub variants: Vec>, + pub span: Span, +} + +impl NoirEnumeration { + pub fn is_abi(&self) -> bool { + self.attributes.iter().any(|attr| attr.is_abi()) + } +} + +/// We only support variants of the form `Name(A, B, ...)` currently. +/// Enum variants like `Name { a: A, b: B, .. }` will be implemented later +#[derive(Clone, Debug, PartialEq, Eq)] +pub struct EnumVariant { + pub name: Ident, + pub parameters: Vec, +} + +impl Display for NoirEnumeration { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + let generics = vecmap(&self.generics, |generic| generic.to_string()); + let generics = if generics.is_empty() { "".into() } else { generics.join(", ") }; + + writeln!(f, "enum {}{} {{", self.name, generics)?; + + for variant in self.variants.iter() { + let parameters = vecmap(&variant.item.parameters, ToString::to_string).join(", "); + writeln!(f, " {}({}),", variant.item.name, parameters)?; + } + + write!(f, "}}") + } +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/ast/expression.rs b/noir/noir-repo/compiler/noirc_frontend/src/ast/expression.rs index ae622f46686..9d521545e7a 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/ast/expression.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/ast/expression.rs @@ -821,8 +821,8 @@ impl FunctionDefinition { is_unconstrained: bool, generics: &UnresolvedGenerics, parameters: &[(Ident, UnresolvedType)], - body: &BlockExpression, - where_clause: &[UnresolvedTraitConstraint], + body: BlockExpression, + where_clause: Vec, return_type: &FunctionReturnType, ) -> FunctionDefinition { let p = parameters @@ -843,9 +843,9 @@ impl FunctionDefinition { visibility: ItemVisibility::Private, generics: generics.clone(), parameters: p, - body: body.clone(), + body, span: name.span(), - where_clause: where_clause.to_vec(), + where_clause, return_type: return_type.clone(), return_visibility: Visibility::Private, } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/ast/function.rs b/noir/noir-repo/compiler/noirc_frontend/src/ast/function.rs index 99ae78c93ea..8957564e0d6 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/ast/function.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/ast/function.rs @@ -19,22 +19,27 @@ pub struct NoirFunction { pub def: FunctionDefinition, } -/// Currently, we support three types of functions: +/// Currently, we support four types of functions: /// - Normal functions /// - LowLevel/Foreign which link to an OPCODE in ACIR /// - BuiltIn which are provided by the runtime +/// - TraitFunctionWithoutBody for which we don't type-check their body #[derive(Copy, Clone, Debug, PartialEq, Eq)] pub enum FunctionKind { LowLevel, Builtin, Normal, Oracle, + TraitFunctionWithoutBody, } impl FunctionKind { pub fn can_ignore_return_type(self) -> bool { match self { - FunctionKind::LowLevel | FunctionKind::Builtin | FunctionKind::Oracle => true, + FunctionKind::LowLevel + | FunctionKind::Builtin + | FunctionKind::Oracle + | FunctionKind::TraitFunctionWithoutBody => true, FunctionKind::Normal => false, } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/ast/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/ast/mod.rs index 35e57cd4528..33f504437c0 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/ast/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/ast/mod.rs @@ -5,6 +5,7 @@ //! Noir's Ast is produced by the parser and taken as input to name resolution, //! where it is converted into the Hir (defined in the hir_def module). mod docs; +mod enumeration; mod expression; mod function; mod statement; @@ -24,6 +25,7 @@ use proptest_derive::Arbitrary; use acvm::FieldElement; pub use docs::*; +pub use enumeration::*; use noirc_errors::Span; use serde::{Deserialize, Serialize}; pub use statement::*; @@ -580,12 +582,13 @@ impl std::fmt::Display for ItemVisibility { } } -#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] +#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, Serialize, Deserialize, Default)] /// Represents whether the parameter is public or known only to the prover. pub enum Visibility { Public, // Constants are not allowed in the ABI for main at the moment. // Constant, + #[default] Private, /// DataBus is public input handled as private input. We use the fact that return values are properly computed by the program to avoid having them as public inputs /// it is useful for recursion and is handled by the proving system. diff --git a/noir/noir-repo/compiler/noirc_frontend/src/ast/statement.rs b/noir/noir-repo/compiler/noirc_frontend/src/ast/statement.rs index c77fe7513a1..57572e80d1e 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/ast/statement.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/ast/statement.rs @@ -1,5 +1,4 @@ use std::fmt::Display; -use std::sync::atomic::{AtomicU32, Ordering}; use acvm::acir::AcirField; use acvm::FieldElement; @@ -46,6 +45,7 @@ pub enum StatementKind { Expression(Expression), Assign(AssignStatement), For(ForLoopStatement), + Loop(Expression), Break, Continue, /// This statement should be executed at compile-time @@ -107,6 +107,9 @@ impl StatementKind { // A semicolon on a for loop is optional and does nothing StatementKind::For(_) => self, + // A semicolon on a loop is optional and does nothing + StatementKind::Loop(..) => self, + // No semicolon needed for a resolved statement StatementKind::Interned(_) => self, @@ -154,6 +157,7 @@ impl StatementKind { r#type, expression, comptime: false, + is_global_let: false, attributes, }) } @@ -562,6 +566,7 @@ pub struct LetStatement { // True if this should only be run during compile-time pub comptime: bool, + pub is_global_let: bool, } #[derive(Debug, PartialEq, Eq, Clone)] @@ -839,10 +844,9 @@ impl ForRange { block: Expression, for_loop_span: Span, ) -> Statement { - /// Counter used to generate unique names when desugaring - /// code in the parser requires the creation of fresh variables. - /// The parser is stateless so this is a static global instead. - static UNIQUE_NAME_COUNTER: AtomicU32 = AtomicU32::new(0); + // Counter used to generate unique names when desugaring + // code in the parser requires the creation of fresh variables. + let mut unique_name_counter: u32 = 0; match self { ForRange::Range(..) => { @@ -853,7 +857,8 @@ impl ForRange { let start_range = ExpressionKind::integer(FieldElement::zero()); let start_range = Expression::new(start_range, array_span); - let next_unique_id = UNIQUE_NAME_COUNTER.fetch_add(1, Ordering::Relaxed); + let next_unique_id = unique_name_counter; + unique_name_counter += 1; let array_name = format!("$i{next_unique_id}"); let array_span = array.span; let array_ident = Ident::new(array_name, array_span); @@ -886,7 +891,7 @@ impl ForRange { })); let end_range = Expression::new(end_range, array_span); - let next_unique_id = UNIQUE_NAME_COUNTER.fetch_add(1, Ordering::Relaxed); + let next_unique_id = unique_name_counter; let index_name = format!("$i{next_unique_id}"); let fresh_identifier = Ident::new(index_name.clone(), array_span); @@ -960,6 +965,7 @@ impl Display for StatementKind { StatementKind::Expression(expression) => expression.fmt(f), StatementKind::Assign(assign) => assign.fmt(f), StatementKind::For(for_loop) => for_loop.fmt(f), + StatementKind::Loop(block) => write!(f, "loop {}", block), StatementKind::Break => write!(f, "break"), StatementKind::Continue => write!(f, "continue"), StatementKind::Comptime(statement) => write!(f, "comptime {}", statement.kind), diff --git a/noir/noir-repo/compiler/noirc_frontend/src/ast/visitor.rs b/noir/noir-repo/compiler/noirc_frontend/src/ast/visitor.rs index 2f60532980a..5c4781df7a5 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/ast/visitor.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/ast/visitor.rs @@ -21,15 +21,17 @@ use crate::{ }; use super::{ - ForBounds, FunctionReturnType, GenericTypeArgs, IntegerBitSize, ItemVisibility, Pattern, - Signedness, TraitBound, TraitImplItemKind, TypePath, UnresolvedGenerics, - UnresolvedTraitConstraint, UnresolvedType, UnresolvedTypeData, UnresolvedTypeExpression, + ForBounds, FunctionReturnType, GenericTypeArgs, IntegerBitSize, ItemVisibility, + NoirEnumeration, Pattern, Signedness, TraitBound, TraitImplItemKind, TypePath, + UnresolvedGenerics, UnresolvedTraitConstraint, UnresolvedType, UnresolvedTypeData, + UnresolvedTypeExpression, }; #[derive(Debug, Copy, Clone, PartialEq, Eq)] pub enum AttributeTarget { Module, Struct, + Enum, Trait, Function, Let, @@ -142,6 +144,10 @@ pub trait Visitor { true } + fn visit_noir_enum(&mut self, _: &NoirEnumeration, _: Span) -> bool { + true + } + fn visit_noir_type_alias(&mut self, _: &NoirTypeAlias, _: Span) -> bool { true } @@ -296,6 +302,10 @@ pub trait Visitor { true } + fn visit_loop_statement(&mut self, _: &Expression) -> bool { + true + } + fn visit_comptime_statement(&mut self, _: &Statement) -> bool { true } @@ -523,6 +533,7 @@ impl Item { } ItemKind::TypeAlias(noir_type_alias) => noir_type_alias.accept(self.span, visitor), ItemKind::Struct(noir_struct) => noir_struct.accept(self.span, visitor), + ItemKind::Enum(noir_enum) => noir_enum.accept(self.span, visitor), ItemKind::ModuleDecl(module_declaration) => { module_declaration.accept(self.span, visitor); } @@ -771,6 +782,26 @@ impl NoirStruct { } } +impl NoirEnumeration { + pub fn accept(&self, span: Span, visitor: &mut impl Visitor) { + if visitor.visit_noir_enum(self, span) { + self.accept_children(visitor); + } + } + + pub fn accept_children(&self, visitor: &mut impl Visitor) { + for attribute in &self.attributes { + attribute.accept(AttributeTarget::Enum, visitor); + } + + for variant in &self.variants { + for parameter in &variant.item.parameters { + parameter.accept(visitor); + } + } + } +} + impl NoirTypeAlias { pub fn accept(&self, span: Span, visitor: &mut impl Visitor) { if visitor.visit_noir_type_alias(self, span) { @@ -1104,6 +1135,11 @@ impl Statement { StatementKind::For(for_loop_statement) => { for_loop_statement.accept(visitor); } + StatementKind::Loop(block) => { + if visitor.visit_loop_statement(block) { + block.accept(visitor); + } + } StatementKind::Comptime(statement) => { if visitor.visit_comptime_statement(statement) { statement.accept(visitor); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/debug/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/debug/mod.rs index f05fc721581..808bbda677c 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/debug/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/debug/mod.rs @@ -523,7 +523,9 @@ pub fn build_debug_crate_file() -> String { __debug_var_assign_oracle(var_id, value); } pub fn __debug_var_assign(var_id: u32, value: T) { - unsafe {{ + /// Safety: debug context + unsafe { + { __debug_var_assign_inner(var_id, value); }} } @@ -534,7 +536,9 @@ pub fn build_debug_crate_file() -> String { __debug_var_drop_oracle(var_id); } pub fn __debug_var_drop(var_id: u32) { - unsafe {{ + /// Safety: debug context + unsafe { + { __debug_var_drop_inner(var_id); }} } @@ -545,7 +549,9 @@ pub fn build_debug_crate_file() -> String { __debug_fn_enter_oracle(fn_id); } pub fn __debug_fn_enter(fn_id: u32) { - unsafe {{ + /// Safety: debug context + unsafe { + { __debug_fn_enter_inner(fn_id); }} } @@ -556,7 +562,9 @@ pub fn build_debug_crate_file() -> String { __debug_fn_exit_oracle(fn_id); } pub fn __debug_fn_exit(fn_id: u32) { - unsafe {{ + /// Safety: debug context + unsafe { + { __debug_fn_exit_inner(fn_id); }} } @@ -567,7 +575,9 @@ pub fn build_debug_crate_file() -> String { __debug_dereference_assign_oracle(var_id, value); } pub fn __debug_dereference_assign(var_id: u32, value: T) { - unsafe {{ + /// Safety: debug context + unsafe { + { __debug_dereference_assign_inner(var_id, value); }} } @@ -593,6 +603,7 @@ pub fn build_debug_crate_file() -> String { __debug_oracle_member_assign_{n}(var_id, value, {vars}); }} pub fn __debug_member_assign_{n}(var_id: u32, value: T, {var_sig}) {{ + /// Safety: debug context unsafe {{ __debug_inner_member_assign_{n}(var_id, value, {vars}); }} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/comptime.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/comptime.rs index fe8c8338b32..9f5eef6e785 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/comptime.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/comptime.rs @@ -93,9 +93,11 @@ impl<'context> Elaborator<'context> { self.interner, self.def_maps, self.usage_tracker, + self.crate_graph, self.crate_id, self.debug_comptime_in_file, self.interpreter_call_stack.clone(), + self.pedantic_solving, ); elaborator.function_context.push(FunctionContext::default()); @@ -246,7 +248,7 @@ impl<'context> Elaborator<'context> { if value != Value::Unit { let items = value - .into_top_level_items(location, self.interner) + .into_top_level_items(location, self) .map_err(|error| error.into_compilation_error_pair())?; self.add_items(items, generated_items, location); @@ -440,7 +442,21 @@ impl<'context> Elaborator<'context> { self.crate_id, &mut self.errors, ) { - generated_items.types.insert(type_id, the_struct); + generated_items.structs.insert(type_id, the_struct); + } + } + ItemKind::Enum(enum_def) => { + if let Some((type_id, the_enum)) = dc_mod::collect_enum( + self.interner, + self.def_maps.get_mut(&self.crate_id).unwrap(), + self.usage_tracker, + Documented::new(enum_def, item.doc_comments), + self.file, + self.local_module, + self.crate_id, + &mut self.errors, + ) { + generated_items.enums.insert(type_id, the_enum); } } ItemKind::Impl(r#impl) => { @@ -473,7 +489,7 @@ impl<'context> Elaborator<'context> { Some(DependencyId::Function(function)) => Some(function), _ => None, }; - Interpreter::new(self, self.crate_id, current_function) + Interpreter::new(self, self.crate_id, current_function, self.pedantic_solving) } pub(super) fn debug_comptime T>( diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/expressions.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/expressions.rs index b4ea06f1030..ef2ae9c4df0 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/expressions.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/expressions.rs @@ -32,7 +32,7 @@ use crate::{ Kind, QuotedType, Shared, StructType, Type, }; -use super::{Elaborator, LambdaContext}; +use super::{Elaborator, LambdaContext, UnsafeBlockStatus}; impl<'context> Elaborator<'context> { pub(crate) fn elaborate_expression(&mut self, expr: Expression) -> (ExprId, Type) { @@ -52,14 +52,14 @@ impl<'context> Elaborator<'context> { ExpressionKind::If(if_) => self.elaborate_if(*if_), ExpressionKind::Variable(variable) => return self.elaborate_variable(variable), ExpressionKind::Tuple(tuple) => self.elaborate_tuple(tuple), - ExpressionKind::Lambda(lambda) => self.elaborate_lambda(*lambda), + ExpressionKind::Lambda(lambda) => self.elaborate_lambda(*lambda, None), ExpressionKind::Parenthesized(expr) => return self.elaborate_expression(*expr), ExpressionKind::Quote(quote) => self.elaborate_quote(quote, expr.span), ExpressionKind::Comptime(comptime, _) => { return self.elaborate_comptime_block(comptime, expr.span) } - ExpressionKind::Unsafe(block_expression, _) => { - self.elaborate_unsafe_block(block_expression) + ExpressionKind::Unsafe(block_expression, span) => { + self.elaborate_unsafe_block(block_expression, span) } ExpressionKind::Resolved(id) => return (id, self.interner.id_type(id)), ExpressionKind::Interned(id) => { @@ -140,15 +140,36 @@ impl<'context> Elaborator<'context> { (HirBlockExpression { statements }, block_type) } - fn elaborate_unsafe_block(&mut self, block: BlockExpression) -> (HirExpression, Type) { + fn elaborate_unsafe_block( + &mut self, + block: BlockExpression, + span: Span, + ) -> (HirExpression, Type) { // Before entering the block we cache the old value of `in_unsafe_block` so it can be restored. - let old_in_unsafe_block = self.in_unsafe_block; - self.in_unsafe_block = true; + let old_in_unsafe_block = self.unsafe_block_status; + let is_nested_unsafe_block = + !matches!(old_in_unsafe_block, UnsafeBlockStatus::NotInUnsafeBlock); + if is_nested_unsafe_block { + let span = Span::from(span.start()..span.start() + 6); // Only highlight the `unsafe` keyword + self.push_err(TypeCheckError::NestedUnsafeBlock { span }); + } + + self.unsafe_block_status = UnsafeBlockStatus::InUnsafeBlockWithoutUnconstrainedCalls; let (hir_block_expression, typ) = self.elaborate_block_expression(block); - // Finally, we restore the original value of `self.in_unsafe_block`. - self.in_unsafe_block = old_in_unsafe_block; + if let UnsafeBlockStatus::InUnsafeBlockWithoutUnconstrainedCalls = self.unsafe_block_status + { + let span = Span::from(span.start()..span.start() + 6); // Only highlight the `unsafe` keyword + self.push_err(TypeCheckError::UnnecessaryUnsafeBlock { span }); + } + + // Finally, we restore the original value of `self.in_unsafe_block`, + // but only if this isn't a nested unsafe block (that way if we found an unconstrained call + // for this unsafe block we'll also consider the outer one as finding one, and we don't double error) + if !is_nested_unsafe_block { + self.unsafe_block_status = old_in_unsafe_block; + } (HirExpression::Unsafe(hir_block_expression), typ) } @@ -366,17 +387,28 @@ impl<'context> Elaborator<'context> { fn elaborate_call(&mut self, call: CallExpression, span: Span) -> (HirExpression, Type) { let (func, func_type) = self.elaborate_expression(*call.func); + let func_arg_types = + if let Type::Function(args, _, _, _) = &func_type { Some(args) } else { None }; let mut arguments = Vec::with_capacity(call.arguments.len()); - let args = vecmap(call.arguments, |arg| { + let args = vecmap(call.arguments.into_iter().enumerate(), |(arg_index, arg)| { let span = arg.span; + let expected_type = func_arg_types.and_then(|args| args.get(arg_index)); let (arg, typ) = if call.is_macro_call { - self.elaborate_in_comptime_context(|this| this.elaborate_expression(arg)) + self.elaborate_in_comptime_context(|this| { + this.elaborate_expression_with_type(arg, expected_type) + }) } else { - self.elaborate_expression(arg) + self.elaborate_expression_with_type(arg, expected_type) }; + // Try to unify this argument type against the function's argument type + // so that a potential lambda following this argument can have more concrete types. + if let Some(expected_type) = expected_type { + let _ = expected_type.unify(&typ); + } + arguments.push(arg); (typ, arg, span) }); @@ -437,6 +469,32 @@ impl<'context> Elaborator<'context> { None }; + let call_span = Span::from(object_span.start()..method_name_span.end()); + let location = Location::new(call_span, self.file); + + let (function_id, function_name) = method_ref.clone().into_function_id_and_name( + object_type.clone(), + generics.clone(), + location, + self.interner, + ); + + let func_type = + self.type_check_variable(function_name.clone(), function_id, generics.clone()); + self.interner.push_expr_type(function_id, func_type.clone()); + + let func_arg_types = + if let Type::Function(args, _, _, _) = &func_type { Some(args) } else { None }; + + // Try to unify the object type with the first argument of the function. + // The reason to do this is that many methods that take a lambda will yield `self` or part of `self` + // as a parameter. By unifying `self` with the first argument we'll potentially get more + // concrete types in the arguments that are function types, which will later be passed as + // lambda parameter hints. + if let Some(first_arg_type) = func_arg_types.and_then(|args| args.first()) { + let _ = first_arg_type.unify(&object_type); + } + // These arguments will be given to the desugared function call. // Compared to the method arguments, they also contain the object. let mut function_args = Vec::with_capacity(method_call.arguments.len() + 1); @@ -444,17 +502,22 @@ impl<'context> Elaborator<'context> { function_args.push((object_type.clone(), object, object_span)); - for arg in method_call.arguments { + for (arg_index, arg) in method_call.arguments.into_iter().enumerate() { let span = arg.span; - let (arg, typ) = self.elaborate_expression(arg); + let expected_type = func_arg_types.and_then(|args| args.get(arg_index + 1)); + let (arg, typ) = self.elaborate_expression_with_type(arg, expected_type); + + // Try to unify this argument type against the function's argument type + // so that a potential lambda following this argument can have more concrete types. + if let Some(expected_type) = expected_type { + let _ = expected_type.unify(&typ); + } + arguments.push(arg); function_args.push((typ, arg, span)); } - let call_span = Span::from(object_span.start()..method_name_span.end()); - let location = Location::new(call_span, self.file); let method = method_call.method_name; - let turbofish_generics = generics.clone(); let is_macro_call = method_call.is_macro_call; let method_call = HirMethodCallExpression { method, object, arguments, location, generics }; @@ -464,18 +527,9 @@ impl<'context> Elaborator<'context> { // Desugar the method call into a normal, resolved function call // so that the backend doesn't need to worry about methods // TODO: update object_type here? - let ((function_id, function_name), function_call) = method_call.into_function_call( - method_ref, - object_type, - is_macro_call, - location, - self.interner, - ); - let func_type = - self.type_check_variable(function_name, function_id, turbofish_generics); - - self.interner.push_expr_type(function_id, func_type.clone()); + let function_call = + method_call.into_function_call(function_id, is_macro_call, location); self.interner .add_function_reference(func_id, Location::new(method_name_span, self.file)); @@ -499,6 +553,26 @@ impl<'context> Elaborator<'context> { } } + /// Elaborates an expression knowing that it has to match a given type. + fn elaborate_expression_with_type( + &mut self, + arg: Expression, + typ: Option<&Type>, + ) -> (ExprId, Type) { + let ExpressionKind::Lambda(lambda) = arg.kind else { + return self.elaborate_expression(arg); + }; + + let span = arg.span; + let type_hint = + if let Some(Type::Function(func_args, _, _, _)) = typ { Some(func_args) } else { None }; + let (hir_expr, typ) = self.elaborate_lambda(*lambda, type_hint); + let id = self.interner.push_expr(hir_expr); + self.interner.push_expr_location(id, span, self.file); + self.interner.push_expr_type(id, typ.clone()); + (id, typ) + } + fn check_method_call_visibility(&mut self, func_id: FuncId, object_type: &Type, name: &Ident) { if !method_call_is_visible( object_type, @@ -754,7 +828,10 @@ impl<'context> Elaborator<'context> { span, }, }; - self.push_trait_constraint(constraint, expr_id); + self.push_trait_constraint( + constraint, expr_id, + true, // this constraint should lead to choosing a trait impl + ); self.type_check_operator_method(expr_id, trait_id, operand_type, span); } typ @@ -822,19 +899,38 @@ impl<'context> Elaborator<'context> { (HirExpression::Tuple(element_ids), Type::Tuple(element_types)) } - fn elaborate_lambda(&mut self, lambda: Lambda) -> (HirExpression, Type) { + /// For elaborating a lambda we might get `parameters_type_hints`. These come from a potential + /// call that has this lambda as the argument. + /// The parameter type hints will be the types of the function type corresponding to the lambda argument. + fn elaborate_lambda( + &mut self, + lambda: Lambda, + parameters_type_hints: Option<&Vec>, + ) -> (HirExpression, Type) { self.push_scope(); let scope_index = self.scopes.current_scope_index(); self.lambda_stack.push(LambdaContext { captures: Vec::new(), scope_index }); let mut arg_types = Vec::with_capacity(lambda.parameters.len()); - let parameters = vecmap(lambda.parameters, |(pattern, typ)| { - let parameter = DefinitionKind::Local(None); - let typ = self.resolve_inferred_type(typ); - arg_types.push(typ.clone()); - (self.elaborate_pattern(pattern, typ.clone(), parameter, true), typ) - }); + let parameters = + vecmap(lambda.parameters.into_iter().enumerate(), |(index, (pattern, typ))| { + let parameter = DefinitionKind::Local(None); + let typ = if let UnresolvedTypeData::Unspecified = typ.typ { + if let Some(parameter_type_hint) = + parameters_type_hints.and_then(|hints| hints.get(index)) + { + parameter_type_hint.clone() + } else { + self.interner.next_type_variable_with_kind(Kind::Any) + } + } else { + self.resolve_type(typ) + }; + + arg_types.push(typ.clone()); + (self.elaborate_pattern(pattern, typ.clone(), parameter, true), typ) + }); let return_type = self.resolve_inferred_type(lambda.return_type); let body_span = lambda.body.span; @@ -906,7 +1002,7 @@ impl<'context> Elaborator<'context> { }; let location = Location::new(span, self.file); - match value.into_expression(self.interner, location) { + match value.into_expression(self, location) { Ok(new_expr) => { // At this point the Expression was already elaborated and we got a Value. // We'll elaborate this value turned into Expression to inline it and get diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/lints.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/lints.rs index d3b776bea24..82a5e19054b 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/lints.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/lints.rs @@ -282,6 +282,7 @@ fn can_return_without_recursing(interner: &NodeInterner, func_id: FuncId, expr_i HirStatement::Semi(e) => check(e), // Rust doesn't seem to check the for loop body (it's bounds might mean it's never called). HirStatement::For(e) => check(e.start_range) && check(e.end_range), + HirStatement::Loop(e) => check(e), HirStatement::Constrain(_) | HirStatement::Comptime(_) | HirStatement::Break diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/mod.rs index fe1d1e38e1a..79f6be444ce 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/mod.rs @@ -3,10 +3,6 @@ use std::{ rc::Rc, }; -use crate::{ - ast::ItemVisibility, hir_def::traits::ResolvedTraitBound, usage_tracker::UsageTracker, - StructField, StructType, TypeBindings, -}; use crate::{ ast::{ BlockExpression, FunctionKind, GenericTypeArgs, Ident, NoirFunction, NoirStruct, Param, @@ -41,6 +37,14 @@ use crate::{ token::SecondaryAttribute, Shared, Type, TypeVariable, }; +use crate::{ + ast::{ItemVisibility, UnresolvedType}, + graph::CrateGraph, + hir_def::traits::ResolvedTraitBound, + node_interner::GlobalValue, + usage_tracker::UsageTracker, + StructField, StructType, TypeBindings, +}; mod comptime; mod expressions; @@ -79,6 +83,16 @@ pub struct LambdaContext { pub scope_index: usize, } +/// Determines whether we are in an unsafe block and, if so, whether +/// any unconstrained calls were found in it (because if not we'll warn +/// that the unsafe block is not needed). +#[derive(Copy, Clone)] +enum UnsafeBlockStatus { + NotInUnsafeBlock, + InUnsafeBlockWithoutUnconstrainedCalls, + InUnsafeBlockWithConstrainedCalls, +} + pub struct Elaborator<'context> { scopes: ScopeForest, @@ -87,10 +101,11 @@ pub struct Elaborator<'context> { pub(crate) interner: &'context mut NodeInterner, pub(crate) def_maps: &'context mut DefMaps, pub(crate) usage_tracker: &'context mut UsageTracker, + pub(crate) crate_graph: &'context CrateGraph, pub(crate) file: FileId, - in_unsafe_block: bool, + unsafe_block_status: UnsafeBlockStatus, nested_loops: usize, /// Contains a mapping of the current struct or functions's generics to @@ -170,6 +185,9 @@ pub struct Elaborator<'context> { /// like `Foo { inner: 5 }`: in that case we already elaborated the code that led to /// that comptime value and any visibility errors were already reported. silence_field_visibility_errors: usize, + + /// Use pedantic ACVM solving + pedantic_solving: bool, } #[derive(Default)] @@ -183,17 +201,24 @@ struct FunctionContext { /// verified at the end of a function. This is because constraints arise /// on each variable, but it is only until function calls when the types /// needed for the trait constraint may become known. - trait_constraints: Vec<(TraitConstraint, ExprId)>, + /// The `select impl` bool indicates whether, after verifying the trait constraint, + /// the resulting trait impl should be the one used for a call (sometimes trait + /// constraints are verified but there's no call associated with them, like in the + /// case of checking generic arguments) + trait_constraints: Vec<(TraitConstraint, ExprId, bool /* select impl */)>, } impl<'context> Elaborator<'context> { + #[allow(clippy::too_many_arguments)] pub fn new( interner: &'context mut NodeInterner, def_maps: &'context mut DefMaps, usage_tracker: &'context mut UsageTracker, + crate_graph: &'context CrateGraph, crate_id: CrateId, debug_comptime_in_file: Option, interpreter_call_stack: im::Vector, + pedantic_solving: bool, ) -> Self { Self { scopes: ScopeForest::default(), @@ -201,8 +226,9 @@ impl<'context> Elaborator<'context> { interner, def_maps, usage_tracker, + crate_graph, file: FileId::dummy(), - in_unsafe_block: false, + unsafe_block_status: UnsafeBlockStatus::NotInUnsafeBlock, nested_loops: 0, generics: Vec::new(), lambda_stack: Vec::new(), @@ -220,6 +246,7 @@ impl<'context> Elaborator<'context> { interpreter_call_stack, in_comptime_context: false, silence_field_visibility_errors: 0, + pedantic_solving, } } @@ -227,14 +254,17 @@ impl<'context> Elaborator<'context> { context: &'context mut Context, crate_id: CrateId, debug_comptime_in_file: Option, + pedantic_solving: bool, ) -> Self { Self::new( &mut context.def_interner, &mut context.def_maps, &mut context.usage_tracker, + &context.crate_graph, crate_id, debug_comptime_in_file, im::Vector::new(), + pedantic_solving, ) } @@ -243,8 +273,16 @@ impl<'context> Elaborator<'context> { crate_id: CrateId, items: CollectedItems, debug_comptime_in_file: Option, + pedantic_solving: bool, ) -> Vec<(CompilationError, FileId)> { - Self::elaborate_and_return_self(context, crate_id, items, debug_comptime_in_file).errors + Self::elaborate_and_return_self( + context, + crate_id, + items, + debug_comptime_in_file, + pedantic_solving, + ) + .errors } pub fn elaborate_and_return_self( @@ -252,8 +290,10 @@ impl<'context> Elaborator<'context> { crate_id: CrateId, items: CollectedItems, debug_comptime_in_file: Option, + pedantic_solving: bool, ) -> Self { - let mut this = Self::from_context(context, crate_id, debug_comptime_in_file); + let mut this = + Self::from_context(context, crate_id, debug_comptime_in_file, pedantic_solving); this.elaborate_items(items); this.check_and_pop_function_context(); this @@ -279,11 +319,11 @@ impl<'context> Elaborator<'context> { } // Must resolve structs before we resolve globals. - self.collect_struct_definitions(&items.types); + self.collect_struct_definitions(&items.structs); self.define_function_metas(&mut items.functions, &mut items.impls, &mut items.trait_impls); - self.collect_traits(&items.traits); + self.collect_traits(&mut items.traits); // Before we resolve any function symbols we must go through our impls and // re-collect the methods within into their proper module. This cannot be @@ -309,7 +349,7 @@ impl<'context> Elaborator<'context> { // since the generated items are checked beforehand as well. self.run_attributes( &items.traits, - &items.types, + &items.structs, &items.functions, &items.module_attributes, ); @@ -318,6 +358,13 @@ impl<'context> Elaborator<'context> { self.elaborate_functions(functions); } + for (trait_id, unresolved_trait) in items.traits { + self.current_trait = Some(trait_id); + self.elaborate_functions(unresolved_trait.fns_with_default_impl); + } + + self.current_trait = None; + for impls in items.impls.into_values() { self.elaborate_impls(impls); } @@ -437,12 +484,13 @@ impl<'context> Elaborator<'context> { self.add_existing_variable_to_scope(name, parameter.clone(), warn_if_unused); } - self.add_trait_constraints_to_scope(&func_meta); + self.add_trait_constraints_to_scope(&func_meta.trait_constraints, func_meta.location.span); let (hir_func, body_type) = match kind { - FunctionKind::Builtin | FunctionKind::LowLevel | FunctionKind::Oracle => { - (HirFunction::empty(), Type::Error) - } + FunctionKind::Builtin + | FunctionKind::LowLevel + | FunctionKind::Oracle + | FunctionKind::TraitFunctionWithoutBody => (HirFunction::empty(), Type::Error), FunctionKind::Normal => { let (block, body_type) = self.elaborate_block(body); let expr_id = self.intern_expr(block, body_span); @@ -462,11 +510,7 @@ impl<'context> Elaborator<'context> { // when multiple impls are available. Instead we default first to choose the Field or u64 impl. self.check_and_pop_function_context(); - // Now remove all the `where` clause constraints we added - for constraint in &func_meta.trait_constraints { - self.interner - .remove_assumed_trait_implementations_for_trait(constraint.trait_bound.trait_id); - } + self.remove_trait_constraints_from_scope(&func_meta.trait_constraints); let func_scope_tree = self.scopes.end_function(); @@ -514,7 +558,7 @@ impl<'context> Elaborator<'context> { } } - for (mut constraint, expr_id) in context.trait_constraints { + for (mut constraint, expr_id, select_impl) in context.trait_constraints { let span = self.interner.expr_span(&expr_id); if matches!(&constraint.typ, Type::MutableReference(_)) { @@ -529,6 +573,7 @@ impl<'context> Elaborator<'context> { &constraint.trait_bound.trait_generics.ordered, &constraint.trait_bound.trait_generics.named, expr_id, + select_impl, span, ); } @@ -698,8 +743,13 @@ impl<'context> Elaborator<'context> { None } - /// TODO: This is currently only respected for generic free functions - /// there's a bunch of other places where trait constraints can pop up + /// Resolve the given trait constraints and add them to scope as we go. + /// This second step is necessary to resolve subsequent constraints such + /// as `::Bar: Eq` which may lookup an impl which was assumed + /// by a previous constraint. + /// + /// If these constraints are unwanted afterward they should be manually + /// removed from the interner. fn resolve_trait_constraints( &mut self, where_clause: &[UnresolvedTraitConstraint], @@ -710,12 +760,102 @@ impl<'context> Elaborator<'context> { .collect() } - pub fn resolve_trait_constraint( + /// Expands any traits in a where clause to mention all associated types if they were + /// elided by the user. See `add_missing_named_generics` for more detail. + /// + /// Returns all newly created generics to be added to this function/trait/impl. + fn desugar_trait_constraints( + &mut self, + where_clause: &mut [UnresolvedTraitConstraint], + ) -> Vec { + where_clause + .iter_mut() + .flat_map(|constraint| { + self.add_missing_named_generics(&constraint.typ, &mut constraint.trait_bound) + }) + .collect() + } + + /// For each associated type that isn't mentioned in a trait bound, this adds + /// the type as an implicit generic to the where clause and returns the newly + /// created generics in a vector to add to the function/trait/impl later. + /// For example, this will turn a function using a trait with 2 associated types: + /// + /// `fn foo() where T: Foo { ... }` + /// + /// into: + /// `fn foo() where T: Foo { ... }` + /// + /// with a vector of `` returned so that the caller can then modify the function to: + /// `fn foo() where T: Foo { ... }` + fn add_missing_named_generics( + &mut self, + object: &UnresolvedType, + bound: &mut TraitBound, + ) -> Vec { + let mut added_generics = Vec::new(); + + let Ok(item) = self.resolve_path_or_error(bound.trait_path.clone()) else { + return Vec::new(); + }; + + let PathResolutionItem::Trait(trait_id) = item else { + return Vec::new(); + }; + + let the_trait = self.get_trait_mut(trait_id); + + if the_trait.associated_types.len() > bound.trait_generics.named_args.len() { + let trait_name = the_trait.name.to_string(); + + for associated_type in &the_trait.associated_types.clone() { + if !bound + .trait_generics + .named_args + .iter() + .any(|(name, _)| name.0.contents == *associated_type.name.as_ref()) + { + // This generic isn't contained in the bound's named arguments, + // so add it by creating a fresh type variable. + let new_generic_id = self.interner.next_type_variable_id(); + let kind = associated_type.type_var.kind(); + let type_var = TypeVariable::unbound(new_generic_id, kind); + + let span = bound.trait_path.span; + let name = format!("<{object} as {trait_name}>::{}", associated_type.name); + let name = Rc::new(name); + let typ = Type::NamedGeneric(type_var.clone(), name.clone()); + let typ = self.interner.push_quoted_type(typ); + let typ = UnresolvedTypeData::Resolved(typ).with_span(span); + let ident = Ident::new(associated_type.name.as_ref().clone(), span); + + bound.trait_generics.named_args.push((ident, typ)); + added_generics.push(ResolvedGeneric { name, span, type_var }); + } + } + } + + added_generics + } + + /// Resolves a trait constraint and adds it to scope as an assumed impl. + /// This second step is necessary to resolve subsequent constraints such + /// as `::Bar: Eq` which may lookup an impl which was assumed + /// by a previous constraint. + fn resolve_trait_constraint( &mut self, constraint: &UnresolvedTraitConstraint, ) -> Option { let typ = self.resolve_type(constraint.typ.clone()); let trait_bound = self.resolve_trait_bound(&constraint.trait_bound)?; + + self.add_trait_bound_to_scope( + constraint.trait_bound.trait_path.span, + &typ, + &trait_bound, + trait_bound.trait_id, + ); + Some(TraitConstraint { typ, trait_bound }) } @@ -765,10 +905,13 @@ impl<'context> Elaborator<'context> { let has_inline_attribute = has_no_predicates_attribute || should_fold; let is_pub_allowed = self.pub_allowed(func, in_contract); self.add_generics(&func.def.generics); + let mut generics = vecmap(&self.generics, |generic| generic.type_var.clone()); + + let new_generics = self.desugar_trait_constraints(&mut func.def.where_clause); + generics.extend(new_generics.into_iter().map(|generic| generic.type_var)); let mut trait_constraints = self.resolve_trait_constraints(&func.def.where_clause); - let mut generics = vecmap(&self.generics, |generic| generic.type_var.clone()); let mut parameters = Vec::new(); let mut parameter_types = Vec::new(); let mut parameter_idents = Vec::new(); @@ -839,6 +982,9 @@ impl<'context> Elaborator<'context> { None }; + // Remove the traits assumed by `resolve_trait_constraints` from scope + self.remove_trait_constraints_from_scope(&trait_constraints); + let meta = FuncMeta { name: name_ident, kind: func.kind, @@ -897,7 +1043,7 @@ impl<'context> Elaborator<'context> { Type::MutableReference(typ) => { self.mark_type_as_used(typ); } - Type::InfixExpr(left, _op, right) => { + Type::InfixExpr(left, _op, right, _) => { self.mark_type_as_used(left); self.mark_type_as_used(right); } @@ -978,20 +1124,46 @@ impl<'context> Elaborator<'context> { } } - fn add_trait_constraints_to_scope(&mut self, func_meta: &FuncMeta) { - for constraint in &func_meta.trait_constraints { + fn add_trait_constraints_to_scope(&mut self, constraints: &[TraitConstraint], span: Span) { + for constraint in constraints { self.add_trait_bound_to_scope( - func_meta, + span, &constraint.typ, &constraint.trait_bound, constraint.trait_bound.trait_id, ); } + + // Also assume `self` implements the current trait if we are inside a trait definition + if let Some(trait_id) = self.current_trait { + let the_trait = self.interner.get_trait(trait_id); + let constraint = the_trait.as_constraint(the_trait.name.span()); + let self_type = + self.self_type.clone().expect("Expected a self type if there's a current trait"); + self.add_trait_bound_to_scope( + span, + &self_type, + &constraint.trait_bound, + constraint.trait_bound.trait_id, + ); + } + } + + fn remove_trait_constraints_from_scope(&mut self, constraints: &[TraitConstraint]) { + for constraint in constraints { + self.interner + .remove_assumed_trait_implementations_for_trait(constraint.trait_bound.trait_id); + } + + // Also remove the assumed trait implementation for `self` if this is a trait definition + if let Some(trait_id) = self.current_trait { + self.interner.remove_assumed_trait_implementations_for_trait(trait_id); + } } fn add_trait_bound_to_scope( &mut self, - func_meta: &FuncMeta, + span: Span, object: &Type, trait_bound: &ResolvedTraitBound, starting_trait_id: TraitId, @@ -1003,7 +1175,6 @@ impl<'context> Elaborator<'context> { if let Some(the_trait) = self.interner.try_get_trait(trait_id) { let trait_name = the_trait.name.to_string(); let typ = object.clone(); - let span = func_meta.location.span; self.push_err(TypeCheckError::UnneededTraitConstraint { trait_name, typ, span }); } } @@ -1020,12 +1191,7 @@ impl<'context> Elaborator<'context> { let parent_trait_bound = self.instantiate_parent_trait_bound(trait_bound, &parent_trait_bound); - self.add_trait_bound_to_scope( - func_meta, - object, - &parent_trait_bound, - starting_trait_id, - ); + self.add_trait_bound_to_scope(span, object, &parent_trait_bound, starting_trait_id); } } } @@ -1229,7 +1395,7 @@ impl<'context> Elaborator<'context> { self.file = unresolved.file_id; let old_generic_count = self.generics.len(); self.add_generics(generics); - self.declare_methods_on_struct(false, unresolved, *span); + self.declare_methods_on_struct(None, unresolved, *span); self.generics.truncate(old_generic_count); } } @@ -1255,15 +1421,22 @@ impl<'context> Elaborator<'context> { self.generics = trait_impl.resolved_generics.clone(); let where_clause = self.resolve_trait_constraints(&trait_impl.where_clause); + self.remove_trait_constraints_from_scope(&where_clause); self.collect_trait_impl_methods(trait_id, trait_impl, &where_clause); let span = trait_impl.object_type.span; - self.declare_methods_on_struct(true, &mut trait_impl.methods, span); + self.declare_methods_on_struct(Some(trait_id), &mut trait_impl.methods, span); + + let trait_visibility = self.interner.get_trait(trait_id).visibility; let methods = trait_impl.methods.function_ids(); for func_id in &methods { self.interner.set_function_trait(*func_id, self_type.clone(), trait_id); + + // A trait impl method has the same visibility as its trait + let modifiers = self.interner.function_modifiers_mut(func_id); + modifiers.visibility = trait_visibility; } let trait_generics = trait_impl.resolved_trait_generics.clone(); @@ -1318,7 +1491,7 @@ impl<'context> Elaborator<'context> { fn declare_methods_on_struct( &mut self, - is_trait_impl: bool, + trait_id: Option, functions: &mut UnresolvedFunctions, span: Span, ) { @@ -1332,7 +1505,7 @@ impl<'context> Elaborator<'context> { let struct_ref = struct_type.borrow(); // `impl`s are only allowed on types defined within the current crate - if !is_trait_impl && struct_ref.id.krate() != self.crate_id { + if trait_id.is_none() && struct_ref.id.krate() != self.crate_id { let type_name = struct_ref.name.to_string(); self.push_err(DefCollectorErrorKind::ForeignImpl { span, type_name }); return; @@ -1349,7 +1522,12 @@ impl<'context> Elaborator<'context> { // object types in each method overlap or not. If they do, we issue an error. // If not, that is specialization which is allowed. let name = method.name_ident().clone(); - if module.declare_function(name, method.def.visibility, *method_id).is_err() { + let result = if let Some(trait_id) = trait_id { + module.declare_trait_function(name, *method_id, trait_id) + } else { + module.declare_function(name, method.def.visibility, *method_id) + }; + if result.is_err() { let existing = module.find_func_with_name(method.name_ident()).expect( "declare_function should only error if there is an existing function", ); @@ -1364,14 +1542,14 @@ impl<'context> Elaborator<'context> { } // Trait impl methods are already declared in NodeInterner::add_trait_implementation - if !is_trait_impl { + if trait_id.is_none() { self.declare_methods(self_type, &function_ids); } // We can define methods on primitive types only if we're in the stdlib - } else if !is_trait_impl && *self_type != Type::Error { + } else if trait_id.is_none() && *self_type != Type::Error { if self.crate_id.is_stdlib() { // Trait impl methods are already declared in NodeInterner::add_trait_implementation - if !is_trait_impl { + if trait_id.is_none() { self.declare_methods(self_type, &function_ids); } } else { @@ -1385,7 +1563,7 @@ impl<'context> Elaborator<'context> { let method_name = self.interner.function_name(method_id).to_owned(); if let Some(first_fn) = - self.interner.add_method(self_type, method_name.clone(), *method_id, false) + self.interner.add_method(self_type, method_name.clone(), *method_id, None) { let error = ResolverError::DuplicateDefinition { name: method_name, @@ -1510,7 +1688,7 @@ impl<'context> Elaborator<'context> { Type::MutableReference(typ) | Type::Array(_, typ) | Type::Slice(typ) => { self.check_type_is_not_more_private_then_item(name, visibility, typ, span); } - Type::InfixExpr(left, _op, right) => { + Type::InfixExpr(left, _op, right, _) => { self.check_type_is_not_more_private_then_item(name, visibility, left, span); self.check_type_is_not_more_private_then_item(name, visibility, right, span); } @@ -1652,20 +1830,13 @@ impl<'context> Elaborator<'context> { self.push_err(ResolverError::MutableGlobal { span }); } - let comptime = let_stmt.comptime; - - let (let_statement, _typ) = if comptime { - self.elaborate_in_comptime_context(|this| this.elaborate_let(let_stmt, Some(global_id))) - } else { - self.elaborate_let(let_stmt, Some(global_id)) - }; + let (let_statement, _typ) = self + .elaborate_in_comptime_context(|this| this.elaborate_let(let_stmt, Some(global_id))); let statement_id = self.interner.get_global(global_id).let_statement; self.interner.replace_statement(statement_id, let_statement); - if comptime { - self.elaborate_comptime_global(global_id); - } + self.elaborate_comptime_global(global_id); if let Some(name) = name { self.interner.register_global(global_id, name, global.visibility, self.module_id()); @@ -1696,7 +1867,17 @@ impl<'context> Elaborator<'context> { self.debug_comptime(location, |interner| value.display(interner).to_string()); - self.interner.get_global_mut(global_id).value = Some(value); + self.interner.get_global_mut(global_id).value = GlobalValue::Resolved(value); + } + } + + /// If the given global is unresolved, elaborate it and return true + fn elaborate_global_if_unresolved(&mut self, global_id: &GlobalId) -> bool { + if let Some(global) = self.unresolved_globals.remove(global_id) { + self.elaborate_global(global); + true + } else { + false } } @@ -1736,6 +1917,17 @@ impl<'context> Elaborator<'context> { self.add_generics(&trait_impl.generics); trait_impl.resolved_generics = self.generics.clone(); + let new_generics = self.desugar_trait_constraints(&mut trait_impl.where_clause); + for new_generic in new_generics { + trait_impl.resolved_generics.push(new_generic.clone()); + self.generics.push(new_generic); + } + + // We need to resolve the where clause before any associated types to be + // able to resolve trait as type syntax, eg. `` in case there + // is a where constraint for `T: Foo`. + let constraints = self.resolve_trait_constraints(&trait_impl.where_clause); + for (_, _, method) in trait_impl.methods.functions.iter_mut() { // Attach any trait constraints on the impl to the function method.def.where_clause.append(&mut trait_impl.where_clause.clone()); @@ -1748,17 +1940,20 @@ impl<'context> Elaborator<'context> { let impl_id = self.interner.next_trait_impl_id(); self.current_trait_impl = Some(impl_id); - // Fetch trait constraints here + let path_span = trait_impl.trait_path.span; let (ordered_generics, named_generics) = trait_impl .trait_id .map(|trait_id| { - self.resolve_type_args(trait_generics, trait_id, trait_impl.trait_path.span) + // Check for missing generics & associated types for the trait being implemented + self.resolve_trait_args_from_trait_impl(trait_generics, trait_id, path_span) }) .unwrap_or_default(); trait_impl.resolved_trait_generics = ordered_generics; self.interner.set_associated_types_for_impl(impl_id, named_generics); + self.remove_trait_constraints_from_scope(&constraints); + let self_type = self.resolve_type(unresolved_type); self.self_type = Some(self_type.clone()); trait_impl.methods.self_type = Some(self_type); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/path_resolution.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/path_resolution.rs index a68991becb7..0d0b153b6b6 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/path_resolution.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/path_resolution.rs @@ -1,7 +1,8 @@ +use iter_extended::vecmap; use noirc_errors::{Location, Span}; -use crate::ast::{Path, PathKind, UnresolvedType}; -use crate::hir::def_map::{ModuleDefId, ModuleId}; +use crate::ast::{Ident, Path, PathKind, UnresolvedType}; +use crate::hir::def_map::{ModuleData, ModuleDefId, ModuleId, PerNs}; use crate::hir::resolution::import::{resolve_path_kind, PathResolutionError}; use crate::hir::resolution::errors::ResolverError; @@ -9,7 +10,7 @@ use crate::hir::resolution::visibility::item_in_module_is_visible; use crate::locations::ReferencesTracker; use crate::node_interner::{FuncId, GlobalId, StructId, TraitId, TypeAliasId}; -use crate::Type; +use crate::{Shared, Type, TypeAlias}; use super::types::SELF_TYPE_NAME; use super::Elaborator; @@ -78,7 +79,7 @@ pub struct Turbofish { /// Any item that can appear before the last segment in a path. #[derive(Debug)] enum IntermediatePathResolutionItem { - Module(ModuleId), + Module, Struct(StructId, Option), TypeAlias(TypeAliasId, Option), Trait(TraitId, Option), @@ -86,6 +87,21 @@ enum IntermediatePathResolutionItem { pub(crate) type PathResolutionResult = Result; +enum StructMethodLookupResult { + /// The method could not be found. There might be trait methods that could be imported, + /// but none of them are. + NotFound(Vec), + /// Found a struct method. + FoundStructMethod(PerNs), + /// Found a trait method and it's currently in scope. + FoundTraitMethod(PerNs, TraitId), + /// There's only one trait method that matches, but it's not in scope + /// (we'll warn about this to avoid introducing a large breaking change) + FoundOneTraitMethodButNotInScope(PerNs, TraitId), + /// Multiple trait method matches were found and they are all in scope. + FoundMultipleTraitMethods(Vec), +} + impl<'context> Elaborator<'context> { pub(super) fn resolve_path_or_error( &mut self, @@ -164,7 +180,7 @@ impl<'context> Elaborator<'context> { let mut current_module_id = starting_module; let mut current_module = self.get_module(starting_module); - let mut intermediate_item = IntermediatePathResolutionItem::Module(current_module_id); + let mut intermediate_item = IntermediatePathResolutionItem::Module; let first_segment = &path.segments.first().expect("ice: could not fetch first segment").ident; @@ -195,7 +211,9 @@ impl<'context> Elaborator<'context> { last_segment.ident.is_self_type_name(), ); - (current_module_id, intermediate_item) = match typ { + let current_module_id_is_struct; + + (current_module_id, current_module_id_is_struct, intermediate_item) = match typ { ModuleDefId::ModuleId(id) => { if last_segment_generics.is_some() { errors.push(PathResolutionError::TurbofishNotAllowedOnItem { @@ -204,10 +222,11 @@ impl<'context> Elaborator<'context> { }); } - (id, IntermediatePathResolutionItem::Module(id)) + (id, false, IntermediatePathResolutionItem::Module) } ModuleDefId::TypeId(id) => ( id.module_id(), + true, IntermediatePathResolutionItem::Struct( id, last_segment_generics.as_ref().map(|generics| Turbofish { @@ -218,22 +237,13 @@ impl<'context> Elaborator<'context> { ), ModuleDefId::TypeAliasId(id) => { let type_alias = self.interner.get_type_alias(id); - let type_alias = type_alias.borrow(); - - let module_id = match &type_alias.typ { - Type::Struct(struct_id, _generics) => struct_id.borrow().id.module_id(), - Type::Error => { - return Err(PathResolutionError::Unresolved(last_ident.clone())); - } - _ => { - // For now we only allow type aliases that point to structs. - // The more general case is captured here: https://github.com/noir-lang/noir/issues/6398 - panic!("Type alias in path not pointing to struct not yet supported") - } + let Some(module_id) = get_type_alias_module_def_id(&type_alias) else { + return Err(PathResolutionError::Unresolved(last_ident.clone())); }; ( module_id, + true, IntermediatePathResolutionItem::TypeAlias( id, last_segment_generics.as_ref().map(|generics| Turbofish { @@ -245,6 +255,7 @@ impl<'context> Elaborator<'context> { } ModuleDefId::TraitId(id) => ( id.0, + false, IntermediatePathResolutionItem::Trait( id, last_segment_generics.as_ref().map(|generics| Turbofish { @@ -273,7 +284,58 @@ impl<'context> Elaborator<'context> { current_module = self.get_module(current_module_id); // Check if namespace - let found_ns = current_module.find_name(current_ident); + let found_ns = if current_module_id_is_struct { + match self.resolve_struct_function(importing_module, current_module, current_ident) + { + StructMethodLookupResult::NotFound(vec) => { + if vec.is_empty() { + return Err(PathResolutionError::Unresolved(current_ident.clone())); + } else { + let traits = vecmap(vec, |trait_id| { + let trait_ = self.interner.get_trait(trait_id); + self.fully_qualified_trait_path(trait_) + }); + return Err( + PathResolutionError::UnresolvedWithPossibleTraitsToImport { + ident: current_ident.clone(), + traits, + }, + ); + } + } + StructMethodLookupResult::FoundStructMethod(per_ns) => per_ns, + StructMethodLookupResult::FoundTraitMethod(per_ns, trait_id) => { + let trait_ = self.interner.get_trait(trait_id); + self.usage_tracker.mark_as_used(importing_module, &trait_.name); + per_ns + } + StructMethodLookupResult::FoundOneTraitMethodButNotInScope( + per_ns, + trait_id, + ) => { + let trait_ = self.interner.get_trait(trait_id); + let trait_name = self.fully_qualified_trait_path(trait_); + errors.push(PathResolutionError::TraitMethodNotInScope { + ident: current_ident.clone(), + trait_name, + }); + per_ns + } + StructMethodLookupResult::FoundMultipleTraitMethods(vec) => { + let traits = vecmap(vec, |trait_id| { + let trait_ = self.interner.get_trait(trait_id); + self.usage_tracker.mark_as_used(importing_module, &trait_.name); + self.fully_qualified_trait_path(trait_) + }); + return Err(PathResolutionError::MultipleTraitsInScope { + ident: current_ident.clone(), + traits, + }); + } + } + } else { + current_module.find_name(current_ident) + }; if found_ns.is_none() { return Err(PathResolutionError::Unresolved(current_ident.clone())); } @@ -317,6 +379,74 @@ impl<'context> Elaborator<'context> { None } } + + fn resolve_struct_function( + &self, + importing_module_id: ModuleId, + current_module: &ModuleData, + ident: &Ident, + ) -> StructMethodLookupResult { + // If the current module is a struct, next we need to find a function for it. + // The function could be in the struct itself, or it could be defined in traits. + let item_scope = current_module.scope(); + let Some(values) = item_scope.values().get(ident) else { + return StructMethodLookupResult::NotFound(vec![]); + }; + + // First search if the function is defined in the struct itself + if let Some(item) = values.get(&None) { + return StructMethodLookupResult::FoundStructMethod(PerNs { + types: None, + values: Some(*item), + }); + } + + // Otherwise, the function could be defined in zero, one or more traits. + let starting_module = self.get_module(importing_module_id); + + // Gather a list of items for which their trait is in scope. + let mut results = Vec::new(); + + for (trait_id, item) in values.iter() { + let trait_id = trait_id.expect("The None option was already considered before"); + let trait_ = self.interner.get_trait(trait_id); + let Some(map) = starting_module.scope().types().get(&trait_.name) else { + continue; + }; + let Some(imported_item) = map.get(&None) else { + continue; + }; + if imported_item.0 == ModuleDefId::TraitId(trait_id) { + results.push((trait_id, item)); + } + } + + if results.is_empty() { + if values.len() == 1 { + // This is the backwards-compatible case where there's a single trait method but it's not in scope + let (trait_id, item) = values.iter().next().expect("Expected an item"); + let trait_id = trait_id.expect("The None option was already considered before"); + let per_ns = PerNs { types: None, values: Some(*item) }; + return StructMethodLookupResult::FoundOneTraitMethodButNotInScope( + per_ns, trait_id, + ); + } else { + let trait_ids = vecmap(values, |(trait_id, _)| { + trait_id.expect("The none option was already considered before") + }); + return StructMethodLookupResult::NotFound(trait_ids); + } + } + + if results.len() > 1 { + let trait_ids = vecmap(results, |(trait_id, _)| trait_id); + return StructMethodLookupResult::FoundMultipleTraitMethods(trait_ids); + } + + let (trait_id, item) = results.remove(0); + let per_ns = PerNs { types: None, values: Some(*item) }; + StructMethodLookupResult::FoundTraitMethod(per_ns, trait_id) + } } fn merge_intermediate_path_resolution_item_with_module_def_id( @@ -330,9 +460,7 @@ fn merge_intermediate_path_resolution_item_with_module_def_id( ModuleDefId::TraitId(trait_id) => PathResolutionItem::Trait(trait_id), ModuleDefId::GlobalId(global_id) => PathResolutionItem::Global(global_id), ModuleDefId::FunctionId(func_id) => match intermediate_item { - IntermediatePathResolutionItem::Module(_) => { - PathResolutionItem::ModuleFunction(func_id) - } + IntermediatePathResolutionItem::Module => PathResolutionItem::ModuleFunction(func_id), IntermediatePathResolutionItem::Struct(struct_id, generics) => { PathResolutionItem::StructFunction(struct_id, generics, func_id) } @@ -345,3 +473,18 @@ fn merge_intermediate_path_resolution_item_with_module_def_id( }, } } + +fn get_type_alias_module_def_id(type_alias: &Shared) -> Option { + let type_alias = type_alias.borrow(); + + match &type_alias.typ { + Type::Struct(struct_id, _generics) => Some(struct_id.borrow().id.module_id()), + Type::Alias(type_alias, _generics) => get_type_alias_module_def_id(type_alias), + Type::Error => None, + _ => { + // For now we only allow type aliases that point to structs. + // The more general case is captured here: https://github.com/noir-lang/noir/issues/6398 + panic!("Type alias in path not pointing to struct not yet supported") + } + } +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/patterns.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/patterns.rs index 3fbdadbbee8..6a672866d7e 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/patterns.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/patterns.rs @@ -604,17 +604,11 @@ impl<'context> Elaborator<'context> { alias_generics }; - // Now instantiate the underlying struct with those generics, the struct might + // Now instantiate the underlying struct or alias with those generics, the struct might // have more generics than those in the alias, like in this example: // // type Alias = Struct; - let typ = type_alias.get_type(&generics); - let Type::Struct(_, generics) = typ else { - // See https://github.com/noir-lang/noir/issues/6398 - panic!("Expected type alias to point to struct") - }; - - generics + get_type_alias_generics(&type_alias, &generics) } PathResolutionItem::TraitFunction(trait_id, Some(generics), _func_id) => { let trait_ = self.interner.get_trait(trait_id); @@ -666,9 +660,7 @@ impl<'context> Elaborator<'context> { self.interner.add_function_reference(func_id, hir_ident.location); } DefinitionKind::Global(global_id) => { - if let Some(global) = self.unresolved_globals.remove(&global_id) { - self.elaborate_global(global); - } + self.elaborate_global_if_unresolved(&global_id); if let Some(current_item) = self.current_item { self.interner.add_global_dependency(current_item, global_id); } @@ -759,7 +751,11 @@ impl<'context> Elaborator<'context> { let function = self.interner.function_meta(&function); for mut constraint in function.trait_constraints.clone() { constraint.apply_bindings(&bindings); - self.push_trait_constraint(constraint, expr_id); + + self.push_trait_constraint( + constraint, expr_id, + false, // This constraint shouldn't lead to choosing a trait impl method + ); } } } @@ -775,7 +771,11 @@ impl<'context> Elaborator<'context> { // Currently only one impl can be selected per expr_id, so this // constraint needs to be pushed after any other constraints so // that monomorphization can resolve this trait method to the correct impl. - self.push_trait_constraint(method.constraint, expr_id); + self.push_trait_constraint( + method.constraint, + expr_id, + true, // this constraint should lead to choosing a trait impl method + ); } } @@ -864,7 +864,7 @@ impl<'context> Elaborator<'context> { let impl_kind = match method { HirMethodReference::FuncId(_) => ImplKind::NotATraitMethod, - HirMethodReference::TraitMethodId(method_id, generics) => { + HirMethodReference::TraitMethodId(method_id, generics, _) => { let mut constraint = self.interner.get_trait(method_id.trait_id).as_constraint(span); constraint.trait_bound.trait_generics = generics; @@ -882,3 +882,14 @@ impl<'context> Elaborator<'context> { (id, typ) } } + +fn get_type_alias_generics(type_alias: &TypeAlias, generics: &[Type]) -> Vec { + let typ = type_alias.get_type(generics); + match typ { + Type::Struct(_, generics) => generics, + Type::Alias(type_alias, generics) => { + get_type_alias_generics(&type_alias.borrow(), &generics) + } + _ => panic!("Expected type alias to point to struct or alias"), + } +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/statements.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/statements.rs index 93009f49071..a01b24c2f0f 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/statements.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/statements.rs @@ -33,6 +33,7 @@ impl<'context> Elaborator<'context> { StatementKind::Constrain(constrain) => self.elaborate_constrain(constrain), StatementKind::Assign(assign) => self.elaborate_assign(assign), StatementKind::For(for_stmt) => self.elaborate_for(for_stmt), + StatementKind::Loop(block) => self.elaborate_loop(block, statement.span), StatementKind::Break => self.elaborate_jump(true, statement.span), StatementKind::Continue => self.elaborate_jump(false, statement.span), StatementKind::Comptime(statement) => self.elaborate_comptime_statement(*statement), @@ -76,6 +77,7 @@ impl<'context> Elaborator<'context> { ) -> (HirStatement, Type) { let expr_span = let_stmt.expression.span; let (expression, expr_type) = self.elaborate_expression(let_stmt.expression); + let type_contains_unspecified = let_stmt.r#type.contains_unspecified(); let annotated_type = self.resolve_inferred_type(let_stmt.r#type); @@ -123,7 +125,9 @@ impl<'context> Elaborator<'context> { let attributes = let_stmt.attributes; let comptime = let_stmt.comptime; - let let_ = HirLetStatement { pattern, r#type, expression, attributes, comptime }; + let is_global_let = let_stmt.is_global_let; + let let_ = + HirLetStatement::new(pattern, r#type, expression, attributes, comptime, is_global_let); (HirStatement::Let(let_), Type::Unit) } @@ -265,6 +269,29 @@ impl<'context> Elaborator<'context> { (statement, Type::Unit) } + pub(super) fn elaborate_loop( + &mut self, + block: Expression, + span: noirc_errors::Span, + ) -> (HirStatement, Type) { + let in_constrained_function = self.in_constrained_function(); + if in_constrained_function { + self.push_err(ResolverError::LoopInConstrainedFn { span }); + } + + self.nested_loops += 1; + self.push_scope(); + + let (block, _block_type) = self.elaborate_expression(block); + + self.pop_scope(); + self.nested_loops -= 1; + + let statement = HirStatement::Loop(block); + + (statement, Type::Unit) + } + fn elaborate_jump(&mut self, is_break: bool, span: noirc_errors::Span) -> (HirStatement, Type) { let in_constrained_function = self.in_constrained_function(); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/traits.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/traits.rs index e1be45927ca..bbc1214de9e 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/traits.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/traits.rs @@ -21,21 +21,31 @@ use crate::{ use super::Elaborator; impl<'context> Elaborator<'context> { - pub fn collect_traits(&mut self, traits: &BTreeMap) { + pub fn collect_traits(&mut self, traits: &mut BTreeMap) { for (trait_id, unresolved_trait) in traits { self.local_module = unresolved_trait.module_id; self.recover_generics(|this| { this.current_trait = Some(*trait_id); + let the_trait = this.interner.get_trait(*trait_id); + let self_typevar = the_trait.self_type_typevar.clone(); + let self_type = Type::TypeVariable(self_typevar.clone()); + this.self_type = Some(self_type.clone()); + let resolved_generics = this.interner.get_trait(*trait_id).generics.clone(); this.add_existing_generics( &unresolved_trait.trait_def.generics, &resolved_generics, ); + let new_generics = + this.desugar_trait_constraints(&mut unresolved_trait.trait_def.where_clause); + this.generics.extend(new_generics); + let where_clause = this.resolve_trait_constraints(&unresolved_trait.trait_def.where_clause); + this.remove_trait_constraints_from_scope(&where_clause); // Each associated type in this trait is also an implicit generic for associated_type in &this.interner.get_trait(*trait_id).associated_types { @@ -48,12 +58,16 @@ impl<'context> Elaborator<'context> { .add_trait_dependency(DependencyId::Trait(bound.trait_id), *trait_id); } + this.interner.update_trait(*trait_id, |trait_def| { + trait_def.set_trait_bounds(resolved_trait_bounds); + trait_def.set_where_clause(where_clause); + trait_def.set_visibility(unresolved_trait.trait_def.visibility); + }); + let methods = this.resolve_trait_methods(*trait_id, unresolved_trait); this.interner.update_trait(*trait_id, |trait_def| { trait_def.set_methods(methods); - trait_def.set_trait_bounds(resolved_trait_bounds); - trait_def.set_where_clause(where_clause); }); }); @@ -94,7 +108,7 @@ impl<'context> Elaborator<'context> { parameters, return_type, where_clause, - body: _, + body, is_unconstrained, visibility: _, is_comptime: _, @@ -103,7 +117,6 @@ impl<'context> Elaborator<'context> { self.recover_generics(|this| { let the_trait = this.interner.get_trait(trait_id); let self_typevar = the_trait.self_type_typevar.clone(); - let self_type = Type::TypeVariable(self_typevar.clone()); let name_span = the_trait.name.span(); this.add_existing_generic( @@ -115,9 +128,12 @@ impl<'context> Elaborator<'context> { span: name_span, }, ); - this.self_type = Some(self_type.clone()); let func_id = unresolved_trait.method_ids[&name.0.contents]; + let mut where_clause = where_clause.to_vec(); + + // Attach any trait constraints on the trait to the function, + where_clause.extend(unresolved_trait.trait_def.where_clause.clone()); this.resolve_trait_function( trait_id, @@ -127,6 +143,8 @@ impl<'context> Elaborator<'context> { parameters, return_type, where_clause, + body, + unresolved_trait.trait_def.visibility, func_id, ); @@ -188,29 +206,45 @@ impl<'context> Elaborator<'context> { generics: &UnresolvedGenerics, parameters: &[(Ident, UnresolvedType)], return_type: &FunctionReturnType, - where_clause: &[UnresolvedTraitConstraint], + where_clause: Vec, + body: &Option, + trait_visibility: ItemVisibility, func_id: FuncId, ) { let old_generic_count = self.generics.len(); self.scopes.start_function(); - let kind = FunctionKind::Normal; + let has_body = body.is_some(); + + let body = match body { + Some(body) => body.clone(), + None => BlockExpression { statements: Vec::new() }, + }; + let kind = + if has_body { FunctionKind::Normal } else { FunctionKind::TraitFunctionWithoutBody }; let mut def = FunctionDefinition::normal( name, is_unconstrained, generics, parameters, - &BlockExpression { statements: Vec::new() }, + body, where_clause, return_type, ); - // Trait functions are always public - def.visibility = ItemVisibility::Public; + + // Trait functions always have the same visibility as the trait they are in + def.visibility = trait_visibility; let mut function = NoirFunction { kind, def }; self.define_function_meta(&mut function, func_id, Some(trait_id)); - self.elaborate_function(func_id); + + // Here we elaborate functions without a body, mainly to check the arguments and return types. + // Later on we'll elaborate functions with a body by fully type-checking them. + if !has_body { + self.elaborate_function(func_id); + } + let _ = self.scopes.end_function(); // Don't check the scope tree for unused variables, they can't be used in a declaration anyway. self.generics.truncate(old_generic_count); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/types.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/types.rs index 2e4809f3511..a1b63910a3e 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/types.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/types.rs @@ -1,6 +1,6 @@ use std::{borrow::Cow, rc::Rc}; -use acvm::acir::AcirField; +use im::HashSet; use iter_extended::vecmap; use noirc_errors::{Location, Span}; use rustc_hash::FxHashMap as HashMap; @@ -12,8 +12,8 @@ use crate::{ UnresolvedTypeData, UnresolvedTypeExpression, WILDCARD_TYPE, }, hir::{ - comptime::{Interpreter, Value}, def_collector::dc_crate::CompilationError, + def_map::{fully_qualified_module_path, ModuleDefId}, resolution::{errors::ResolverError, import::PathResolutionError}, type_check::{ generics::{Generic, TraitGenerics}, @@ -30,14 +30,14 @@ use crate::{ traits::{NamedType, ResolvedTraitBound, Trait, TraitConstraint}, }, node_interner::{ - DefinitionKind, DependencyId, ExprId, GlobalId, ImplSearchErrorKind, NodeInterner, TraitId, + DependencyId, ExprId, FuncId, GlobalValue, ImplSearchErrorKind, NodeInterner, TraitId, TraitImplKind, TraitMethodId, }, token::SecondaryAttribute, Generics, Kind, ResolvedGeneric, Type, TypeBinding, TypeBindings, UnificationError, }; -use super::{lints, path_resolution::PathResolutionItem, Elaborator}; +use super::{lints, path_resolution::PathResolutionItem, Elaborator, UnsafeBlockStatus}; pub const SELF_TYPE_NAME: &str = "Self"; @@ -310,11 +310,32 @@ impl<'context> Elaborator<'context> { } } + /// Identical to `resolve_type_args` but does not allow + /// associated types to be elided since trait impls must specify them. + pub(super) fn resolve_trait_args_from_trait_impl( + &mut self, + args: GenericTypeArgs, + item: TraitId, + span: Span, + ) -> (Vec, Vec) { + self.resolve_type_args_inner(args, item, span, false) + } + pub(super) fn resolve_type_args( + &mut self, + args: GenericTypeArgs, + item: impl Generic, + span: Span, + ) -> (Vec, Vec) { + self.resolve_type_args_inner(args, item, span, true) + } + + pub(super) fn resolve_type_args_inner( &mut self, mut args: GenericTypeArgs, item: impl Generic, span: Span, + allow_implicit_named_args: bool, ) -> (Vec, Vec) { let expected_kinds = item.generics(self.interner); @@ -336,7 +357,12 @@ impl<'context> Elaborator<'context> { let mut associated = Vec::new(); if item.accepts_named_type_args() { - associated = self.resolve_associated_type_args(args.named_args, item, span); + associated = self.resolve_associated_type_args( + args.named_args, + item, + span, + allow_implicit_named_args, + ); } else if !args.named_args.is_empty() { let item_kind = item.item_kind(); self.push_err(ResolverError::NamedTypeArgs { span, item_kind }); @@ -350,6 +376,7 @@ impl<'context> Elaborator<'context> { args: Vec<(Ident, UnresolvedType)>, item: impl Generic, span: Span, + allow_implicit_named_args: bool, ) -> Vec { let mut seen_args = HashMap::default(); let mut required_args = item.named_generics(self.interner); @@ -379,11 +406,19 @@ impl<'context> Elaborator<'context> { resolved.push(NamedType { name, typ }); } - // Anything that hasn't been removed yet is missing + // Anything that hasn't been removed yet is missing. + // Fill it in to avoid a panic if we allow named args to be elided, otherwise error. for generic in required_args { - let item = item.item_name(self.interner); let name = generic.name.clone(); - self.push_err(TypeCheckError::MissingNamedTypeArg { item, span, name }); + + if allow_implicit_named_args { + let name = Ident::new(name.as_ref().clone(), span); + let typ = self.interner.next_type_variable(); + resolved.push(NamedType { name, typ }); + } else { + let item = item.item_name(self.interner); + self.push_err(TypeCheckError::MissingNamedTypeArg { item, span, name }); + } } resolved @@ -415,17 +450,41 @@ impl<'context> Elaborator<'context> { .map(|let_statement| Kind::numeric(let_statement.r#type)) .unwrap_or(Kind::u32()); - // TODO(https://github.com/noir-lang/noir/issues/6238): - // support non-u32 generics here - if !kind.unifies(&Kind::u32()) { - let error = TypeCheckError::EvaluatedGlobalIsntU32 { - expected_kind: Kind::u32().to_string(), - expr_kind: kind.to_string(), - expr_span: path.span(), - }; - self.push_err(error); - } - Some(Type::Constant(self.eval_global_as_array_length(id, path).into(), kind)) + let Some(stmt) = self.interner.get_global_let_statement(id) else { + if self.elaborate_global_if_unresolved(&id) { + return self.lookup_generic_or_global_type(path); + } else { + let path = path.clone(); + self.push_err(ResolverError::NoSuchNumericTypeVariable { path }); + return None; + } + }; + + let rhs = stmt.expression; + let span = self.interner.expr_span(&rhs); + + let GlobalValue::Resolved(global_value) = &self.interner.get_global(id).value + else { + self.push_err(ResolverError::UnevaluatedGlobalType { span }); + return None; + }; + + let Some(global_value) = global_value.to_field_element() else { + let global_value = global_value.clone(); + if global_value.is_integral() { + self.push_err(ResolverError::NegativeGlobalType { span, global_value }); + } else { + self.push_err(ResolverError::NonIntegralGlobalType { span, global_value }); + } + return None; + }; + + let Ok(global_value) = kind.ensure_value_fits(global_value, span) else { + self.push_err(ResolverError::GlobalLargerThanKind { span, global_value, kind }); + return None; + }; + + Some(Type::Constant(global_value, kind)) } _ => None, } @@ -476,7 +535,7 @@ impl<'context> Elaborator<'context> { } } (lhs, rhs) => { - let infix = Type::InfixExpr(Box::new(lhs), op, Box::new(rhs)); + let infix = Type::infix_expr(Box::new(lhs), op, Box::new(rhs)); Type::CheckedCast { from: Box::new(infix.clone()), to: Box::new(infix) } .canonicalize() } @@ -544,12 +603,17 @@ impl<'context> Elaborator<'context> { } // this resolves Self::some_static_method, inside an impl block (where we don't have a concrete self_type) + // or inside a trait default method. // // Returns the trait method, trait constraint, and whether the impl is assumed to exist by a where clause or not // E.g. `t.method()` with `where T: Foo` in scope will return `(Foo::method, T, vec![Bar])` fn resolve_trait_static_method_by_self(&mut self, path: &Path) -> Option { - let trait_impl = self.current_trait_impl?; - let trait_id = self.interner.try_get_trait_implementation(trait_impl)?.borrow().trait_id; + let trait_id = if let Some(current_trait) = self.current_trait { + current_trait + } else { + let trait_impl = self.current_trait_impl?; + self.interner.try_get_trait_implementation(trait_impl)?.borrow().trait_id + }; if path.kind == PathKind::Plain && path.segments.len() == 2 { let name = &path.segments[0].ident.0.contents; @@ -633,31 +697,6 @@ impl<'context> Elaborator<'context> { .or_else(|| self.resolve_trait_method_by_named_generic(path)) } - fn eval_global_as_array_length(&mut self, global_id: GlobalId, path: &Path) -> u32 { - let Some(stmt) = self.interner.get_global_let_statement(global_id) else { - if let Some(global) = self.unresolved_globals.remove(&global_id) { - self.elaborate_global(global); - return self.eval_global_as_array_length(global_id, path); - } else { - let path = path.clone(); - self.push_err(ResolverError::NoSuchNumericTypeVariable { path }); - return 0; - } - }; - - let length = stmt.expression; - let span = self.interner.expr_span(&length); - let result = try_eval_array_length_id(self.interner, length, span); - - match result.map(|length| length.try_into()) { - Ok(Ok(length_value)) => return length_value, - Ok(Err(_cast_err)) => self.push_err(ResolverError::IntegerTooLarge { span }), - Err(Some(error)) => self.push_err(error), - Err(None) => (), - } - 0 - } - pub(super) fn unify( &mut self, actual: &Type, @@ -1307,7 +1346,7 @@ impl<'context> Elaborator<'context> { } } - pub(super) fn lookup_method( + pub(crate) fn lookup_method( &mut self, object_type: &Type, method_name: &str, @@ -1315,33 +1354,6 @@ impl<'context> Elaborator<'context> { has_self_arg: bool, ) -> Option { match object_type.follow_bindings() { - Type::Struct(typ, _args) => { - let id = typ.borrow().id; - match self.interner.lookup_method(object_type, id, method_name, false, has_self_arg) - { - Some(method_id) => Some(HirMethodReference::FuncId(method_id)), - None => { - let has_field_with_function_type = - typ.borrow().get_fields_as_written().into_iter().any(|field| { - field.name.0.contents == method_name && field.typ.is_function() - }); - if has_field_with_function_type { - self.push_err(TypeCheckError::CannotInvokeStructFieldFunctionType { - method_name: method_name.to_string(), - object_type: object_type.clone(), - span, - }); - } else { - self.push_err(TypeCheckError::UnresolvedMethodCall { - method_name: method_name.to_string(), - object_type: object_type.clone(), - span, - }); - } - None - } - } - } // TODO: We should allow method calls on `impl Trait`s eventually. // For now it is fine since they are only allowed on return types. Type::TraitAsType(..) => { @@ -1357,11 +1369,9 @@ impl<'context> Elaborator<'context> { } // Mutable references to another type should resolve to methods of their element type. // This may be a struct or a primitive type. - Type::MutableReference(element) => self - .interner - .lookup_primitive_trait_method_mut(element.as_ref(), method_name, has_self_arg) - .map(HirMethodReference::FuncId) - .or_else(|| self.lookup_method(&element, method_name, span, has_self_arg)), + Type::MutableReference(element) => { + self.lookup_method(&element, method_name, span, has_self_arg) + } // If we fail to resolve the object to a struct type, we have no way of type // checking its arguments as we can't even resolve the name of the function @@ -1373,17 +1383,171 @@ impl<'context> Elaborator<'context> { None } - other => match self.interner.lookup_primitive_method(&other, method_name, has_self_arg) - { - Some(method_id) => Some(HirMethodReference::FuncId(method_id)), - None => { - // It could be that this type is a composite type that is bound to a trait, - // for example `x: (T, U) ... where (T, U): SomeTrait` - // (so this case is a generalization of the NamedGeneric case) - self.lookup_method_in_trait_constraints(object_type, method_name, span) + other => { + self.lookup_struct_or_primitive_method(&other, method_name, span, has_self_arg) + } + } + } + + fn lookup_struct_or_primitive_method( + &mut self, + object_type: &Type, + method_name: &str, + span: Span, + has_self_arg: bool, + ) -> Option { + // First search in the type methods. If there is one, that's the one. + if let Some(method_id) = + self.interner.lookup_direct_method(object_type, method_name, has_self_arg) + { + return Some(HirMethodReference::FuncId(method_id)); + } + + // Next lookup all matching trait methods. + let trait_methods = + self.interner.lookup_trait_methods(object_type, method_name, has_self_arg); + + // If there's at least one matching trait method we need to see if only one is in scope. + if !trait_methods.is_empty() { + return self.return_trait_method_in_scope(&trait_methods, method_name, span); + } + + // If we couldn't find any trait methods, search in + // impls for all types `T`, e.g. `impl Foo for T` + let generic_methods = + self.interner.lookup_generic_methods(object_type, method_name, has_self_arg); + if !generic_methods.is_empty() { + return self.return_trait_method_in_scope(&generic_methods, method_name, span); + } + + if let Type::Struct(struct_type, _) = object_type { + let has_field_with_function_type = struct_type + .borrow() + .get_fields_as_written() + .into_iter() + .any(|field| field.name.0.contents == method_name && field.typ.is_function()); + if has_field_with_function_type { + self.push_err(TypeCheckError::CannotInvokeStructFieldFunctionType { + method_name: method_name.to_string(), + object_type: object_type.clone(), + span, + }); + } else { + self.push_err(TypeCheckError::UnresolvedMethodCall { + method_name: method_name.to_string(), + object_type: object_type.clone(), + span, + }); + } + None + } else { + // It could be that this type is a composite type that is bound to a trait, + // for example `x: (T, U) ... where (T, U): SomeTrait` + // (so this case is a generalization of the NamedGeneric case) + self.lookup_method_in_trait_constraints(object_type, method_name, span) + } + } + + /// Given a list of functions and the trait they belong to, returns the one function + /// that is in scope. + fn return_trait_method_in_scope( + &mut self, + trait_methods: &[(FuncId, TraitId)], + method_name: &str, + span: Span, + ) -> Option { + let module_id = self.module_id(); + let module_data = self.get_module(module_id); + + // Only keep unique trait IDs: multiple trait methods might come from the same trait + // but implemented with different generics (like `Convert` and `Convert`). + let traits: HashSet = + trait_methods.iter().map(|(_, trait_id)| *trait_id).collect(); + + let traits_in_scope: Vec<_> = traits + .iter() + .filter_map(|trait_id| { + let trait_ = self.interner.get_trait(*trait_id); + let trait_name = &trait_.name; + let map = module_data.scope().types().get(trait_name)?; + let imported_item = map.get(&None)?; + if imported_item.0 == ModuleDefId::TraitId(*trait_id) { + Some((*trait_id, trait_name)) + } else { + None } - }, + }) + .collect(); + + for (_, trait_name) in &traits_in_scope { + self.usage_tracker.mark_as_used(module_id, trait_name); + } + + if traits_in_scope.is_empty() { + if traits.len() == 1 { + // This is the backwards-compatible case where there's a single trait but it's not in scope + let trait_id = *traits.iter().next().unwrap(); + let trait_ = self.interner.get_trait(trait_id); + let trait_name = self.fully_qualified_trait_path(trait_); + + self.push_err(PathResolutionError::TraitMethodNotInScope { + ident: Ident::new(method_name.into(), span), + trait_name, + }); + + return Some(self.trait_hir_method_reference( + trait_id, + trait_methods, + method_name, + span, + )); + } else { + let traits = vecmap(traits, |trait_id| { + let trait_ = self.interner.get_trait(trait_id); + self.fully_qualified_trait_path(trait_) + }); + self.push_err(PathResolutionError::UnresolvedWithPossibleTraitsToImport { + ident: Ident::new(method_name.into(), span), + traits, + }); + return None; + } + } + + if traits_in_scope.len() > 1 { + let traits = vecmap(traits, |trait_id| { + let trait_ = self.interner.get_trait(trait_id); + self.fully_qualified_trait_path(trait_) + }); + self.push_err(PathResolutionError::MultipleTraitsInScope { + ident: Ident::new(method_name.into(), span), + traits, + }); + return None; + } + + let trait_id = traits_in_scope[0].0; + Some(self.trait_hir_method_reference(trait_id, trait_methods, method_name, span)) + } + + fn trait_hir_method_reference( + &self, + trait_id: TraitId, + trait_methods: &[(FuncId, TraitId)], + method_name: &str, + span: Span, + ) -> HirMethodReference { + // If we find a single trait impl method, return it so we don't have to later determine the impl + if trait_methods.len() == 1 { + let (func_id, _) = trait_methods[0]; + return HirMethodReference::FuncId(func_id); } + + // Return a TraitMethodId with unbound generics. These will later be bound by the type-checker. + let trait_ = self.interner.get_trait(trait_id); + let generics = trait_.as_constraint(span).trait_bound.trait_generics; + let trait_method_id = trait_.find_method(method_name).unwrap(); + HirMethodReference::TraitMethodId(trait_method_id, generics, false) } fn lookup_method_in_trait_constraints( @@ -1394,10 +1558,29 @@ impl<'context> Elaborator<'context> { ) -> Option { let func_id = match self.current_item { Some(DependencyId::Function(id)) => id, - _ => panic!("unexpected method outside a function"), + _ => panic!("unexpected method outside a function: {method_name}"), }; let func_meta = self.interner.function_meta(&func_id); + // If inside a trait method, check if it's a method on `self` + if let Some(trait_id) = func_meta.trait_id { + if Some(object_type) == self.self_type.as_ref() { + let the_trait = self.interner.get_trait(trait_id); + let constraint = the_trait.as_constraint(the_trait.name.span()); + if let Some(HirMethodReference::TraitMethodId(method_id, generics, _)) = self + .lookup_method_in_trait( + the_trait, + method_name, + &constraint.trait_bound, + the_trait.id, + ) + { + // If it is, it's an assumed trait + return Some(HirMethodReference::TraitMethodId(method_id, generics, true)); + } + } + } + for constraint in &func_meta.trait_constraints { if *object_type == constraint.typ { if let Some(the_trait) = @@ -1435,6 +1618,7 @@ impl<'context> Elaborator<'context> { return Some(HirMethodReference::TraitMethodId( trait_method, trait_bound.trait_generics.clone(), + false, )); } @@ -1486,8 +1670,14 @@ impl<'context> Elaborator<'context> { func_type_is_unconstrained || self.is_unconstrained_call(call.func); let crossing_runtime_boundary = is_current_func_constrained && is_unconstrained_call; if crossing_runtime_boundary { - if !self.in_unsafe_block { - self.push_err(TypeCheckError::Unsafe { span }); + match self.unsafe_block_status { + UnsafeBlockStatus::NotInUnsafeBlock => { + self.push_err(TypeCheckError::Unsafe { span }); + } + UnsafeBlockStatus::InUnsafeBlockWithoutUnconstrainedCalls => { + self.unsafe_block_status = UnsafeBlockStatus::InUnsafeBlockWithConstrainedCalls; + } + UnsafeBlockStatus::InUnsafeBlockWithConstrainedCalls => (), } if let Some(called_func_id) = self.interner.lookup_function_from_expr(&call.func) { @@ -1655,6 +1845,7 @@ impl<'context> Elaborator<'context> { (expr_span, empty_function) } + #[allow(clippy::too_many_arguments)] pub fn verify_trait_constraint( &mut self, object_type: &Type, @@ -1662,6 +1853,7 @@ impl<'context> Elaborator<'context> { trait_generics: &[Type], associated_types: &[NamedType], function_ident_id: ExprId, + select_impl: bool, span: Span, ) { match self.interner.lookup_trait_implementation( @@ -1671,7 +1863,9 @@ impl<'context> Elaborator<'context> { associated_types, ) { Ok(impl_kind) => { - self.interner.select_impl_for_expression(function_ident_id, impl_kind); + if select_impl { + self.interner.select_impl_for_expression(function_ident_id, impl_kind); + } } Err(error) => self.push_trait_constraint_error(object_type, error, span), } @@ -1745,10 +1939,15 @@ impl<'context> Elaborator<'context> { /// Push a trait constraint into the current FunctionContext to be solved if needed /// at the end of the earlier of either the current function or the current comptime scope. - pub fn push_trait_constraint(&mut self, constraint: TraitConstraint, expr_id: ExprId) { + pub fn push_trait_constraint( + &mut self, + constraint: TraitConstraint, + expr_id: ExprId, + select_impl: bool, + ) { let context = self.function_context.last_mut(); let context = context.expect("The function_context stack should always be non-empty"); - context.trait_constraints.push((constraint, expr_id)); + context.trait_constraints.push((constraint, expr_id, select_impl)); } pub fn bind_generics_from_trait_constraint( @@ -1795,6 +1994,10 @@ impl<'context> Elaborator<'context> { ..*parent_trait_bound } } + + pub(crate) fn fully_qualified_trait_path(&self, trait_: &Trait) -> String { + fully_qualified_module_path(self.def_maps, self.crate_graph, &trait_.crate_id, trait_.id.0) + } } pub(crate) fn bind_ordered_generics( @@ -1814,7 +2017,7 @@ pub(crate) fn bind_named_generics( args: &[NamedType], bindings: &mut TypeBindings, ) { - assert_eq!(params.len(), args.len()); + assert!(args.len() <= params.len()); for arg in args { let i = params @@ -1825,6 +2028,10 @@ pub(crate) fn bind_named_generics( let param = params.swap_remove(i); bind_generic(¶m, &arg.typ, bindings); } + + for unbound_param in params { + bind_generic(&unbound_param, &Type::Error, bindings); + } } fn bind_generic(param: &ResolvedGeneric, arg: &Type, bindings: &mut TypeBindings) { @@ -1834,88 +2041,6 @@ fn bind_generic(param: &ResolvedGeneric, arg: &Type, bindings: &mut TypeBindings } } -pub fn try_eval_array_length_id( - interner: &NodeInterner, - rhs: ExprId, - span: Span, -) -> Result> { - // Arbitrary amount of recursive calls to try before giving up - let fuel = 100; - try_eval_array_length_id_with_fuel(interner, rhs, span, fuel) -} - -fn try_eval_array_length_id_with_fuel( - interner: &NodeInterner, - rhs: ExprId, - span: Span, - fuel: u32, -) -> Result> { - if fuel == 0 { - // If we reach here, it is likely from evaluating cyclic globals. We expect an error to - // be issued for them after name resolution so issue no error now. - return Err(None); - } - - match interner.expression(&rhs) { - HirExpression::Literal(HirLiteral::Integer(int, false)) => { - int.try_into_u128().ok_or(Some(ResolverError::IntegerTooLarge { span })) - } - HirExpression::Ident(ident, _) => { - if let Some(definition) = interner.try_definition(ident.id) { - match definition.kind { - DefinitionKind::Global(global_id) => { - let let_statement = interner.get_global_let_statement(global_id); - if let Some(let_statement) = let_statement { - let expression = let_statement.expression; - try_eval_array_length_id_with_fuel(interner, expression, span, fuel - 1) - } else { - Err(Some(ResolverError::InvalidArrayLengthExpr { span })) - } - } - _ => Err(Some(ResolverError::InvalidArrayLengthExpr { span })), - } - } else { - Err(Some(ResolverError::InvalidArrayLengthExpr { span })) - } - } - HirExpression::Infix(infix) => { - let lhs = try_eval_array_length_id_with_fuel(interner, infix.lhs, span, fuel - 1)?; - let rhs = try_eval_array_length_id_with_fuel(interner, infix.rhs, span, fuel - 1)?; - - match infix.operator.kind { - BinaryOpKind::Add => Ok(lhs + rhs), - BinaryOpKind::Subtract => Ok(lhs - rhs), - BinaryOpKind::Multiply => Ok(lhs * rhs), - BinaryOpKind::Divide => Ok(lhs / rhs), - BinaryOpKind::Equal => Ok((lhs == rhs) as u128), - BinaryOpKind::NotEqual => Ok((lhs != rhs) as u128), - BinaryOpKind::Less => Ok((lhs < rhs) as u128), - BinaryOpKind::LessEqual => Ok((lhs <= rhs) as u128), - BinaryOpKind::Greater => Ok((lhs > rhs) as u128), - BinaryOpKind::GreaterEqual => Ok((lhs >= rhs) as u128), - BinaryOpKind::And => Ok(lhs & rhs), - BinaryOpKind::Or => Ok(lhs | rhs), - BinaryOpKind::Xor => Ok(lhs ^ rhs), - BinaryOpKind::ShiftRight => Ok(lhs >> rhs), - BinaryOpKind::ShiftLeft => Ok(lhs << rhs), - BinaryOpKind::Modulo => Ok(lhs % rhs), - } - } - HirExpression::Cast(cast) => { - let lhs = try_eval_array_length_id_with_fuel(interner, cast.lhs, span, fuel - 1)?; - let lhs_value = Value::Field(lhs.into()); - let evaluated_value = - Interpreter::evaluate_cast_one_step(&cast, rhs, lhs_value, interner) - .map_err(|error| Some(ResolverError::ArrayLengthInterpreter { error }))?; - - evaluated_value - .to_u128() - .ok_or_else(|| Some(ResolverError::InvalidArrayLengthExpr { span })) - } - _other => Err(Some(ResolverError::InvalidArrayLengthExpr { span })), - } -} - /// Gives an error if a user tries to create a mutable reference /// to an immutable variable. fn verify_mutable_reference(interner: &NodeInterner, rhs: ExprId) -> Result<(), ResolverError> { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/graph/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/graph/mod.rs index 094594a50ac..c007d6792bd 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/graph/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/graph/mod.rs @@ -113,6 +113,41 @@ pub struct CrateGraph { arena: FxHashMap, } +impl CrateGraph { + /// Tries to find the requested crate in the current one's dependencies, + /// otherwise walks down the crate dependency graph from crate_id until we reach it. + /// This is needed in case a library (lib1) re-export a structure defined in another library (lib2) + /// In that case, we will get [lib1,lib2] when looking for a struct defined in lib2, + /// re-exported by lib1 and used by the main crate. + /// Returns the path from crate_id to target_crate_id + pub(crate) fn find_dependencies( + &self, + crate_id: &CrateId, + target_crate_id: &CrateId, + ) -> Option> { + self[crate_id] + .dependencies + .iter() + .find_map(|dep| { + if &dep.crate_id == target_crate_id { + Some(vec![dep.name.to_string()]) + } else { + None + } + }) + .or_else(|| { + self[crate_id].dependencies.iter().find_map(|dep| { + if let Some(mut path) = self.find_dependencies(&dep.crate_id, target_crate_id) { + path.insert(0, dep.name.to_string()); + Some(path) + } else { + None + } + }) + }) + } +} + /// List of characters that are not allowed in a crate name /// For example, Hyphen(-) is disallowed as it is similar to underscore(_) /// and we do not want names that differ by a hyphen diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/display.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/display.rs index 29d1448f07e..ccdfdf00e72 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/display.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/display.rs @@ -732,6 +732,9 @@ fn remove_interned_in_statement_kind( block: remove_interned_in_expression(interner, for_loop.block), ..for_loop }), + StatementKind::Loop(block) => { + StatementKind::Loop(remove_interned_in_expression(interner, block)) + } StatementKind::Comptime(statement) => { StatementKind::Comptime(Box::new(remove_interned_in_statement(interner, *statement))) } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/errors.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/errors.rs index 3df20b39209..e9a615f2c59 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/errors.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/errors.rs @@ -243,6 +243,12 @@ pub enum InterpreterError { CannotInterpretFormatStringWithErrors { location: Location, }, + GlobalsDependencyCycle { + location: Location, + }, + LoopHaltedForUiResponsiveness { + location: Location, + }, // These cases are not errors, they are just used to prevent us from running more code // until the loop can be resumed properly. These cases will never be displayed to users. @@ -319,7 +325,9 @@ impl InterpreterError { | InterpreterError::CannotResolveExpression { location, .. } | InterpreterError::CannotSetFunctionBody { location, .. } | InterpreterError::UnknownArrayLength { location, .. } - | InterpreterError::CannotInterpretFormatStringWithErrors { location } => *location, + | InterpreterError::CannotInterpretFormatStringWithErrors { location } + | InterpreterError::GlobalsDependencyCycle { location } + | InterpreterError::LoopHaltedForUiResponsiveness { location } => *location, InterpreterError::FailedToParseMacro { error, file, .. } => { Location::new(error.span(), *file) @@ -674,6 +682,18 @@ impl<'a> From<&'a InterpreterError> for CustomDiagnostic { "Some of the variables to interpolate could not be evaluated".to_string(); CustomDiagnostic::simple_error(msg, secondary, location.span) } + InterpreterError::GlobalsDependencyCycle { location } => { + let msg = "This global recursively depends on itself".to_string(); + let secondary = String::new(); + CustomDiagnostic::simple_error(msg, secondary, location.span) + } + InterpreterError::LoopHaltedForUiResponsiveness { location } => { + let msg = "This loop took too much time to execute so it was halted for UI responsiveness" + .to_string(); + let secondary = + "This error doesn't happen in normal executions of `nargo`".to_string(); + CustomDiagnostic::simple_warning(msg, secondary, location.span) + } } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/hir_to_display_ast.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/hir_to_display_ast.rs index 9338c0fc37f..71a462c9066 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/hir_to_display_ast.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/hir_to_display_ast.rs @@ -59,6 +59,7 @@ impl HirStatement { block: for_stmt.block.to_display_ast(interner), span, }), + HirStatement::Loop(block) => StatementKind::Loop(block.to_display_ast(interner)), HirStatement::Break => StatementKind::Break, HirStatement::Continue => StatementKind::Continue, HirStatement::Expression(expr) => { @@ -359,7 +360,7 @@ impl Type { Type::Constant(..) => panic!("Type::Constant where a type was expected: {self:?}"), Type::Quoted(quoted_type) => UnresolvedTypeData::Quoted(*quoted_type), Type::Error => UnresolvedTypeData::Error, - Type::InfixExpr(lhs, op, rhs) => { + Type::InfixExpr(lhs, op, rhs, _) => { let lhs = Box::new(lhs.to_type_expression()); let rhs = Box::new(rhs.to_type_expression()); let span = Span::default(); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter.rs index dfa55a9d79b..d48a27c4181 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter.rs @@ -20,6 +20,7 @@ use crate::monomorphization::{ perform_impl_bindings, perform_instantiation_bindings, resolve_trait_method, undo_instantiation_bindings, }; +use crate::node_interner::GlobalValue; use crate::token::{FmtStrFragment, Tokens}; use crate::TypeVariable; use crate::{ @@ -66,6 +67,9 @@ pub struct Interpreter<'local, 'interner> { /// Stateful bigint calculator. bigint_solver: BigIntSolverWithId, + + /// Use pedantic ACVM solving + pedantic_solving: bool, } #[allow(unused)] @@ -74,14 +78,17 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { elaborator: &'local mut Elaborator<'interner>, crate_id: CrateId, current_function: Option, + pedantic_solving: bool, ) -> Self { + let bigint_solver = BigIntSolverWithId::with_pedantic_solving(pedantic_solving); Self { elaborator, crate_id, current_function, bound_generics: Vec::new(), in_loop: false, - bigint_solver: BigIntSolverWithId::default(), + bigint_solver, + pedantic_solving, } } @@ -568,26 +575,39 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { DefinitionKind::Local(_) => self.lookup(&ident), DefinitionKind::Global(global_id) => { // Avoid resetting the value if it is already known - if let Some(value) = &self.elaborator.interner.get_global(*global_id).value { - Ok(value.clone()) - } else { - let global_id = *global_id; - let crate_of_global = self.elaborator.interner.get_global(global_id).crate_id; - let let_ = - self.elaborator.interner.get_global_let_statement(global_id).ok_or_else( - || { + let global_id = *global_id; + let global_info = self.elaborator.interner.get_global(global_id); + let global_crate_id = global_info.crate_id; + match &global_info.value { + GlobalValue::Resolved(value) => Ok(value.clone()), + GlobalValue::Resolving => { + // Note that the error we issue here isn't very informative (it doesn't include the actual cycle) + // but the general dependency cycle detector will give a better error later on during compilation. + let location = self.elaborator.interner.expr_location(&id); + Err(InterpreterError::GlobalsDependencyCycle { location }) + } + GlobalValue::Unresolved => { + let let_ = self + .elaborator + .interner + .get_global_let_statement(global_id) + .ok_or_else(|| { let location = self.elaborator.interner.expr_location(&id); InterpreterError::VariableNotInScope { location } - }, - )?; + })?; - if let_.comptime || crate_of_global != self.crate_id { - self.evaluate_let(let_.clone())?; - } + self.elaborator.interner.get_global_mut(global_id).value = + GlobalValue::Resolving; + + if let_.runs_comptime() || global_crate_id != self.crate_id { + self.evaluate_let(let_.clone())?; + } - let value = self.lookup(&ident)?; - self.elaborator.interner.get_global_mut(global_id).value = Some(value.clone()); - Ok(value) + let value = self.lookup(&ident)?; + self.elaborator.interner.get_global_mut(global_id).value = + GlobalValue::Resolved(value.clone()); + Ok(value) + } } } DefinitionKind::NumericGeneric(type_variable, numeric_typ) => { @@ -1308,7 +1328,7 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { let bindings = unwrap_rc(bindings); let mut result = self.call_function(function_id, arguments, bindings, location)?; if call.is_macro_call { - let expr = result.into_expression(self.elaborator.interner, location)?; + let expr = result.into_expression(self.elaborator, location)?; let expr = self.elaborate_in_function(self.current_function, |elaborator| { elaborator.elaborate_expression(expr).0 }); @@ -1359,17 +1379,10 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { let typ = object.get_type().follow_bindings(); let method_name = &call.method.0.contents; - // TODO: Traits - let method = match &typ { - Type::Struct(struct_def, _) => self.elaborator.interner.lookup_method( - &typ, - struct_def.borrow().id, - method_name, - false, - true, - ), - _ => self.elaborator.interner.lookup_primitive_method(&typ, method_name, true), - }; + let method = self + .elaborator + .lookup_method(&typ, method_name, location.span, true) + .and_then(|method| method.func_id(self.elaborator.interner)); if let Some(method) = method { self.call_function(method, arguments, TypeBindings::new(), location) @@ -1380,13 +1393,19 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { fn evaluate_cast(&mut self, cast: &HirCastExpression, id: ExprId) -> IResult { let evaluated_lhs = self.evaluate(cast.lhs)?; - Self::evaluate_cast_one_step(cast, id, evaluated_lhs, self.elaborator.interner) + let location = self.elaborator.interner.expr_location(&id); + Self::evaluate_cast_one_step( + &cast.r#type, + location, + evaluated_lhs, + self.elaborator.interner, + ) } /// evaluate_cast without recursion pub fn evaluate_cast_one_step( - cast: &HirCastExpression, - id: ExprId, + typ: &Type, + location: Location, evaluated_lhs: Value, interner: &NodeInterner, ) -> IResult { @@ -1405,6 +1424,7 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { let (mut lhs, lhs_is_negative) = match evaluated_lhs { Value::Field(value) => (value, false), + Value::U1(value) => ((value as u128).into(), false), Value::U8(value) => ((value as u128).into(), false), Value::U16(value) => ((value as u128).into(), false), Value::U32(value) => ((value as u128).into(), false), @@ -1417,7 +1437,6 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { (if value { FieldElement::one() } else { FieldElement::zero() }, false) } value => { - let location = interner.expr_location(&id); let typ = value.get_type().into_owned(); return Err(InterpreterError::NonNumericCasted { typ, location }); } @@ -1434,7 +1453,7 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { } // Now actually cast the lhs, bit casting and wrapping as necessary - match cast.r#type.follow_bindings() { + match typ.follow_bindings() { Type::FieldElement => { if lhs_is_negative { lhs = FieldElement::zero() - lhs; @@ -1443,8 +1462,7 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { } Type::Integer(sign, bit_size) => match (sign, bit_size) { (Signedness::Unsigned, IntegerBitSize::One) => { - let location = interner.expr_location(&id); - Err(InterpreterError::TypeUnsupported { typ: cast.r#type.clone(), location }) + Err(InterpreterError::TypeUnsupported { typ: typ.clone(), location }) } (Signedness::Unsigned, IntegerBitSize::Eight) => cast_to_int!(lhs, to_u128, u8, U8), (Signedness::Unsigned, IntegerBitSize::Sixteen) => { @@ -1457,8 +1475,7 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { cast_to_int!(lhs, to_u128, u64, U64) } (Signedness::Signed, IntegerBitSize::One) => { - let location = interner.expr_location(&id); - Err(InterpreterError::TypeUnsupported { typ: cast.r#type.clone(), location }) + Err(InterpreterError::TypeUnsupported { typ: typ.clone(), location }) } (Signedness::Signed, IntegerBitSize::Eight) => cast_to_int!(lhs, to_i128, i8, I8), (Signedness::Signed, IntegerBitSize::Sixteen) => { @@ -1472,10 +1489,7 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { } }, Type::Bool => Ok(Value::Bool(!lhs.is_zero() || lhs_is_negative)), - typ => { - let location = interner.expr_location(&id); - Err(InterpreterError::CastToNonNumericType { typ, location }) - } + typ => Err(InterpreterError::CastToNonNumericType { typ, location }), } } @@ -1536,6 +1550,7 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { HirStatement::Constrain(constrain) => self.evaluate_constrain(constrain), HirStatement::Assign(assign) => self.evaluate_assign(assign), HirStatement::For(for_) => self.evaluate_for(for_), + HirStatement::Loop(expression) => self.evaluate_loop(expression), HirStatement::Break => self.evaluate_break(statement), HirStatement::Continue => self.evaluate_continue(statement), HirStatement::Expression(expression) => self.evaluate(expression), @@ -1727,6 +1742,34 @@ impl<'local, 'interner> Interpreter<'local, 'interner> { Ok(Value::Unit) } + fn evaluate_loop(&mut self, expr: ExprId) -> IResult { + let was_in_loop = std::mem::replace(&mut self.in_loop, true); + let in_lsp = self.elaborator.interner.is_in_lsp_mode(); + let mut counter = 0; + + loop { + self.push_scope(); + + match self.evaluate(expr) { + Ok(_) => (), + Err(InterpreterError::Break) => break, + Err(InterpreterError::Continue) => continue, + Err(other) => return Err(other), + } + + self.pop_scope(); + + counter += 1; + if in_lsp && counter == 10_000 { + let location = self.elaborator.interner.expr_location(&expr); + return Err(InterpreterError::LoopHaltedForUiResponsiveness { location }); + } + } + + self.in_loop = was_in_loop; + Ok(Value::Unit) + } + fn evaluate_break(&mut self, id: StmtId) -> IResult { if self.in_loop { Err(InterpreterError::Break) diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs index 3d8ccf78926..3506b63919c 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs @@ -1,6 +1,6 @@ use std::rc::Rc; -use acvm::{acir::BlackBoxFunc, AcirField, FieldElement}; +use acvm::{AcirField, FieldElement}; use builtin_helpers::{ block_expression_to_value, byte_array_type, check_argument_count, check_function_not_yet_resolved, check_one_argument, check_three_arguments, @@ -23,6 +23,7 @@ use crate::{ Pattern, Signedness, Statement, StatementKind, UnaryOp, UnresolvedType, UnresolvedTypeData, Visibility, }, + elaborator::Elaborator, hir::{ comptime::{ errors::IResult, @@ -32,9 +33,11 @@ use crate::{ def_collector::dc_crate::CollectedItems, def_map::ModuleDefId, }, - hir_def::expr::{HirExpression, HirLiteral}, - hir_def::function::FunctionBody, - hir_def::{self}, + hir_def::{ + self, + expr::{HirExpression, HirLiteral}, + function::FunctionBody, + }, node_interner::{DefinitionKind, NodeInterner, TraitImplKind}, parser::{Parser, StatementOrExpressionOrLValue}, token::{Attribute, Token}, @@ -158,7 +161,7 @@ impl<'local, 'context> Interpreter<'local, 'context> { "modulus_le_bits" => modulus_le_bits(arguments, location), "modulus_le_bytes" => modulus_le_bytes(arguments, location), "modulus_num_bits" => modulus_num_bits(arguments, location), - "quoted_as_expr" => quoted_as_expr(interner, arguments, return_type, location), + "quoted_as_expr" => quoted_as_expr(self.elaborator, arguments, return_type, location), "quoted_as_module" => quoted_as_module(self, arguments, return_type, location), "quoted_as_trait_constraint" => quoted_as_trait_constraint(self, arguments, location), "quoted_as_type" => quoted_as_type(self, arguments, location), @@ -178,8 +181,13 @@ impl<'local, 'context> Interpreter<'local, 'context> { "struct_def_add_generic" => struct_def_add_generic(interner, arguments, location), "struct_def_as_type" => struct_def_as_type(interner, arguments, location), "struct_def_eq" => struct_def_eq(arguments, location), - "struct_def_fields" => struct_def_fields(interner, arguments, location), - "struct_def_generics" => struct_def_generics(interner, arguments, location), + "struct_def_fields" => struct_def_fields(interner, arguments, location, call_stack), + "struct_def_fields_as_written" => { + struct_def_fields_as_written(interner, arguments, location) + } + "struct_def_generics" => { + struct_def_generics(interner, arguments, return_type, location) + } "struct_def_has_named_attribute" => { struct_def_has_named_attribute(interner, arguments, location) } @@ -189,6 +197,8 @@ impl<'local, 'context> Interpreter<'local, 'context> { "struct_def_set_fields" => struct_def_set_fields(interner, arguments, location), "to_be_radix" => to_be_radix(arguments, return_type, location), "to_le_radix" => to_le_radix(arguments, return_type, location), + "to_be_bits" => to_be_bits(arguments, return_type, location), + "to_le_bits" => to_le_bits(arguments, return_type, location), "trait_constraint_eq" => trait_constraint_eq(arguments, location), "trait_constraint_hash" => trait_constraint_hash(arguments, location), "trait_def_as_trait_constraint" => { @@ -236,9 +246,6 @@ impl<'local, 'context> Interpreter<'local, 'context> { "unresolved_type_is_field" => unresolved_type_is_field(interner, arguments, location), "unresolved_type_is_unit" => unresolved_type_is_unit(interner, arguments, location), "zeroed" => zeroed(return_type, location.span), - blackbox if BlackBoxFunc::is_valid_black_box_func_name(blackbox) => { - self.call_foreign(blackbox, arguments, return_type, location) - } _ => { let item = format!("Comptime evaluation for builtin function '{name}'"); Err(InterpreterError::Unimplemented { item, location }) @@ -440,10 +447,11 @@ fn struct_def_as_type( Ok(Value::Type(Type::Struct(struct_def_rc, generics))) } -/// fn generics(self) -> [Type] +/// fn generics(self) -> [(Type, Option)] fn struct_def_generics( interner: &NodeInterner, arguments: Vec<(Value, Location)>, + return_type: Type, location: Location, ) -> IResult { let argument = check_one_argument(arguments, location)?; @@ -451,11 +459,38 @@ fn struct_def_generics( let struct_def = interner.get_struct(struct_id); let struct_def = struct_def.borrow(); - let generics = - struct_def.generics.iter().map(|generic| Value::Type(generic.clone().as_named_generic())); + let expected = Type::Slice(Box::new(Type::Tuple(vec![ + Type::Quoted(QuotedType::Type), + interner.next_type_variable(), // Option + ]))); + + let actual = return_type.clone(); + + let slice_item_type = match return_type { + Type::Slice(item_type) => *item_type, + _ => return Err(InterpreterError::TypeMismatch { expected, actual, location }), + }; + + let option_typ = match &slice_item_type { + Type::Tuple(types) if types.len() == 2 => types[1].clone(), + _ => return Err(InterpreterError::TypeMismatch { expected, actual, location }), + }; - let typ = Type::Slice(Box::new(Type::Quoted(QuotedType::Type))); - Ok(Value::Slice(generics.collect(), typ)) + let generics: IResult<_> = struct_def + .generics + .iter() + .map(|generic| -> IResult { + let generic_as_named = generic.clone().as_named_generic(); + let numeric_type = match generic_as_named.kind() { + Kind::Numeric(numeric_type) => Some(Value::Type(*numeric_type)), + _ => None, + }; + let numeric_type = option(option_typ.clone(), numeric_type, location.span)?; + Ok(Value::Tuple(vec![Value::Type(generic_as_named), numeric_type])) + }) + .collect(); + + Ok(Value::Slice(generics?, slice_item_type)) } fn struct_def_hash(arguments: Vec<(Value, Location)>, location: Location) -> IResult { @@ -480,12 +515,57 @@ fn struct_def_has_named_attribute( Ok(Value::Bool(has_named_attribute(&name, interner.struct_attributes(&struct_id)))) } -/// fn fields(self) -> [(Quoted, Type)] -/// Returns (name, type) pairs of each field of this StructDefinition +/// fn fields(self, generic_args: [Type]) -> [(Quoted, Type)] +/// Returns (name, type) pairs of each field of this StructDefinition. +/// Applies the given generic arguments to each field. fn struct_def_fields( interner: &mut NodeInterner, arguments: Vec<(Value, Location)>, location: Location, + call_stack: &im::Vector, +) -> IResult { + let (typ, generic_args) = check_two_arguments(arguments, location)?; + let struct_id = get_struct(typ)?; + let struct_def = interner.get_struct(struct_id); + let struct_def = struct_def.borrow(); + + let args_location = generic_args.1; + let generic_args = get_slice(interner, generic_args)?.0; + let generic_args = try_vecmap(generic_args, |arg| get_type((arg, args_location)))?; + + let actual = generic_args.len(); + let expected = struct_def.generics.len(); + if actual != expected { + let s = if expected == 1 { "" } else { "s" }; + let was_were = if actual == 1 { "was" } else { "were" }; + let message = Some(format!("`StructDefinition::fields` expected {expected} generic{s} for `{}` but {actual} {was_were} given", struct_def.name)); + let location = args_location; + let call_stack = call_stack.clone(); + return Err(InterpreterError::FailingConstraint { message, location, call_stack }); + } + + let mut fields = im::Vector::new(); + + for (field_name, field_type) in struct_def.get_fields(&generic_args) { + let name = Value::Quoted(Rc::new(vec![Token::Ident(field_name)])); + fields.push_back(Value::Tuple(vec![name, Value::Type(field_type)])); + } + + let typ = Type::Slice(Box::new(Type::Tuple(vec![ + Type::Quoted(QuotedType::Quoted), + Type::Quoted(QuotedType::Type), + ]))); + Ok(Value::Slice(fields, typ)) +} + +/// fn fields_as_written(self) -> [(Quoted, Type)] +/// Returns (name, type) pairs of each field of this StructDefinition. +/// +/// Note that any generic arguments won't be applied: if you need them to be, use `fields`. +fn struct_def_fields_as_written( + interner: &mut NodeInterner, + arguments: Vec<(Value, Location)>, + location: Location, ) -> IResult { let argument = check_one_argument(arguments, location)?; let struct_id = get_struct(argument)?; @@ -679,15 +759,19 @@ fn slice_insert( // fn as_expr(quoted: Quoted) -> Option fn quoted_as_expr( - interner: &NodeInterner, + elaborator: &mut Elaborator, arguments: Vec<(Value, Location)>, return_type: Type, location: Location, ) -> IResult { let argument = check_one_argument(arguments, location)?; - let result = - parse(interner, argument, Parser::parse_statement_or_expression_or_lvalue, "an expression"); + let result = parse( + elaborator, + argument, + Parser::parse_statement_or_expression_or_lvalue, + "an expression", + ); let value = result.ok().map( @@ -712,13 +796,9 @@ fn quoted_as_module( ) -> IResult { let argument = check_one_argument(arguments, location)?; - let path = parse( - interpreter.elaborator.interner, - argument, - Parser::parse_path_no_turbofish_or_error, - "a path", - ) - .ok(); + let path = + parse(interpreter.elaborator, argument, Parser::parse_path_no_turbofish_or_error, "a path") + .ok(); let option_value = path.and_then(|path| { let module = interpreter .elaborate_in_function(interpreter.current_function, |elaborator| { @@ -738,7 +818,7 @@ fn quoted_as_trait_constraint( ) -> IResult { let argument = check_one_argument(arguments, location)?; let trait_bound = parse( - interpreter.elaborator.interner, + interpreter.elaborator, argument, Parser::parse_trait_bound_or_error, "a trait constraint", @@ -759,8 +839,7 @@ fn quoted_as_type( location: Location, ) -> IResult { let argument = check_one_argument(arguments, location)?; - let typ = - parse(interpreter.elaborator.interner, argument, Parser::parse_type_or_error, "a type")?; + let typ = parse(interpreter.elaborator, argument, Parser::parse_type_or_error, "a type")?; let typ = interpreter .elaborate_in_function(interpreter.current_function, |elab| elab.resolve_type(typ)); Ok(Value::Type(typ)) @@ -777,6 +856,26 @@ fn quoted_tokens(arguments: Vec<(Value, Location)>, location: Location) -> IResu )) } +fn to_be_bits( + arguments: Vec<(Value, Location)>, + return_type: Type, + location: Location, +) -> IResult { + let value = check_one_argument(arguments, location)?; + let radix = (Value::U32(2), value.1); + to_be_radix(vec![value, radix], return_type, location) +} + +fn to_le_bits( + arguments: Vec<(Value, Location)>, + return_type: Type, + location: Location, +) -> IResult { + let value = check_one_argument(arguments, location)?; + let radix = (Value::U32(2), value.1); + to_le_radix(vec![value, radix], return_type, location) +} + fn to_be_radix( arguments: Vec<(Value, Location)>, return_type: Type, @@ -801,10 +900,10 @@ fn to_le_radix( let value = get_field(value)?; let radix = get_u32(radix)?; - let limb_count = if let Type::Array(length, _) = return_type { + let (limb_count, element_type) = if let Type::Array(length, element_type) = return_type { if let Type::Constant(limb_count, kind) = *length { if kind.unifies(&Kind::u32()) { - limb_count + (limb_count, element_type) } else { return Err(InterpreterError::TypeAnnotationsNeededForMethodCall { location }); } @@ -815,14 +914,29 @@ fn to_le_radix( return Err(InterpreterError::TypeAnnotationsNeededForMethodCall { location }); }; + let return_type_is_bits = + *element_type == Type::Integer(Signedness::Unsigned, IntegerBitSize::One); + // Decompose the integer into its radix digits in little endian form. let decomposed_integer = compute_to_radix_le(value, radix); - let decomposed_integer = - vecmap(0..limb_count.to_u128() as usize, |i| match decomposed_integer.get(i) { - Some(digit) => Value::U8(*digit), - None => Value::U8(0), - }); - let result_type = byte_array_type(decomposed_integer.len()); + let decomposed_integer = vecmap(0..limb_count.to_u128() as usize, |i| { + let digit = match decomposed_integer.get(i) { + Some(digit) => *digit, + None => 0, + }; + // The only built-ins that use these either return `[u1; N]` or `[u8; N]` + if return_type_is_bits { + Value::U1(digit != 0) + } else { + Value::U8(digit) + } + }); + + let result_type = Type::Array( + Box::new(Type::Constant(decomposed_integer.len().into(), Kind::u32())), + element_type, + ); + Ok(Value::Array(decomposed_integer.into(), result_type)) } @@ -2456,7 +2570,7 @@ fn function_def_set_parameters( )?; let parameter_type = get_type((tuple.pop().unwrap(), parameters_argument_location))?; let parameter_pattern = parse( - interpreter.elaborator.interner, + interpreter.elaborator, (tuple.pop().unwrap(), parameters_argument_location), Parser::parse_pattern_or_error, "a pattern", @@ -2573,12 +2687,11 @@ fn module_add_item( ) -> IResult { let (self_argument, item) = check_two_arguments(arguments, location)?; let module_id = get_module(self_argument)?; - let module_data = interpreter.elaborator.get_module(module_id); let parser = Parser::parse_top_level_items; - let top_level_statements = - parse(interpreter.elaborator.interner, item, parser, "a top-level item")?; + let top_level_statements = parse(interpreter.elaborator, item, parser, "a top-level item")?; + let module_data = interpreter.elaborator.get_module(module_id); interpreter.elaborate_in_module(module_id, module_data.location.file, |elaborator| { let mut generated_items = CollectedItems::default(); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin/builtin_helpers.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin/builtin_helpers.rs index cf90aab32e0..a3f84a00bfb 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin/builtin_helpers.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin/builtin_helpers.rs @@ -5,10 +5,11 @@ use acvm::FieldElement; use iter_extended::try_vecmap; use noirc_errors::Location; +use crate::elaborator::Elaborator; use crate::hir::comptime::display::tokens_to_string; use crate::hir::comptime::value::add_token_spans; use crate::lexer::Lexer; -use crate::parser::Parser; +use crate::parser::{Parser, ParserError}; use crate::{ ast::{ BlockExpression, ExpressionKind, Ident, IntegerBitSize, LValue, Pattern, Signedness, @@ -493,7 +494,7 @@ pub(super) fn lex(input: &str) -> Vec { } pub(super) fn parse<'a, T, F>( - interner: &NodeInterner, + elaborator: &mut Elaborator, (value, location): (Value, Location), parser: F, rule: &'static str, @@ -503,7 +504,12 @@ where { let tokens = get_quoted((value, location))?; let quoted = add_token_spans(tokens.clone(), location.span); - parse_tokens(tokens, quoted, interner, location, parser, rule) + let (result, warnings) = + parse_tokens(tokens, quoted, elaborator.interner, location, parser, rule)?; + for warning in warnings { + elaborator.errors.push((warning.into(), location.file)); + } + Ok(result) } pub(super) fn parse_tokens<'a, T, F>( @@ -513,7 +519,7 @@ pub(super) fn parse_tokens<'a, T, F>( location: Location, parsing_function: F, rule: &'static str, -) -> IResult +) -> IResult<(T, Vec)> where F: FnOnce(&mut Parser<'a>) -> T, { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/foreign.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/foreign.rs index 99cc11ecd2a..0221280ae1b 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/foreign.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/foreign.rs @@ -40,6 +40,7 @@ impl<'local, 'context> Interpreter<'local, 'context> { arguments, return_type, location, + self.pedantic_solving, ) } } @@ -52,6 +53,7 @@ fn call_foreign( args: Vec<(Value, Location)>, return_type: Type, location: Location, + pedantic_solving: bool, ) -> IResult { use BlackBoxFunc::*; @@ -79,9 +81,11 @@ fn call_foreign( location, acvm::blackbox_solver::ecdsa_secp256r1_verify, ), - "embedded_curve_add" => embedded_curve_add(args, location), - "multi_scalar_mul" => multi_scalar_mul(interner, args, location), - "poseidon2_permutation" => poseidon2_permutation(interner, args, location), + "embedded_curve_add" => embedded_curve_add(args, location, pedantic_solving), + "multi_scalar_mul" => multi_scalar_mul(interner, args, location, pedantic_solving), + "poseidon2_permutation" => { + poseidon2_permutation(interner, args, location, pedantic_solving) + } "keccakf1600" => keccakf1600(interner, args, location), "range" => apply_range_constraint(args, location), "sha256_compression" => sha256_compression(interner, args, location), @@ -269,13 +273,17 @@ fn ecdsa_secp256_verify( /// point2: EmbeddedCurvePoint, /// ) -> [Field; 3] /// ``` -fn embedded_curve_add(arguments: Vec<(Value, Location)>, location: Location) -> IResult { +fn embedded_curve_add( + arguments: Vec<(Value, Location)>, + location: Location, + pedantic_solving: bool, +) -> IResult { let (point1, point2) = check_two_arguments(arguments, location)?; let (p1x, p1y, p1inf) = get_embedded_curve_point(point1)?; let (p2x, p2y, p2inf) = get_embedded_curve_point(point2)?; - let (x, y, inf) = Bn254BlackBoxSolver + let (x, y, inf) = Bn254BlackBoxSolver(pedantic_solving) .ec_add(&p1x, &p1y, &p1inf.into(), &p2x, &p2y, &p2inf.into()) .map_err(|e| InterpreterError::BlackBoxError(e, location))?; @@ -292,6 +300,7 @@ fn multi_scalar_mul( interner: &mut NodeInterner, arguments: Vec<(Value, Location)>, location: Location, + pedantic_solving: bool, ) -> IResult { let (points, scalars) = check_two_arguments(arguments, location)?; @@ -306,7 +315,7 @@ fn multi_scalar_mul( scalars_hi.push(hi); } - let (x, y, inf) = Bn254BlackBoxSolver + let (x, y, inf) = Bn254BlackBoxSolver(pedantic_solving) .multi_scalar_mul(&points, &scalars_lo, &scalars_hi) .map_err(|e| InterpreterError::BlackBoxError(e, location))?; @@ -318,13 +327,14 @@ fn poseidon2_permutation( interner: &mut NodeInterner, arguments: Vec<(Value, Location)>, location: Location, + pedantic_solving: bool, ) -> IResult { let (input, state_length) = check_two_arguments(arguments, location)?; let (input, typ) = get_array_map(interner, input, get_field)?; let state_length = get_u32(state_length)?; - let fields = Bn254BlackBoxSolver + let fields = Bn254BlackBoxSolver(pedantic_solving) .poseidon2_permutation(&input, state_length) .map_err(|error| InterpreterError::BlackBoxError(error, location))?; @@ -435,6 +445,7 @@ mod tests { for blackbox in BlackBoxFunc::iter() { let name = blackbox.name(); + let pedantic_solving = true; match call_foreign( interpreter.elaborator.interner, &mut interpreter.bigint_solver, @@ -442,6 +453,7 @@ mod tests { Vec::new(), Type::Unit, no_location, + pedantic_solving, ) { Ok(_) => { // Exists and works with no args (unlikely) diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/tests.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/tests.rs index 2d3bf928917..342d0a616a0 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/tests.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/tests.rs @@ -19,6 +19,8 @@ use crate::node_interner::FuncId; use crate::parse_program; /// Create an interpreter for a code snippet and pass it to a test function. +/// +/// The stdlib is not made available as a dependency. pub(crate) fn with_interpreter( src: &str, f: impl FnOnce(&mut Interpreter, FuncId, &[(CompilationError, FileId)]) -> T, @@ -58,8 +60,14 @@ pub(crate) fn with_interpreter( let main = context.get_main_function(&krate).expect("Expected 'main' function"); - let mut elaborator = - Elaborator::elaborate_and_return_self(&mut context, krate, collector.items, None); + let pedantic_solving = true; + let mut elaborator = Elaborator::elaborate_and_return_self( + &mut context, + krate, + collector.items, + None, + pedantic_solving, + ); let errors = elaborator.errors.clone(); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/value.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/value.rs index 945fb45026d..77933ba9361 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/value.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/value.rs @@ -1,6 +1,6 @@ use std::{borrow::Cow, rc::Rc, vec}; -use acvm::{AcirField, FieldElement}; +use acvm::FieldElement; use im::Vector; use iter_extended::{try_vecmap, vecmap}; use noirc_errors::{Location, Span}; @@ -12,6 +12,7 @@ use crate::{ IntegerBitSize, LValue, Literal, Path, Pattern, Signedness, Statement, StatementKind, UnresolvedType, UnresolvedTypeData, }, + elaborator::Elaborator, hir::{def_map::ModuleId, type_check::generics::TraitGenerics}, hir_def::expr::{ HirArrayLiteral, HirConstructorExpression, HirExpression, HirIdent, HirLambda, HirLiteral, @@ -158,7 +159,7 @@ impl Value { pub(crate) fn into_expression( self, - interner: &mut NodeInterner, + elaborator: &mut Elaborator, location: Location, ) -> IResult { let kind = match self { @@ -212,22 +213,23 @@ impl Value { ExpressionKind::Literal(Literal::Str(unwrap_rc(value))) } Value::Function(id, typ, bindings) => { - let id = interner.function_definition_id(id); + let id = elaborator.interner.function_definition_id(id); let impl_kind = ImplKind::NotATraitMethod; let ident = HirIdent { location, id, impl_kind }; - let expr_id = interner.push_expr(HirExpression::Ident(ident, None)); - interner.push_expr_location(expr_id, location.span, location.file); - interner.push_expr_type(expr_id, typ); - interner.store_instantiation_bindings(expr_id, unwrap_rc(bindings)); + let expr_id = elaborator.interner.push_expr(HirExpression::Ident(ident, None)); + elaborator.interner.push_expr_location(expr_id, location.span, location.file); + elaborator.interner.push_expr_type(expr_id, typ); + elaborator.interner.store_instantiation_bindings(expr_id, unwrap_rc(bindings)); ExpressionKind::Resolved(expr_id) } Value::Tuple(fields) => { - let fields = try_vecmap(fields, |field| field.into_expression(interner, location))?; + let fields = + try_vecmap(fields, |field| field.into_expression(elaborator, location))?; ExpressionKind::Tuple(fields) } Value::Struct(fields, typ) => { let fields = try_vecmap(fields, |(name, field)| { - let field = field.into_expression(interner, location)?; + let field = field.into_expression(elaborator, location)?; Ok((Ident::new(unwrap_rc(name), location.span), field)) })?; @@ -246,12 +248,12 @@ impl Value { } Value::Array(elements, _) => { let elements = - try_vecmap(elements, |element| element.into_expression(interner, location))?; + try_vecmap(elements, |element| element.into_expression(elaborator, location))?; ExpressionKind::Literal(Literal::Array(ArrayLiteral::Standard(elements))) } Value::Slice(elements, _) => { let elements = - try_vecmap(elements, |element| element.into_expression(interner, location))?; + try_vecmap(elements, |element| element.into_expression(elaborator, location))?; ExpressionKind::Literal(Literal::Slice(ArrayLiteral::Standard(elements))) } Value::Quoted(tokens) => { @@ -262,12 +264,18 @@ impl Value { let parser = Parser::for_tokens(tokens_to_parse); return match parser.parse_result(Parser::parse_expression_or_error) { - Ok(expr) => Ok(expr), + Ok((expr, warnings)) => { + for warning in warnings { + elaborator.errors.push((warning.into(), location.file)); + } + + Ok(expr) + } Err(mut errors) => { let error = errors.swap_remove(0); let file = location.file; let rule = "an expression"; - let tokens = tokens_to_string(tokens, interner); + let tokens = tokens_to_string(tokens, elaborator.interner); Err(InterpreterError::FailedToParseMacro { error, file, tokens, rule }) } }; @@ -293,7 +301,7 @@ impl Value { | Value::Closure(..) | Value::ModuleDefinition(_) => { let typ = self.get_type().into_owned(); - let value = self.display(interner).to_string(); + let value = self.display(elaborator.interner).to_string(); return Err(InterpreterError::CannotInlineMacro { typ, value, location }); } }; @@ -409,6 +417,9 @@ impl Value { Value::Pointer(element, true) => { return element.unwrap_or_clone().into_hir_expression(interner, location); } + Value::Closure(hir_lambda, _args, _typ, _opt_func_id, _module_id) => { + HirExpression::Lambda(hir_lambda) + } Value::TypedExpr(TypedExpr::StmtId(..)) | Value::Expr(..) | Value::Pointer(..) @@ -420,7 +431,6 @@ impl Value { | Value::Zeroed(_) | Value::Type(_) | Value::UnresolvedType(_) - | Value::Closure(..) | Value::ModuleDefinition(_) => { let typ = self.get_type().into_owned(); let value = self.display(interner).to_string(); @@ -502,19 +512,32 @@ impl Value { Ok(vec![token]) } - /// Converts any unsigned `Value` into a `u128`. - /// Returns `None` for negative integers. - pub(crate) fn to_u128(&self) -> Option { + /// Returns false for non-integral `Value`s. + pub(crate) fn is_integral(&self) -> bool { + use Value::*; + matches!( + self, + Field(_) | I8(_) | I16(_) | I32(_) | I64(_) | U8(_) | U16(_) | U32(_) | U64(_) + ) + } + + pub(crate) fn is_closure(&self) -> bool { + matches!(self, Value::Closure(..)) + } + + /// Converts any non-negative `Value` into a `FieldElement`. + /// Returns `None` for negative integers and non-integral `Value`s. + pub(crate) fn to_field_element(&self) -> Option { match self { - Self::Field(value) => Some(value.to_u128()), - Self::I8(value) => (*value >= 0).then_some(*value as u128), - Self::I16(value) => (*value >= 0).then_some(*value as u128), - Self::I32(value) => (*value >= 0).then_some(*value as u128), - Self::I64(value) => (*value >= 0).then_some(*value as u128), - Self::U8(value) => Some(*value as u128), - Self::U16(value) => Some(*value as u128), - Self::U32(value) => Some(*value as u128), - Self::U64(value) => Some(*value as u128), + Self::Field(value) => Some(*value), + Self::I8(value) => (*value >= 0).then_some((*value as u128).into()), + Self::I16(value) => (*value >= 0).then_some((*value as u128).into()), + Self::I32(value) => (*value >= 0).then_some((*value as u128).into()), + Self::I64(value) => (*value >= 0).then_some((*value as u128).into()), + Self::U8(value) => Some((*value as u128).into()), + Self::U16(value) => Some((*value as u128).into()), + Self::U32(value) => Some((*value as u128).into()), + Self::U64(value) => Some((*value as u128).into()), _ => None, } } @@ -522,16 +545,16 @@ impl Value { pub(crate) fn into_top_level_items( self, location: Location, - interner: &NodeInterner, + elaborator: &mut Elaborator, ) -> IResult> { let parser = Parser::parse_top_level_items; match self { Value::Quoted(tokens) => { - parse_tokens(tokens, interner, parser, location, "top-level item") + parse_tokens(tokens, elaborator, parser, location, "top-level item") } _ => { let typ = self.get_type().into_owned(); - let value = self.display(interner).to_string(); + let value = self.display(elaborator.interner).to_string(); Err(InterpreterError::CannotInlineMacro { value, typ, location }) } } @@ -545,7 +568,7 @@ pub(crate) fn unwrap_rc(rc: Rc) -> T { fn parse_tokens<'a, T, F>( tokens: Rc>, - interner: &NodeInterner, + elaborator: &mut Elaborator, parsing_function: F, location: Location, rule: &'static str, @@ -555,11 +578,16 @@ where { let parser = Parser::for_tokens(add_token_spans(tokens.clone(), location.span)); match parser.parse_result(parsing_function) { - Ok(expr) => Ok(expr), + Ok((expr, warnings)) => { + for warning in warnings { + elaborator.errors.push((warning.into(), location.file)); + } + Ok(expr) + } Err(mut errors) => { let error = errors.swap_remove(0); let file = location.file; - let tokens = tokens_to_string(tokens, interner); + let tokens = tokens_to_string(tokens, elaborator.interner); Err(InterpreterError::FailedToParseMacro { error, file, tokens, rule }) } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs index 33dab802b21..10866f4b309 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs @@ -3,7 +3,7 @@ use super::errors::{DefCollectorErrorKind, DuplicateType}; use crate::elaborator::Elaborator; use crate::graph::CrateId; use crate::hir::comptime::InterpreterError; -use crate::hir::def_map::{CrateDefMap, LocalModuleId, ModuleId}; +use crate::hir::def_map::{CrateDefMap, LocalModuleId, ModuleDefId, ModuleId}; use crate::hir::resolution::errors::ResolverError; use crate::hir::type_check::TypeCheckError; use crate::locations::ReferencesTracker; @@ -14,7 +14,7 @@ use crate::{Generics, Type}; use crate::hir::resolution::import::{resolve_import, ImportDirective}; use crate::hir::Context; -use crate::ast::Expression; +use crate::ast::{Expression, NoirEnumeration}; use crate::node_interner::{ FuncId, GlobalId, ModuleAttributes, NodeInterner, ReferenceId, StructId, TraitId, TraitImplId, TypeAliasId, @@ -64,6 +64,12 @@ pub struct UnresolvedStruct { pub struct_def: NoirStruct, } +pub struct UnresolvedEnum { + pub file_id: FileId, + pub module_id: LocalModuleId, + pub enum_def: NoirEnumeration, +} + #[derive(Clone)] pub struct UnresolvedTrait { pub file_id: FileId, @@ -141,7 +147,8 @@ pub struct DefCollector { #[derive(Default)] pub struct CollectedItems { pub functions: Vec, - pub(crate) types: BTreeMap, + pub(crate) structs: BTreeMap, + pub(crate) enums: BTreeMap, pub(crate) type_aliases: BTreeMap, pub(crate) traits: BTreeMap, pub globals: Vec, @@ -153,7 +160,7 @@ pub struct CollectedItems { impl CollectedItems { pub fn is_empty(&self) -> bool { self.functions.is_empty() - && self.types.is_empty() + && self.structs.is_empty() && self.type_aliases.is_empty() && self.traits.is_empty() && self.globals.is_empty() @@ -254,7 +261,8 @@ impl DefCollector { imports: vec![], items: CollectedItems { functions: vec![], - types: BTreeMap::new(), + structs: BTreeMap::new(), + enums: BTreeMap::new(), type_aliases: BTreeMap::new(), traits: BTreeMap::new(), impls: HashMap::default(), @@ -275,7 +283,7 @@ impl DefCollector { ast: SortedModule, root_file_id: FileId, debug_comptime_in_file: Option<&str>, - error_on_unused_items: bool, + pedantic_solving: bool, ) -> Vec<(CompilationError, FileId)> { let mut errors: Vec<(CompilationError, FileId)> = vec![]; let crate_id = def_map.krate; @@ -288,12 +296,11 @@ impl DefCollector { let crate_graph = &context.crate_graph[crate_id]; for dep in crate_graph.dependencies.clone() { - let error_on_usage_tracker = false; errors.extend(CrateDefMap::collect_defs( dep.crate_id, context, debug_comptime_in_file, - error_on_usage_tracker, + pedantic_solving, )); let dep_def_map = @@ -413,13 +420,24 @@ impl DefCollector { visibility, ); - if visibility != ItemVisibility::Private { + if context.def_interner.is_in_lsp_mode() + && visibility != ItemVisibility::Private + { context.def_interner.register_name_for_auto_import( name.to_string(), module_def_id, visibility, Some(defining_module), ); + + if let ModuleDefId::TraitId(trait_id) = module_def_id { + context.def_interner.add_trait_reexport( + trait_id, + defining_module, + name.clone(), + visibility, + ); + } } } @@ -459,14 +477,17 @@ impl DefCollector { }) }); - let mut more_errors = - Elaborator::elaborate(context, crate_id, def_collector.items, debug_comptime_in_file); + let mut more_errors = Elaborator::elaborate( + context, + crate_id, + def_collector.items, + debug_comptime_in_file, + pedantic_solving, + ); errors.append(&mut more_errors); - if error_on_unused_items { - Self::check_unused_items(context, crate_id, &mut errors); - } + Self::check_unused_items(context, crate_id, &mut errors); errors } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/dc_mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/dc_mod.rs index e7953aab5a4..41234980942 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/dc_mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/dc_mod.rs @@ -12,8 +12,9 @@ use rustc_hash::FxHashMap as HashMap; use crate::ast::{ Documented, Expression, FunctionDefinition, Ident, ItemVisibility, LetStatement, - ModuleDeclaration, NoirFunction, NoirStruct, NoirTrait, NoirTraitImpl, NoirTypeAlias, Pattern, - TraitImplItemKind, TraitItem, TypeImpl, UnresolvedType, UnresolvedTypeData, + ModuleDeclaration, NoirEnumeration, NoirFunction, NoirStruct, NoirTrait, NoirTraitImpl, + NoirTypeAlias, Pattern, TraitImplItemKind, TraitItem, TypeImpl, UnresolvedType, + UnresolvedTypeData, }; use crate::hir::resolution::errors::ResolverError; use crate::node_interner::{ModuleAttributes, NodeInterner, ReferenceId, StructId}; @@ -27,8 +28,8 @@ use crate::{ }; use crate::{Generics, Kind, ResolvedGeneric, Type, TypeVariable}; -use super::dc_crate::CollectedItems; use super::dc_crate::ModuleAttribute; +use super::dc_crate::{CollectedItems, UnresolvedEnum}; use super::{ dc_crate::{ CompilationError, DefCollector, UnresolvedFunctions, UnresolvedGlobal, UnresolvedTraitImpl, @@ -91,7 +92,7 @@ pub fn collect_defs( errors.extend(collector.collect_traits(context, ast.traits, crate_id)); - errors.extend(collector.collect_structs(context, ast.types, crate_id)); + errors.extend(collector.collect_structs(context, ast.structs, crate_id)); errors.extend(collector.collect_type_aliases(context, ast.type_aliases, crate_id)); @@ -317,7 +318,7 @@ impl<'a> ModCollector<'a> { krate, &mut definition_errors, ) { - self.def_collector.items.types.insert(id, the_struct); + self.def_collector.items.structs.insert(id, the_struct); } } definition_errors @@ -482,12 +483,14 @@ impl<'a> ModCollector<'a> { is_comptime, } => { let func_id = context.def_interner.push_empty_fn(); - method_ids.insert(name.to_string(), func_id); + if !method_ids.contains_key(&name.0.contents) { + method_ids.insert(name.to_string(), func_id); + } let location = Location::new(name.span(), self.file_id); let modifiers = FunctionModifiers { name: name.to_string(), - visibility: ItemVisibility::Public, + visibility: trait_definition.visibility, // TODO(Maddiaa): Investigate trait implementations with attributes see: https://github.com/noir-lang/noir/issues/2629 attributes: crate::token::Attributes::empty(), is_unconstrained: *is_unconstrained, @@ -500,6 +503,9 @@ impl<'a> ModCollector<'a> { .def_interner .push_function_definition(func_id, modifiers, trait_id.0, location); + let referenced = ReferenceId::Function(func_id); + context.def_interner.add_definition_location(referenced, Some(trait_id.0)); + if !trait_item.doc_comments.is_empty() { context.def_interner.set_doc_comments( ReferenceId::Function(func_id), @@ -518,8 +524,8 @@ impl<'a> ModCollector<'a> { *is_unconstrained, generics, parameters, - body, - where_clause, + body.clone(), + where_clause.clone(), return_type, )); unresolved_functions.push_fn( @@ -1073,6 +1079,20 @@ pub fn collect_struct( Some((id, unresolved)) } +#[allow(clippy::too_many_arguments)] +pub fn collect_enum( + _interner: &mut NodeInterner, + _def_map: &mut CrateDefMap, + _usage_tracker: &mut UsageTracker, + _enum_definition: Documented, + _file_id: FileId, + _module_id: LocalModuleId, + _krate: CrateId, + _definition_errors: &mut [(CompilationError, FileId)], +) -> Option<(StructId, UnresolvedEnum)> { + todo!("Implement collect_enum") +} + pub fn collect_impl( interner: &mut NodeInterner, items: &mut CollectedItems, @@ -1222,7 +1242,12 @@ pub(crate) fn collect_trait_impl_items( for item in std::mem::take(&mut trait_impl.items) { match item.item.kind { - TraitImplItemKind::Function(impl_method) => { + TraitImplItemKind::Function(mut impl_method) => { + // Set the impl method visibility as temporarily private. + // Eventually when we find out what trait is this impl for we'll set it + // to the trait's visibility. + impl_method.def.visibility = ItemVisibility::Private; + let func_id = interner.push_empty_fn(); let location = Location::new(impl_method.span(), file_id); interner.push_function(func_id, &impl_method.def, module, location); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/errors.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/errors.rs index cafbc670e32..1582e297144 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/errors.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_collector/errors.rs @@ -183,7 +183,7 @@ impl<'a> From<&'a DefCollectorErrorKind> for Diagnostic { } DefCollectorErrorKind::PathResolutionError(error) => error.into(), DefCollectorErrorKind::CannotReexportItemWithLessVisibility{item_name, desired_visibility} => { - Diagnostic::simple_warning( + Diagnostic::simple_error( format!("cannot re-export {item_name} because it has less visibility than this use statement"), format!("consider marking {item_name} as {desired_visibility}"), item_name.span()) diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_map/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_map/mod.rs index d9d6e150a7a..f7fc6ca08ea 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_map/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_map/mod.rs @@ -1,4 +1,4 @@ -use crate::graph::CrateId; +use crate::graph::{CrateGraph, CrateId}; use crate::hir::def_collector::dc_crate::{CompilationError, DefCollector}; use crate::hir::Context; use crate::node_interner::{FuncId, GlobalId, NodeInterner, StructId}; @@ -8,7 +8,7 @@ use crate::token::{FunctionAttribute, SecondaryAttribute, TestScope}; use fm::{FileId, FileManager}; use noirc_arena::{Arena, Index}; use noirc_errors::Location; -use std::collections::{BTreeMap, HashMap}; +use std::collections::{BTreeMap, HashMap, HashSet}; mod module_def; pub use module_def::*; mod item_scope; @@ -78,7 +78,7 @@ impl CrateDefMap { crate_id: CrateId, context: &mut Context, debug_comptime_in_file: Option<&str>, - error_on_unused_imports: bool, + pedantic_solving: bool, ) -> Vec<(CompilationError, FileId)> { // Check if this Crate has already been compiled // XXX: There is probably a better alternative for this. @@ -121,7 +121,7 @@ impl CrateDefMap { ast, root_file_id, debug_comptime_in_file, - error_on_unused_imports, + pedantic_solving, )); errors.extend( @@ -159,6 +159,10 @@ impl CrateDefMap { self.modules[module_id.0].location.file } + pub fn file_ids(&self) -> HashSet { + self.modules.iter().map(|(_, module_data)| module_data.location.file).collect() + } + /// Go through all modules in this crate, and find all functions in /// each module with the #[test] attribute pub fn get_all_test_functions<'a>( @@ -314,6 +318,31 @@ impl CrateDefMap { } } +pub fn fully_qualified_module_path( + def_maps: &DefMaps, + crate_graph: &CrateGraph, + crate_id: &CrateId, + module_id: ModuleId, +) -> String { + let child_id = module_id.local_id.0; + + let def_map = + def_maps.get(&module_id.krate).expect("The local crate should be analyzed already"); + + let module = &def_map.modules()[module_id.local_id.0]; + + let module_path = def_map.get_module_path_with_separator(child_id, module.parent, "::"); + + if &module_id.krate == crate_id { + module_path + } else { + let crates = crate_graph + .find_dependencies(crate_id, &module_id.krate) + .expect("The module was supposed to be defined in a dependency"); + crates.join("::") + "::" + &module_path + } +} + /// Specifies a contract function and extra metadata that /// one can use when processing a contract function. /// diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_map/namespace.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_map/namespace.rs index a600d98dd8b..255f5c14a84 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/def_map/namespace.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/def_map/namespace.rs @@ -2,7 +2,7 @@ use super::ModuleDefId; use crate::ast::ItemVisibility; // This works exactly the same as in r-a, just simplified -#[derive(Debug, PartialEq, Eq, Copy, Clone)] +#[derive(Debug, PartialEq, Eq, Copy, Clone, Default)] pub struct PerNs { pub types: Option<(ModuleDefId, ItemVisibility, bool)>, pub values: Option<(ModuleDefId, ItemVisibility, bool)>, diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/mod.rs index 2bd1a852f05..b231f8c9698 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/mod.rs @@ -14,12 +14,12 @@ use crate::parser::ParserError; use crate::usage_tracker::UsageTracker; use crate::{Generics, Kind, ParsedModule, ResolvedGeneric, TypeVariable}; use def_collector::dc_crate::CompilationError; -use def_map::{Contract, CrateDefMap}; +use def_map::{fully_qualified_module_path, Contract, CrateDefMap}; use fm::{FileId, FileManager}; use iter_extended::vecmap; use noirc_errors::Location; use std::borrow::Cow; -use std::collections::{BTreeMap, HashMap}; +use std::collections::{BTreeMap, HashMap, HashSet}; use std::path::PathBuf; use std::rc::Rc; @@ -152,56 +152,7 @@ impl Context<'_, '_> { /// For example, if you project contains a `main.nr` and `foo.nr` and you provide the `main_crate_id` and the /// `bar_struct_id` where the `Bar` struct is inside `foo.nr`, this function would return `foo::Bar` as a [String]. pub fn fully_qualified_struct_path(&self, crate_id: &CrateId, id: StructId) -> String { - let module_id = id.module_id(); - let child_id = module_id.local_id.0; - let def_map = - self.def_map(&module_id.krate).expect("The local crate should be analyzed already"); - - let module = self.module(module_id); - - let module_path = def_map.get_module_path_with_separator(child_id, module.parent, "::"); - - if &module_id.krate == crate_id { - module_path - } else { - let crates = self - .find_dependencies(crate_id, &module_id.krate) - .expect("The Struct was supposed to be defined in a dependency"); - crates.join("::") + "::" + &module_path - } - } - - /// Tries to find the requested crate in the current one's dependencies, - /// otherwise walks down the crate dependency graph from crate_id until we reach it. - /// This is needed in case a library (lib1) re-export a structure defined in another library (lib2) - /// In that case, we will get [lib1,lib2] when looking for a struct defined in lib2, - /// re-exported by lib1 and used by the main crate. - /// Returns the path from crate_id to target_crate_id - fn find_dependencies( - &self, - crate_id: &CrateId, - target_crate_id: &CrateId, - ) -> Option> { - self.crate_graph[crate_id] - .dependencies - .iter() - .find_map(|dep| { - if &dep.crate_id == target_crate_id { - Some(vec![dep.name.to_string()]) - } else { - None - } - }) - .or_else(|| { - self.crate_graph[crate_id].dependencies.iter().find_map(|dep| { - if let Some(mut path) = self.find_dependencies(&dep.crate_id, target_crate_id) { - path.insert(0, dep.name.to_string()); - Some(path) - } else { - None - } - }) - }) + fully_qualified_module_path(&self.def_maps, &self.crate_graph, crate_id, id.module_id()) } pub fn function_meta(&self, func_id: &FuncId) -> &FuncMeta { @@ -301,6 +252,10 @@ impl Context<'_, '_> { }) } + pub fn crate_files(&self, crate_id: &CrateId) -> HashSet { + self.def_maps.get(crate_id).map(|def_map| def_map.file_ids()).unwrap_or_default() + } + /// Activates LSP mode, which will track references for all definitions. pub fn activate_lsp_mode(&mut self) { self.def_interner.lsp_mode = true; diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/errors.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/errors.rs index 774836f8992..77ba76a0595 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/errors.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/errors.rs @@ -5,10 +5,13 @@ use thiserror::Error; use crate::{ ast::{Ident, UnsupportedNumericGenericType}, - hir::{comptime::InterpreterError, type_check::TypeCheckError}, + hir::{ + comptime::{InterpreterError, Value}, + type_check::TypeCheckError, + }, parser::ParserError, usage_tracker::UnusedItem, - Type, + Kind, Type, }; use super::import::PathResolutionError; @@ -95,12 +98,22 @@ pub enum ResolverError { DependencyCycle { span: Span, item: String, cycle: String }, #[error("break/continue are only allowed in unconstrained functions")] JumpInConstrainedFn { is_break: bool, span: Span }, + #[error("loop is only allowed in unconstrained functions")] + LoopInConstrainedFn { span: Span }, #[error("break/continue are only allowed within loops")] JumpOutsideLoop { is_break: bool, span: Span }, #[error("Only `comptime` globals can be mutable")] MutableGlobal { span: Span }, #[error("Globals must have a specified type")] UnspecifiedGlobalType { span: Span, expected_type: Type }, + #[error("Global failed to evaluate")] + UnevaluatedGlobalType { span: Span }, + #[error("Globals used in a type position must be non-negative")] + NegativeGlobalType { span: Span, global_value: Value }, + #[error("Globals used in a type position must be integers")] + NonIntegralGlobalType { span: Span, global_value: Value }, + #[error("Global value `{global_value}` is larger than its kind's maximum value")] + GlobalLargerThanKind { span: Span, global_value: FieldElement, kind: Kind }, #[error("Self-referential structs are not supported")] SelfReferentialStruct { span: Span }, #[error("#[no_predicates] attribute is only allowed on constrained functions")] @@ -159,6 +172,8 @@ pub enum ResolverError { span: Span, missing_trait_location: Location, }, + #[error("`loop` statements are not yet implemented")] + LoopNotYetSupported { span: Span }, } impl ResolverError { @@ -421,6 +436,13 @@ impl<'a> From<&'a ResolverError> for Diagnostic { *span, ) }, + ResolverError::LoopInConstrainedFn { span } => { + Diagnostic::simple_error( + "loop is only allowed in unconstrained functions".into(), + "Constrained code must always have a known number of loop iterations".into(), + *span, + ) + }, ResolverError::JumpOutsideLoop { is_break, span } => { let item = if *is_break { "break" } else { "continue" }; Diagnostic::simple_error( @@ -443,6 +465,34 @@ impl<'a> From<&'a ResolverError> for Diagnostic { *span, ) }, + ResolverError::UnevaluatedGlobalType { span } => { + Diagnostic::simple_error( + "Global failed to evaluate".to_string(), + String::new(), + *span, + ) + } + ResolverError::NegativeGlobalType { span, global_value } => { + Diagnostic::simple_error( + "Globals used in a type position must be non-negative".to_string(), + format!("But found value `{global_value:?}`"), + *span, + ) + } + ResolverError::NonIntegralGlobalType { span, global_value } => { + Diagnostic::simple_error( + "Globals used in a type position must be integers".to_string(), + format!("But found value `{global_value:?}`"), + *span, + ) + } + ResolverError::GlobalLargerThanKind { span, global_value, kind } => { + Diagnostic::simple_error( + format!("Global value `{global_value}` is larger than its kind's maximum value"), + format!("Global's kind inferred to be `{kind}`"), + *span, + ) + } ResolverError::SelfReferentialStruct { span } => { Diagnostic::simple_error( "Self-referential structs are not supported".into(), @@ -568,7 +618,7 @@ impl<'a> From<&'a ResolverError> for Diagnostic { }, ResolverError::UnsupportedNumericGenericType(err) => err.into(), ResolverError::TypeIsMorePrivateThenItem { typ, item, span } => { - Diagnostic::simple_warning( + Diagnostic::simple_error( format!("Type `{typ}` is more private than item `{item}`"), String::new(), *span, @@ -603,6 +653,13 @@ impl<'a> From<&'a ResolverError> for Diagnostic { diagnostic.add_secondary_with_file(format!("required by this bound in `{impl_trait}"), missing_trait_location.span, missing_trait_location.file); diagnostic }, + ResolverError::LoopNotYetSupported { span } => { + Diagnostic::simple_error( + "`loop` statements are not yet implemented".to_string(), + String::new(), + *span) + + } } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/import.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/import.rs index 376b85bfbd9..11b694aa61b 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/import.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/import.rs @@ -1,3 +1,4 @@ +use iter_extended::vecmap; use noirc_errors::{CustomDiagnostic, Span}; use thiserror::Error; @@ -48,6 +49,12 @@ pub enum PathResolutionError { TurbofishNotAllowedOnItem { item: String, span: Span }, #[error("{ident} is a {kind}, not a module")] NotAModule { ident: Ident, kind: &'static str }, + #[error("trait `{trait_name}` which provides `{ident}` is implemented but not in scope, please import it")] + TraitMethodNotInScope { ident: Ident, trait_name: String }, + #[error("Could not resolve '{ident}' in path")] + UnresolvedWithPossibleTraitsToImport { ident: Ident, traits: Vec }, + #[error("Multiple applicable items in scope")] + MultipleTraitsInScope { ident: Ident, traits: Vec }, } #[derive(Debug)] @@ -85,6 +92,28 @@ impl<'a> From<&'a PathResolutionError> for CustomDiagnostic { PathResolutionError::NotAModule { ident, kind: _ } => { CustomDiagnostic::simple_error(error.to_string(), String::new(), ident.span()) } + PathResolutionError::TraitMethodNotInScope { ident, .. } => { + CustomDiagnostic::simple_warning(error.to_string(), String::new(), ident.span()) + } + PathResolutionError::UnresolvedWithPossibleTraitsToImport { ident, traits } => { + let traits = vecmap(traits, |trait_name| format!("`{}`", trait_name)); + CustomDiagnostic::simple_error( + error.to_string(), + format!("The following traits which provide `{ident}` are implemented but not in scope: {}", traits.join(", ")), + ident.span(), + ) + } + PathResolutionError::MultipleTraitsInScope { ident, traits } => { + let traits = vecmap(traits, |trait_name| format!("`{}`", trait_name)); + CustomDiagnostic::simple_error( + error.to_string(), + format!( + "All these trait which provide `{ident}` are implemented and in scope: {}", + traits.join(", ") + ), + ident.span(), + ) + } } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/visibility.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/visibility.rs index c2fe887fe62..557f799df89 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/visibility.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/visibility.rs @@ -1,5 +1,5 @@ use crate::graph::CrateId; -use crate::node_interner::{FuncId, NodeInterner, StructId}; +use crate::node_interner::{FuncId, NodeInterner, StructId, TraitId}; use crate::Type; use std::collections::BTreeMap; @@ -79,26 +79,47 @@ pub fn struct_member_is_visible( visibility: ItemVisibility, current_module_id: ModuleId, def_maps: &BTreeMap, +) -> bool { + type_member_is_visible(struct_id.module_id(), visibility, current_module_id, def_maps) +} + +pub fn trait_member_is_visible( + trait_id: TraitId, + visibility: ItemVisibility, + current_module_id: ModuleId, + def_maps: &BTreeMap, +) -> bool { + type_member_is_visible(trait_id.0, visibility, current_module_id, def_maps) +} + +fn type_member_is_visible( + type_module_id: ModuleId, + visibility: ItemVisibility, + current_module_id: ModuleId, + def_maps: &BTreeMap, ) -> bool { match visibility { ItemVisibility::Public => true, ItemVisibility::PublicCrate => { - struct_id.parent_module_id(def_maps).krate == current_module_id.krate + let type_parent_module_id = + type_module_id.parent(def_maps).expect("Expected parent module to exist"); + type_parent_module_id.krate == current_module_id.krate } ItemVisibility::Private => { - let struct_parent_module_id = struct_id.parent_module_id(def_maps); - if struct_parent_module_id.krate != current_module_id.krate { + let type_parent_module_id = + type_module_id.parent(def_maps).expect("Expected parent module to exist"); + if type_parent_module_id.krate != current_module_id.krate { return false; } - if struct_parent_module_id.local_id == current_module_id.local_id { + if type_parent_module_id.local_id == current_module_id.local_id { return true; } let def_map = &def_maps[¤t_module_id.krate]; module_descendent_of_target( def_map, - struct_parent_module_id.local_id, + type_parent_module_id.local_id, current_module_id.local_id, ) } @@ -115,35 +136,48 @@ pub fn method_call_is_visible( let modifiers = interner.function_modifiers(&func_id); match modifiers.visibility { ItemVisibility::Public => true, - ItemVisibility::PublicCrate => { - if object_type.is_primitive() { - current_module.krate.is_stdlib() - } else { - interner.function_module(func_id).krate == current_module.krate + ItemVisibility::PublicCrate | ItemVisibility::Private => { + let func_meta = interner.function_meta(&func_id); + + if let Some(trait_id) = func_meta.trait_id { + return trait_member_is_visible( + trait_id, + modifiers.visibility, + current_module, + def_maps, + ); } - } - ItemVisibility::Private => { + + if let Some(trait_impl_id) = func_meta.trait_impl { + let trait_impl = interner.get_trait_implementation(trait_impl_id); + return trait_member_is_visible( + trait_impl.borrow().trait_id, + modifiers.visibility, + current_module, + def_maps, + ); + } + + if let Some(struct_id) = func_meta.struct_id { + return struct_member_is_visible( + struct_id, + modifiers.visibility, + current_module, + def_maps, + ); + } + if object_type.is_primitive() { let func_module = interner.function_module(func_id); - item_in_module_is_visible( + return item_in_module_is_visible( def_maps, current_module, func_module, modifiers.visibility, - ) - } else { - let func_meta = interner.function_meta(&func_id); - if let Some(struct_id) = func_meta.struct_id { - struct_member_is_visible( - struct_id, - modifiers.visibility, - current_module, - def_maps, - ) - } else { - true - } + ); } + + true } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/scope/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/scope/mod.rs index 1a9087a7408..ebb271b86b7 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/scope/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/scope/mod.rs @@ -25,10 +25,10 @@ It's not implemented yet, because nothing has been benched pub struct Scope(pub HashMap); impl Scope { - pub fn find(&mut self, key: &Q) -> Option<&mut V> + pub fn find(&mut self, key: &Q) -> Option<&mut V> where K: std::borrow::Borrow, - Q: std::hash::Hash + Eq, + Q: ?Sized + std::hash::Hash + Eq, { self.0.get_mut(key) } @@ -75,10 +75,10 @@ impl ScopeTree { // Recursively search for a key in the scope tree. // Returns the value if found, along with the index it was found at. - pub fn find(&mut self, key: &Q) -> Option<(&mut V, usize)> + pub fn find(&mut self, key: &Q) -> Option<(&mut V, usize)> where K: std::borrow::Borrow, - Q: std::hash::Hash + Eq, + Q: ?Sized + std::hash::Hash + Eq, { for (i, scope) in self.0.iter_mut().enumerate().rev() { if let Some(value_found) = scope.find(key) { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/type_check/errors.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/type_check/errors.rs index 15b8d50c78b..d29e1aa4339 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/type_check/errors.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/type_check/errors.rs @@ -66,9 +66,6 @@ pub enum TypeCheckError { from_value: FieldElement, span: Span, }, - // TODO(https://github.com/noir-lang/noir/issues/6238): implement handling for larger types - #[error("Expected type {expected_kind} when evaluating globals, but found {expr_kind} (this warning may become an error in the future)")] - EvaluatedGlobalIsntU32 { expected_kind: String, expr_kind: String, expr_span: Span }, #[error("Expected {expected:?} found {found:?}")] ArityMisMatch { expected: usize, found: usize, span: Span }, #[error("Return type in a function cannot be public")] @@ -208,6 +205,10 @@ pub enum TypeCheckError { CyclicType { typ: Type, span: Span }, #[error("Type annotations required before indexing this array or slice")] TypeAnnotationsNeededForIndex { span: Span }, + #[error("Unnecessary `unsafe` block")] + UnnecessaryUnsafeBlock { span: Span }, + #[error("Unnecessary `unsafe` block")] + NestedUnsafeBlock { span: Span }, } #[derive(Debug, Clone, PartialEq, Eq)] @@ -275,15 +276,6 @@ impl<'a> From<&'a TypeCheckError> for Diagnostic { *span, ) } - // TODO(https://github.com/noir-lang/noir/issues/6238): implement - // handling for larger types - TypeCheckError::EvaluatedGlobalIsntU32 { expected_kind, expr_kind, expr_span } => { - Diagnostic::simple_warning( - format!("Expected type {expected_kind} when evaluating globals, but found {expr_kind} (this warning may become an error in the future)"), - String::new(), - *expr_span, - ) - } TypeCheckError::TraitMethodParameterTypeMismatch { method_name, expected_typ, actual_typ, parameter_index, parameter_span } => { Diagnostic::simple_error( format!("Parameter #{parameter_index} of method `{method_name}` must be of type {expected_typ}, not {actual_typ}"), @@ -504,10 +496,10 @@ impl<'a> From<&'a TypeCheckError> for Diagnostic { Diagnostic::simple_error(msg.to_string(), "".to_string(), *span) }, TypeCheckError::Unsafe { span } => { - Diagnostic::simple_warning(error.to_string(), String::new(), *span) + Diagnostic::simple_error(error.to_string(), String::new(), *span) } TypeCheckError::UnsafeFn { span } => { - Diagnostic::simple_warning(error.to_string(), String::new(), *span) + Diagnostic::simple_error(error.to_string(), String::new(), *span) } TypeCheckError::UnspecifiedType { span } => { Diagnostic::simple_error(error.to_string(), String::new(), *span) @@ -529,6 +521,20 @@ impl<'a> From<&'a TypeCheckError> for Diagnostic { *span, ) }, + TypeCheckError::UnnecessaryUnsafeBlock { span } => { + Diagnostic::simple_warning( + "Unnecessary `unsafe` block".into(), + "".into(), + *span, + ) + }, + TypeCheckError::NestedUnsafeBlock { span } => { + Diagnostic::simple_warning( + "Unnecessary `unsafe` block".into(), + "Because it's nested inside another `unsafe` block".into(), + *span, + ) + }, } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/expr.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/expr.rs index e243fc88cff..5ac228a56d6 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/expr.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/expr.rs @@ -209,14 +209,14 @@ pub enum HirMethodReference { /// Or a method can come from a Trait impl block, in which case /// the actual function called will depend on the instantiated type, /// which can be only known during monomorphization. - TraitMethodId(TraitMethodId, TraitGenerics), + TraitMethodId(TraitMethodId, TraitGenerics, bool /* assumed */), } impl HirMethodReference { pub fn func_id(&self, interner: &NodeInterner) -> Option { match self { HirMethodReference::FuncId(func_id) => Some(*func_id), - HirMethodReference::TraitMethodId(method_id, _) => { + HirMethodReference::TraitMethodId(method_id, _, _) => { let id = interner.trait_method_id(*method_id); match &interner.try_definition(id)?.kind { DefinitionKind::Function(func_id) => Some(*func_id), @@ -225,28 +225,19 @@ impl HirMethodReference { } } } -} -impl HirMethodCallExpression { - /// Converts a method call into a function call - /// - /// Returns ((func_var_id, func_var), call_expr) - pub fn into_function_call( - mut self, - method: HirMethodReference, + pub fn into_function_id_and_name( + self, object_type: Type, - is_macro_call: bool, + generics: Option>, location: Location, interner: &mut NodeInterner, - ) -> ((ExprId, HirIdent), HirCallExpression) { - let mut arguments = vec![self.object]; - arguments.append(&mut self.arguments); - - let (id, impl_kind) = match method { + ) -> (ExprId, HirIdent) { + let (id, impl_kind) = match self { HirMethodReference::FuncId(func_id) => { (interner.function_definition_id(func_id), ImplKind::NotATraitMethod) } - HirMethodReference::TraitMethodId(method_id, trait_generics) => { + HirMethodReference::TraitMethodId(method_id, trait_generics, assumed) => { let id = interner.trait_method_id(method_id); let constraint = TraitConstraint { typ: object_type, @@ -256,14 +247,27 @@ impl HirMethodCallExpression { span: location.span, }, }; - (id, ImplKind::TraitMethod(TraitMethod { method_id, constraint, assumed: false })) + + (id, ImplKind::TraitMethod(TraitMethod { method_id, constraint, assumed })) } }; let func_var = HirIdent { location, id, impl_kind }; - let func = interner.push_expr(HirExpression::Ident(func_var.clone(), self.generics)); + let func = interner.push_expr(HirExpression::Ident(func_var.clone(), generics)); interner.push_expr_location(func, location.span, location.file); - let expr = HirCallExpression { func, arguments, location, is_macro_call }; - ((func, func_var), expr) + (func, func_var) + } +} + +impl HirMethodCallExpression { + pub fn into_function_call( + mut self, + func: ExprId, + is_macro_call: bool, + location: Location, + ) -> HirCallExpression { + let mut arguments = vec![self.object]; + arguments.append(&mut self.arguments); + HirCallExpression { func, arguments, location, is_macro_call } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/function.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/function.rs index db6c3507b15..aa04738733f 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/function.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/function.rs @@ -175,12 +175,13 @@ pub enum FunctionBody { impl FuncMeta { /// A stub function does not have a body. This includes Builtin, LowLevel, - /// and Oracle functions in addition to method declarations within a trait. + /// and Oracle functions in addition to method declarations within a trait + /// without a body. /// /// We don't check the return type of these functions since it will always have /// an empty body, and we don't check for unused parameters. pub fn is_stub(&self) -> bool { - self.kind.can_ignore_return_type() || self.trait_id.is_some() + self.kind.can_ignore_return_type() } pub fn function_signature(&self) -> FunctionSignature { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/stmt.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/stmt.rs index 0258cfd8ddb..8a580e735b1 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/stmt.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/stmt.rs @@ -16,6 +16,7 @@ pub enum HirStatement { Constrain(HirConstrainStatement), Assign(HirAssignStatement), For(HirForStatement), + Loop(ExprId), Break, Continue, Expression(ExprId), @@ -31,15 +32,31 @@ pub struct HirLetStatement { pub expression: ExprId, pub attributes: Vec, pub comptime: bool, + pub is_global_let: bool, } impl HirLetStatement { + pub fn new( + pattern: HirPattern, + r#type: Type, + expression: ExprId, + attributes: Vec, + comptime: bool, + is_global_let: bool, + ) -> HirLetStatement { + Self { pattern, r#type, expression, attributes, comptime, is_global_let } + } + pub fn ident(&self) -> HirIdent { match &self.pattern { HirPattern::Identifier(ident) => ident.clone(), _ => panic!("can only fetch hir ident from HirPattern::Identifier"), } } + + pub fn runs_comptime(&self) -> bool { + self.comptime || self.is_global_let + } } #[derive(Debug, Clone)] diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/traits.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/traits.rs index 6fd3c4f7a24..ff0cac027b1 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/traits.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/traits.rs @@ -1,7 +1,7 @@ use iter_extended::vecmap; use rustc_hash::FxHashMap as HashMap; -use crate::ast::{Ident, NoirFunction}; +use crate::ast::{Ident, ItemVisibility, NoirFunction}; use crate::hir::type_check::generics::TraitGenerics; use crate::ResolvedGeneric; use crate::{ @@ -66,6 +66,7 @@ pub struct Trait { pub name: Ident, pub generics: Generics, pub location: Location, + pub visibility: ItemVisibility, /// When resolving the types of Trait elements, all references to `Self` resolve /// to this TypeVariable. Then when we check if the types of trait impl elements @@ -160,6 +161,10 @@ impl Trait { self.where_clause = where_clause; } + pub fn set_visibility(&mut self, visibility: ItemVisibility) { + self.visibility = visibility; + } + pub fn find_method(&self, name: &str) -> Option { for (idx, method) in self.methods.iter().enumerate() { if &method.name == name { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/types.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/types.rs index 2c9a44c079d..4eeec314917 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/types.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/types.rs @@ -97,10 +97,7 @@ pub enum Type { /// A cast (to, from) that's checked at monomorphization. /// /// Simplifications on arithmetic generics are only allowed on the LHS. - CheckedCast { - from: Box, - to: Box, - }, + CheckedCast { from: Box, to: Box }, /// A functions with arguments, a return type and environment. /// the environment should be `Unit` by default, @@ -132,7 +129,13 @@ pub enum Type { /// The type of quoted code in macros. This is always a comptime-only type Quoted(QuotedType), - InfixExpr(Box, BinaryTypeOperator, Box), + /// An infix expression in the form `lhs * rhs`. + /// + /// The `inversion` bool keeps track of whether this expression came from + /// an expression like `4 = a / b` which was transformed to `a = 4 / b` + /// so that if at some point a infix expression `b * (4 / b)` is created, + /// it could be simplified back to `4`. + InfixExpr(Box, BinaryTypeOperator, Box, bool /* inversion */), /// The result of some type error. Remembering type errors as their own type variant lets /// us avoid issuing repeat type errors for the same item. For example, a lambda with @@ -226,6 +229,10 @@ impl Kind { // Kind::Integer unifies with Kind::IntegerOrField (Kind::Integer | Kind::IntegerOrField, Kind::Integer | Kind::IntegerOrField) => true, + // Kind::IntegerOrField unifies with Kind::Numeric(_) + (Kind::IntegerOrField, Kind::Numeric(_typ)) + | (Kind::Numeric(_typ), Kind::IntegerOrField) => true, + // Kind::Numeric unifies along its Type argument (Kind::Numeric(lhs), Kind::Numeric(rhs)) => { let mut bindings = TypeBindings::new(); @@ -255,7 +262,8 @@ impl Kind { match self { Kind::IntegerOrField => Some(Type::default_int_or_field_type()), Kind::Integer => Some(Type::default_int_type()), - Kind::Any | Kind::Normal | Kind::Numeric(..) => None, + Kind::Numeric(typ) => Some(*typ.clone()), + Kind::Any | Kind::Normal => None, } } @@ -267,7 +275,7 @@ impl Kind { } /// Ensure the given value fits in self.integral_maximum_size() - fn ensure_value_fits( + pub(crate) fn ensure_value_fits( &self, value: FieldElement, span: Span, @@ -307,6 +315,7 @@ pub enum QuotedType { Type, TypedExpr, StructDefinition, + EnumDefinition, TraitConstraint, TraitDefinition, TraitImpl, @@ -900,7 +909,7 @@ impl std::fmt::Display for Type { write!(f, "&mut {element}") } Type::Quoted(quoted) => write!(f, "{}", quoted), - Type::InfixExpr(lhs, op, rhs) => { + Type::InfixExpr(lhs, op, rhs, _) => { let this = self.canonicalize_checked(); // Prevent infinite recursion @@ -950,6 +959,7 @@ impl std::fmt::Display for QuotedType { QuotedType::Type => write!(f, "Type"), QuotedType::TypedExpr => write!(f, "TypedExpr"), QuotedType::StructDefinition => write!(f, "StructDefinition"), + QuotedType::EnumDefinition => write!(f, "EnumDefinition"), QuotedType::TraitDefinition => write!(f, "TraitDefinition"), QuotedType::TraitConstraint => write!(f, "TraitConstraint"), QuotedType::TraitImpl => write!(f, "TraitImpl"), @@ -1141,7 +1151,7 @@ impl Type { .into_iter() .all(|(_, field)| field.is_valid_for_program_input()), - Type::InfixExpr(lhs, _, rhs) => { + Type::InfixExpr(lhs, _, rhs, _) => { lhs.is_valid_for_program_input() && rhs.is_valid_for_program_input() } } @@ -1302,7 +1312,7 @@ impl Type { TypeBinding::Bound(ref typ) => typ.kind(), TypeBinding::Unbound(_, ref type_var_kind) => type_var_kind.clone(), }, - Type::InfixExpr(lhs, _op, rhs) => lhs.infix_kind(rhs), + Type::InfixExpr(lhs, _op, rhs, _) => lhs.infix_kind(rhs), Type::Alias(def, generics) => def.borrow().get_type(generics).kind(), // This is a concrete FieldElement, not an IntegerOrField Type::FieldElement @@ -1335,6 +1345,48 @@ impl Type { } } + /// Creates an `InfixExpr`. + pub fn infix_expr(lhs: Box, op: BinaryTypeOperator, rhs: Box) -> Type { + Self::new_infix_expr(lhs, op, rhs, false) + } + + /// Creates an `InfixExpr` that results from the compiler trying to unify something like + /// `4 = a * b` into `a = 4 / b` (where `4 / b` is the "inverted" expression). + pub fn inverted_infix_expr(lhs: Box, op: BinaryTypeOperator, rhs: Box) -> Type { + Self::new_infix_expr(lhs, op, rhs, true) + } + + pub fn new_infix_expr( + lhs: Box, + op: BinaryTypeOperator, + rhs: Box, + inversion: bool, + ) -> Type { + // If an InfixExpr like this is tried to be created: + // + // a * (b / a) + // + // where `b / a` resulted from the compiler creating an inverted InfixExpr from a previous + // unification (that is, the compiler had `b = a / y` and ended up doing `y = b / a` where + // `y` is `rhs` here) then we can simplify this to just `b` because there wasn't an actual + // division in the original expression, so multiplying it back is just going back to the + // original `y` + if let Type::InfixExpr(rhs_lhs, rhs_op, rhs_rhs, true) = &*rhs { + if op.approx_inverse() == Some(*rhs_op) && lhs == *rhs_rhs { + return *rhs_lhs.clone(); + } + } + + // Same thing but on the other side. + if let Type::InfixExpr(lhs_lhs, lhs_op, lhs_rhs, true) = &*lhs { + if op.approx_inverse() == Some(*lhs_op) && rhs == *lhs_rhs { + return *lhs_lhs.clone(); + } + } + + Self::InfixExpr(lhs, op, rhs, inversion) + } + /// Returns the number of field elements required to represent the type once encoded. pub fn field_count(&self, location: &Location) -> u32 { match self { @@ -1692,7 +1744,7 @@ impl Type { elem_a.try_unify(elem_b, bindings) } - (InfixExpr(lhs_a, op_a, rhs_a), InfixExpr(lhs_b, op_b, rhs_b)) => { + (InfixExpr(lhs_a, op_a, rhs_a, _), InfixExpr(lhs_b, op_b, rhs_b, _)) => { if op_a == op_b { // We need to preserve the original bindings since if syntactic equality // fails we fall back to other equality strategies. @@ -1719,14 +1771,15 @@ impl Type { } else { Err(UnificationError) } - } else if let InfixExpr(lhs, op, rhs) = other { + } else if let InfixExpr(lhs, op, rhs, _) = other { if let Some(inverse) = op.approx_inverse() { // Handle cases like `4 = a + b` by trying to solve to `a = 4 - b` - let new_type = InfixExpr( + let new_type = Type::inverted_infix_expr( Box::new(Constant(*value, kind.clone())), inverse, rhs.clone(), ); + new_type.try_unify(lhs, bindings)?; Ok(()) } else { @@ -1857,7 +1910,7 @@ impl Type { if let (Type::Array(_size, element1), Type::Slice(element2)) = (&this, &target) { // We can only do the coercion if the `as_slice` method exists. // This is usually true, but some tests don't have access to the standard library. - if let Some(as_slice) = interner.lookup_primitive_method(&this, "as_slice", true) { + if let Some(as_slice) = interner.lookup_direct_method(&this, "as_slice", true) { // Still have to ensure the element types match. // Don't need to issue an error here if not, it will be done in unify_with_coercions let mut bindings = TypeBindings::new(); @@ -1932,7 +1985,7 @@ impl Type { }) } } - Type::InfixExpr(lhs, op, rhs) => { + Type::InfixExpr(lhs, op, rhs, _) => { let infix_kind = lhs.infix_kind(&rhs); if kind.unifies(&infix_kind) { let lhs_value = lhs.evaluate_to_field_element_helper( @@ -2210,6 +2263,7 @@ impl Type { Type::NamedGeneric(binding, _) | Type::TypeVariable(binding) => { substitute_binding(binding) } + // Do not substitute_helper fields, it can lead to infinite recursion // and we should not match fields when type checking anyway. Type::Struct(fields, args) => { @@ -2261,10 +2315,10 @@ impl Type { }); Type::TraitAsType(*s, name.clone(), TraitGenerics { ordered, named }) } - Type::InfixExpr(lhs, op, rhs) => { + Type::InfixExpr(lhs, op, rhs, inversion) => { let lhs = lhs.substitute_helper(type_bindings, substitute_bound_typevars); let rhs = rhs.substitute_helper(type_bindings, substitute_bound_typevars); - Type::InfixExpr(Box::new(lhs), *op, Box::new(rhs)) + Type::InfixExpr(Box::new(lhs), *op, Box::new(rhs), *inversion) } Type::FieldElement @@ -2314,7 +2368,7 @@ impl Type { || env.occurs(target_id) } Type::MutableReference(element) => element.occurs(target_id), - Type::InfixExpr(lhs, _op, rhs) => lhs.occurs(target_id) || rhs.occurs(target_id), + Type::InfixExpr(lhs, _op, rhs, _) => lhs.occurs(target_id) || rhs.occurs(target_id), Type::FieldElement | Type::Integer(_, _) @@ -2383,10 +2437,10 @@ impl Type { }); TraitAsType(*s, name.clone(), TraitGenerics { ordered, named }) } - InfixExpr(lhs, op, rhs) => { + InfixExpr(lhs, op, rhs, inversion) => { let lhs = lhs.follow_bindings(); let rhs = rhs.follow_bindings(); - InfixExpr(Box::new(lhs), *op, Box::new(rhs)) + InfixExpr(Box::new(lhs), *op, Box::new(rhs), *inversion) } // Expect that this function should only be called on instantiated types @@ -2496,7 +2550,7 @@ impl Type { } Type::MutableReference(elem) => elem.replace_named_generics_with_type_variables(), Type::Forall(_, typ) => typ.replace_named_generics_with_type_variables(), - Type::InfixExpr(lhs, _op, rhs) => { + Type::InfixExpr(lhs, _op, rhs, _) => { lhs.replace_named_generics_with_type_variables(); rhs.replace_named_generics_with_type_variables(); } @@ -2538,7 +2592,7 @@ impl Type { TypeBinding::Unbound(_, kind) => kind.integral_maximum_size(), }, Type::MutableReference(typ) => typ.integral_maximum_size(), - Type::InfixExpr(lhs, _op, rhs) => lhs.infix_kind(rhs).integral_maximum_size(), + Type::InfixExpr(lhs, _op, rhs, _) => lhs.infix_kind(rhs).integral_maximum_size(), Type::Constant(_, kind) => kind.integral_maximum_size(), Type::Array(..) @@ -2821,7 +2875,7 @@ impl std::fmt::Debug for Type { write!(f, "&mut {element:?}") } Type::Quoted(quoted) => write!(f, "{}", quoted), - Type::InfixExpr(lhs, op, rhs) => write!(f, "({lhs:?} {op} {rhs:?})"), + Type::InfixExpr(lhs, op, rhs, _) => write!(f, "({lhs:?} {op} {rhs:?})"), } } } @@ -2907,7 +2961,7 @@ impl std::hash::Hash for Type { Type::CheckedCast { to, .. } => to.hash(state), Type::Constant(value, _) => value.hash(state), Type::Quoted(typ) => typ.hash(state), - Type::InfixExpr(lhs, op, rhs) => { + Type::InfixExpr(lhs, op, rhs, _) => { lhs.hash(state); op.hash(state); rhs.hash(state); @@ -2976,7 +3030,7 @@ impl PartialEq for Type { lhs == rhs && lhs_kind == rhs_kind } (Quoted(lhs), Quoted(rhs)) => lhs == rhs, - (InfixExpr(l_lhs, l_op, l_rhs), InfixExpr(r_lhs, r_op, r_rhs)) => { + (InfixExpr(l_lhs, l_op, l_rhs, _), InfixExpr(r_lhs, r_op, r_rhs, _)) => { l_lhs == r_lhs && l_op == r_op && l_rhs == r_rhs } // Special case: we consider unbound named generics and type variables to be equal to each diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/types/arithmetic.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/types/arithmetic.rs index 8cdf6f5502c..5750365c62d 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir_def/types/arithmetic.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir_def/types/arithmetic.rs @@ -58,7 +58,7 @@ impl Type { run_simplifications: bool, ) -> Type { match self.follow_bindings() { - Type::InfixExpr(lhs, op, rhs) => { + Type::InfixExpr(lhs, op, rhs, inversion) => { let kind = lhs.infix_kind(&rhs); let dummy_span = Span::default(); // evaluate_to_field_element also calls canonicalize so if we just called @@ -76,7 +76,7 @@ impl Type { let rhs = rhs.canonicalize_helper(found_checked_cast, run_simplifications); if !run_simplifications { - return Type::InfixExpr(Box::new(lhs), op, Box::new(rhs)); + return Type::InfixExpr(Box::new(lhs), op, Box::new(rhs), inversion); } if let Some(result) = Self::try_simplify_non_constants_in_lhs(&lhs, op, &rhs) { @@ -97,7 +97,7 @@ impl Type { return Self::sort_commutative(&lhs, op, &rhs); } - Type::InfixExpr(Box::new(lhs), op, Box::new(rhs)) + Type::InfixExpr(Box::new(lhs), op, Box::new(rhs), inversion) } Type::CheckedCast { from, to } => { let inner_found_checked_cast = true; @@ -131,7 +131,7 @@ impl Type { // Push each non-constant term to `sorted` to sort them. Recur on InfixExprs with the same operator. while let Some(item) = queue.pop() { match item.canonicalize_unchecked() { - Type::InfixExpr(lhs_inner, new_op, rhs_inner) if new_op == op => { + Type::InfixExpr(lhs_inner, new_op, rhs_inner, _) if new_op == op => { queue.push(*lhs_inner); queue.push(*rhs_inner); } @@ -157,18 +157,18 @@ impl Type { // - 1 since `typ` already is set to the first instance for _ in 0..first_type_count - 1 { - typ = Type::InfixExpr(Box::new(typ), op, Box::new(first.0.clone())); + typ = Type::infix_expr(Box::new(typ), op, Box::new(first.0.clone())); } for (rhs, rhs_count) in sorted { for _ in 0..rhs_count { - typ = Type::InfixExpr(Box::new(typ), op, Box::new(rhs.clone())); + typ = Type::infix_expr(Box::new(typ), op, Box::new(rhs.clone())); } } if constant != zero_value { let constant = Type::Constant(constant, lhs.infix_kind(rhs)); - typ = Type::InfixExpr(Box::new(typ), op, Box::new(constant)); + typ = Type::infix_expr(Box::new(typ), op, Box::new(constant)); } typ @@ -192,11 +192,11 @@ impl Type { match lhs.follow_bindings() { Type::CheckedCast { from, to } => { // Apply operation directly to `from` while attempting simplification to `to`. - let from = Type::InfixExpr(from, op, Box::new(rhs.clone())); + let from = Type::infix_expr(from, op, Box::new(rhs.clone())); let to = Self::try_simplify_non_constants_in_lhs(&to, op, rhs)?; Some(Type::CheckedCast { from: Box::new(from), to: Box::new(to) }) } - Type::InfixExpr(l_lhs, l_op, l_rhs) => { + Type::InfixExpr(l_lhs, l_op, l_rhs, _) => { // Note that this is exact, syntactic equality, not unification. // `rhs` is expected to already be in canonical form. if l_op.approx_inverse() != Some(op) @@ -229,11 +229,11 @@ impl Type { match rhs.follow_bindings() { Type::CheckedCast { from, to } => { // Apply operation directly to `from` while attempting simplification to `to`. - let from = Type::InfixExpr(Box::new(lhs.clone()), op, from); + let from = Type::infix_expr(Box::new(lhs.clone()), op, from); let to = Self::try_simplify_non_constants_in_rhs(lhs, op, &to)?; Some(Type::CheckedCast { from: Box::new(from), to: Box::new(to) }) } - Type::InfixExpr(r_lhs, r_op, r_rhs) => { + Type::InfixExpr(r_lhs, r_op, r_rhs, _) => { // `N / (M * N)` should be simplified to `1 / M`, but we only handle // simplifying to `M` in this function. if op == BinaryTypeOperator::Division && r_op == BinaryTypeOperator::Multiplication @@ -268,7 +268,7 @@ impl Type { let dummy_span = Span::default(); let rhs = rhs.evaluate_to_field_element(&kind, dummy_span).ok()?; - let Type::InfixExpr(l_type, l_op, l_rhs) = lhs.follow_bindings() else { + let Type::InfixExpr(l_type, l_op, l_rhs, _) = lhs.follow_bindings() else { return None; }; @@ -302,7 +302,7 @@ impl Type { let result = op.function(l_const, r_const, &lhs.infix_kind(rhs), dummy_span).ok()?; let constant = Type::Constant(result, lhs.infix_kind(rhs)); - Some(Type::InfixExpr(l_type, l_op, Box::new(constant))) + Some(Type::infix_expr(l_type, l_op, Box::new(constant))) } (Multiplication, Division) => { // We need to ensure the result divides evenly to preserve integer division semantics @@ -317,7 +317,7 @@ impl Type { let result = op.function(l_const, r_const, &lhs.infix_kind(rhs), dummy_span).ok()?; let constant = Box::new(Type::Constant(result, lhs.infix_kind(rhs))); - Some(Type::InfixExpr(l_type, l_op, constant)) + Some(Type::infix_expr(l_type, l_op, constant)) } } _ => None, @@ -331,13 +331,14 @@ impl Type { other: &Type, bindings: &mut TypeBindings, ) -> Result<(), UnificationError> { - if let Type::InfixExpr(lhs_a, op_a, rhs_a) = self { + if let Type::InfixExpr(lhs_a, op_a, rhs_a, _) = self { if let Some(inverse) = op_a.approx_inverse() { let kind = lhs_a.infix_kind(rhs_a); let dummy_span = Span::default(); if let Ok(rhs_a_value) = rhs_a.evaluate_to_field_element(&kind, dummy_span) { let rhs_a = Box::new(Type::Constant(rhs_a_value, kind)); - let new_other = Type::InfixExpr(Box::new(other.clone()), inverse, rhs_a); + let new_other = + Type::inverted_infix_expr(Box::new(other.clone()), inverse, rhs_a); let mut tmp_bindings = bindings.clone(); if lhs_a.try_unify(&new_other, &mut tmp_bindings).is_ok() { @@ -348,13 +349,14 @@ impl Type { } } - if let Type::InfixExpr(lhs_b, op_b, rhs_b) = other { + if let Type::InfixExpr(lhs_b, op_b, rhs_b, inversion) = other { if let Some(inverse) = op_b.approx_inverse() { let kind = lhs_b.infix_kind(rhs_b); let dummy_span = Span::default(); if let Ok(rhs_b_value) = rhs_b.evaluate_to_field_element(&kind, dummy_span) { let rhs_b = Box::new(Type::Constant(rhs_b_value, kind)); - let new_self = Type::InfixExpr(Box::new(self.clone()), inverse, rhs_b); + let new_self = + Type::InfixExpr(Box::new(self.clone()), inverse, rhs_b, !inversion); let mut tmp_bindings = bindings.clone(); if new_self.try_unify(lhs_b, &mut tmp_bindings).is_ok() { @@ -384,7 +386,7 @@ mod tests { TypeVariable::unbound(TypeVariableId(0), Kind::u32()), std::rc::Rc::new("N".to_owned()), ); - let n_minus_one = Type::InfixExpr( + let n_minus_one = Type::infix_expr( Box::new(n.clone()), BinaryTypeOperator::Subtraction, Box::new(Type::Constant(FieldElement::one(), Kind::u32())), @@ -392,7 +394,7 @@ mod tests { let checked_cast_n_minus_one = Type::CheckedCast { from: Box::new(n_minus_one.clone()), to: Box::new(n_minus_one) }; - let n_minus_one_plus_one = Type::InfixExpr( + let n_minus_one_plus_one = Type::infix_expr( Box::new(checked_cast_n_minus_one.clone()), BinaryTypeOperator::Addition, Box::new(Type::Constant(FieldElement::one(), Kind::u32())), @@ -405,7 +407,7 @@ mod tests { // We also want to check that if the `CheckedCast` is on the RHS then we'll still be able to canonicalize // the expression `1 + (N - 1)` to `N`. - let one_plus_n_minus_one = Type::InfixExpr( + let one_plus_n_minus_one = Type::infix_expr( Box::new(Type::Constant(FieldElement::one(), Kind::u32())), BinaryTypeOperator::Addition, Box::new(checked_cast_n_minus_one), @@ -423,13 +425,13 @@ mod tests { let x_type = Type::TypeVariable(x_var.clone()); let one = Type::Constant(FieldElement::one(), field_element_kind.clone()); - let lhs = Type::InfixExpr( + let lhs = Type::infix_expr( Box::new(x_type.clone()), BinaryTypeOperator::Addition, Box::new(one.clone()), ); let rhs = - Type::InfixExpr(Box::new(one), BinaryTypeOperator::Addition, Box::new(x_type.clone())); + Type::infix_expr(Box::new(one), BinaryTypeOperator::Addition, Box::new(x_type.clone())); // canonicalize let lhs = lhs.canonicalize(); @@ -546,7 +548,7 @@ mod proptests { 10, // We put up to 10 items per collection |inner| { (inner.clone(), any::(), inner) - .prop_map(|(lhs, op, rhs)| Type::InfixExpr(Box::new(lhs), op, Box::new(rhs))) + .prop_map(|(lhs, op, rhs)| Type::infix_expr(Box::new(lhs), op, Box::new(rhs))) }, ) } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/lexer/lexer.rs b/noir/noir-repo/compiler/noirc_frontend/src/lexer/lexer.rs index a5c4b2cd772..0b7bd0991d9 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/lexer/lexer.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/lexer/lexer.rs @@ -787,7 +787,6 @@ impl<'a> Lexer<'a> { ch => content.push(ch), } } - if depth == 0 { if !content.is_ascii() { let span = Span::from(start..self.position); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/lexer/token.rs b/noir/noir-repo/compiler/noirc_frontend/src/lexer/token.rs index f35515045db..7d11b97ca16 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/lexer/token.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/lexer/token.rs @@ -1019,6 +1019,8 @@ pub enum Keyword { CtString, Dep, Else, + Enum, + EnumDefinition, Expr, Field, Fn, @@ -1030,6 +1032,8 @@ pub enum Keyword { Impl, In, Let, + Loop, + Match, Mod, Module, Mut, @@ -1076,6 +1080,8 @@ impl fmt::Display for Keyword { Keyword::CtString => write!(f, "CtString"), Keyword::Dep => write!(f, "dep"), Keyword::Else => write!(f, "else"), + Keyword::Enum => write!(f, "enum"), + Keyword::EnumDefinition => write!(f, "EnumDefinition"), Keyword::Expr => write!(f, "Expr"), Keyword::Field => write!(f, "Field"), Keyword::Fn => write!(f, "fn"), @@ -1087,6 +1093,8 @@ impl fmt::Display for Keyword { Keyword::Impl => write!(f, "impl"), Keyword::In => write!(f, "in"), Keyword::Let => write!(f, "let"), + Keyword::Loop => write!(f, "loop"), + Keyword::Match => write!(f, "match"), Keyword::Mod => write!(f, "mod"), Keyword::Module => write!(f, "Module"), Keyword::Mut => write!(f, "mut"), @@ -1136,6 +1144,8 @@ impl Keyword { "CtString" => Keyword::CtString, "dep" => Keyword::Dep, "else" => Keyword::Else, + "enum" => Keyword::Enum, + "EnumDefinition" => Keyword::EnumDefinition, "Expr" => Keyword::Expr, "Field" => Keyword::Field, "fn" => Keyword::Fn, @@ -1147,6 +1157,8 @@ impl Keyword { "impl" => Keyword::Impl, "in" => Keyword::In, "let" => Keyword::Let, + "loop" => Keyword::Loop, + "match" => Keyword::Match, "mod" => Keyword::Mod, "Module" => Keyword::Module, "mut" => Keyword::Mut, diff --git a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/ast.rs b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/ast.rs index c9ae3438e42..621eb30e4f8 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/ast.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/ast.rs @@ -1,4 +1,4 @@ -use std::fmt::Display; +use std::{collections::BTreeMap, fmt::Display}; use acvm::FieldElement; use iter_extended::vecmap; @@ -36,6 +36,7 @@ pub enum Expression { Index(Index), Cast(Cast), For(For), + Loop(Box), If(If), Tuple(Vec), ExtractTupleField(Box, usize), @@ -59,6 +60,7 @@ impl Expression { #[derive(Debug, Clone, PartialEq, Eq, Hash)] pub enum Definition { Local(LocalId), + Global(GlobalId), Function(FuncId), Builtin(String), LowLevel(String), @@ -71,6 +73,10 @@ pub enum Definition { #[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)] pub struct LocalId(pub u32); +/// A function ID corresponds directly to an index of `Program::globals` +#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord, Serialize, Deserialize)] +pub struct GlobalId(pub u32); + /// A function ID corresponds directly to an index of `Program::functions` #[derive(Debug, Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)] pub struct FuncId(pub u32); @@ -222,7 +228,9 @@ pub type Parameters = Vec<(LocalId, /*mutable:*/ bool, /*name:*/ String, Type)>; /// Represents how an Acir function should be inlined. /// This type is only relevant for ACIR functions as we do not inline any Brillig functions -#[derive(Default, Clone, Copy, PartialEq, Eq, Debug, Hash, Serialize, Deserialize)] +#[derive( + Default, Clone, Copy, PartialEq, Eq, Debug, Hash, Serialize, Deserialize, PartialOrd, Ord, +)] pub enum InlineType { /// The most basic entry point can expect all its functions to be inlined. /// All function calls are expected to be inlined into a single ACIR. @@ -322,13 +330,14 @@ impl Type { } } -#[derive(Debug, Clone, Hash)] +#[derive(Debug, Clone, Hash, Default)] pub struct Program { pub functions: Vec, pub function_signatures: Vec, pub main_function_signature: FunctionSignature, pub return_location: Option, pub return_visibility: Visibility, + pub globals: BTreeMap, pub debug_variables: DebugVariables, pub debug_functions: DebugFunctions, pub debug_types: DebugTypes, @@ -342,6 +351,7 @@ impl Program { main_function_signature: FunctionSignature, return_location: Option, return_visibility: Visibility, + globals: BTreeMap, debug_variables: DebugVariables, debug_functions: DebugFunctions, debug_types: DebugTypes, @@ -352,6 +362,7 @@ impl Program { main_function_signature, return_location, return_visibility, + globals, debug_variables, debug_functions, debug_types, @@ -370,6 +381,13 @@ impl Program { FuncId(0) } + /// Globals are expected to be generated within a different context than + /// all other functions in the program. Thus, the globals space has the same + /// ID as `main`, although we should never expect a clash in these IDs. + pub fn global_space_id() -> FuncId { + FuncId(0) + } + /// Takes a function body by replacing it with `false` and /// returning the previous value pub fn take_function_body(&mut self, function: FuncId) -> Expression { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/mod.rs index b31a5744d09..191c3937e4b 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/mod.rs @@ -11,7 +11,7 @@ use crate::ast::{FunctionKind, IntegerBitSize, Signedness, UnaryOp, Visibility}; use crate::hir::comptime::InterpreterError; use crate::hir::type_check::{NoMatchingImplFoundError, TypeCheckError}; -use crate::node_interner::{ExprId, ImplSearchErrorKind}; +use crate::node_interner::{ExprId, GlobalValue, ImplSearchErrorKind}; use crate::token::FmtStrFragment; use crate::{ debug::DebugInstrumenter, @@ -25,11 +25,13 @@ use crate::{ Kind, Type, TypeBinding, TypeBindings, }; use acvm::{acir::AcirField, FieldElement}; +use ast::GlobalId; +use fxhash::FxHashMap as HashMap; use iter_extended::{btree_map, try_vecmap, vecmap}; use noirc_errors::Location; use noirc_printable_type::PrintableType; use std::{ - collections::{BTreeMap, HashMap, VecDeque}, + collections::{BTreeMap, VecDeque}, unreachable, }; @@ -56,14 +58,12 @@ struct LambdaContext { /// This struct holds the FIFO queue of functions to monomorphize, which is added to /// whenever a new (function, type) combination is encountered. struct Monomorphizer<'interner> { - /// Functions are keyed by their unique ID and expected type so that we can monomorphize - /// a new version of the function for each type. - /// We also key by any turbofish generics that are specified. - /// This is necessary for a case where we may have a trait generic that can be instantiated - /// outside of a function parameter or return value. + /// Functions are keyed by their unique ID, whether they're unconstrained, their expected type, + /// and any generics they have so that we can monomorphize a new version of the function for each type. /// - /// Using nested HashMaps here lets us avoid cloning HirTypes when calling .get() - functions: HashMap), FuncId>>, + /// Keying by any turbofish generics that are specified is necessary for a case where we may have a + /// trait generic that can be instantiated outside of a function parameter or return value. + functions: Functions, /// Unlike functions, locals are only keyed by their unique ID because they are never /// duplicated during monomorphization. Doing so would allow them to be used polymorphically @@ -71,9 +71,21 @@ struct Monomorphizer<'interner> { /// confuse users. locals: HashMap, + /// Globals are keyed by their unique ID because they are never duplicated during monomorphization. + globals: HashMap, + + finished_globals: HashMap, + /// Queue of functions to monomorphize next each item in the queue is a tuple of: - /// (old_id, new_monomorphized_id, any type bindings to apply, the trait method if old_id is from a trait impl) - queue: VecDeque<(node_interner::FuncId, FuncId, TypeBindings, Option, Location)>, + /// (old_id, new_monomorphized_id, any type bindings to apply, the trait method if old_id is from a trait impl, is_unconstrained, location) + queue: VecDeque<( + node_interner::FuncId, + FuncId, + TypeBindings, + Option, + bool, + Location, + )>, /// When a function finishes being monomorphized, the monomorphized ast::Function is /// stored here along with its FuncId. @@ -85,6 +97,7 @@ struct Monomorphizer<'interner> { lambda_envs_stack: Vec, next_local_id: u32, + next_global_id: u32, next_function_id: u32, is_range_loop: bool, @@ -92,8 +105,16 @@ struct Monomorphizer<'interner> { return_location: Option, debug_type_tracker: DebugTypeTracker, + + in_unconstrained_function: bool, } +/// Using nested HashMaps here lets us avoid cloning HirTypes when calling .get() +type Functions = HashMap< + (node_interner::FuncId, /*is_unconstrained:*/ bool), + HashMap, FuncId>>, +>; + type HirType = crate::Type; /// Starting from the given `main` function, monomorphize the entire program, @@ -111,24 +132,29 @@ type HirType = crate::Type; pub fn monomorphize( main: node_interner::FuncId, interner: &mut NodeInterner, + force_unconstrained: bool, ) -> Result { - monomorphize_debug(main, interner, &DebugInstrumenter::default()) + monomorphize_debug(main, interner, &DebugInstrumenter::default(), force_unconstrained) } pub fn monomorphize_debug( main: node_interner::FuncId, interner: &mut NodeInterner, debug_instrumenter: &DebugInstrumenter, + force_unconstrained: bool, ) -> Result { let debug_type_tracker = DebugTypeTracker::build_from_debug_instrumenter(debug_instrumenter); let mut monomorphizer = Monomorphizer::new(interner, debug_type_tracker); + monomorphizer.in_unconstrained_function = force_unconstrained; let function_sig = monomorphizer.compile_main(main)?; while !monomorphizer.queue.is_empty() { - let (next_fn_id, new_id, bindings, trait_method, location) = + let (next_fn_id, new_id, bindings, trait_method, is_unconstrained, location) = monomorphizer.queue.pop_front().unwrap(); monomorphizer.locals.clear(); + monomorphizer.in_unconstrained_function = is_unconstrained; + perform_instantiation_bindings(&bindings); let interner = &monomorphizer.interner; let impl_bindings = perform_impl_bindings(interner, trait_method, next_fn_id, location) @@ -143,7 +169,9 @@ pub fn monomorphize_debug( .finished_functions .iter() .flat_map(|(_, f)| { - if f.inline_type.is_entry_point() || f.id == Program::main_id() { + if (!force_unconstrained && f.inline_type.is_entry_point()) + || f.id == Program::main_id() + { Some(f.func_sig.clone()) } else { None @@ -154,14 +182,18 @@ pub fn monomorphize_debug( let functions = vecmap(monomorphizer.finished_functions, |(_, f)| f); let FuncMeta { return_visibility, .. } = monomorphizer.interner.function_meta(&main); + let globals = monomorphizer.finished_globals.into_iter().collect::>(); + let (debug_variables, debug_functions, debug_types) = monomorphizer.debug_type_tracker.extract_vars_and_types(); + let program = Program::new( functions, func_sigs, function_sig, monomorphizer.return_location, *return_visibility, + globals, debug_variables, debug_functions, debug_types, @@ -172,17 +204,21 @@ pub fn monomorphize_debug( impl<'interner> Monomorphizer<'interner> { fn new(interner: &'interner mut NodeInterner, debug_type_tracker: DebugTypeTracker) -> Self { Monomorphizer { - functions: HashMap::new(), - locals: HashMap::new(), + functions: HashMap::default(), + locals: HashMap::default(), + globals: HashMap::default(), + finished_globals: HashMap::default(), queue: VecDeque::new(), finished_functions: BTreeMap::new(), next_local_id: 0, + next_global_id: 0, next_function_id: 0, interner, lambda_envs_stack: Vec::new(), is_range_loop: false, return_location: None, debug_type_tracker, + in_unconstrained_function: false, } } @@ -198,6 +234,12 @@ impl<'interner> Monomorphizer<'interner> { ast::FuncId(id) } + fn next_global_id(&mut self) -> GlobalId { + let id = self.next_global_id; + self.next_global_id += 1; + GlobalId(id) + } + fn lookup_local(&mut self, id: node_interner::DefinitionId) -> Option { self.locals.get(&id).copied().map(Definition::Local) } @@ -207,14 +249,18 @@ impl<'interner> Monomorphizer<'interner> { id: node_interner::FuncId, expr_id: node_interner::ExprId, typ: &HirType, - turbofish_generics: Vec, + turbofish_generics: &[HirType], trait_method: Option, ) -> Definition { let typ = typ.follow_bindings(); + let turbofish_generics = vecmap(turbofish_generics, |typ| typ.follow_bindings()); + let is_unconstrained = self.is_unconstrained(id); + match self .functions - .get(&id) - .and_then(|inner_map| inner_map.get(&(typ.clone(), turbofish_generics.clone()))) + .get(&(id, is_unconstrained)) + .and_then(|inner_map| inner_map.get(&typ)) + .and_then(|inner_map| inner_map.get(&turbofish_generics)) { Some(id) => Definition::Function(*id), None => { @@ -235,7 +281,7 @@ impl<'interner> Monomorphizer<'interner> { ); Definition::Builtin(opcode.to_string()) } - FunctionKind::Normal => { + FunctionKind::Normal | FunctionKind::TraitFunctionWithoutBody => { let id = self.queue_function(id, expr_id, typ, turbofish_generics, trait_method); Definition::Function(id) @@ -257,14 +303,21 @@ impl<'interner> Monomorphizer<'interner> { } /// Prerequisite: typ = typ.follow_bindings() + /// and: turbofish_generics = vecmap(turbofish_generics, Type::follow_bindings) fn define_function( &mut self, id: node_interner::FuncId, typ: HirType, turbofish_generics: Vec, + is_unconstrained: bool, new_id: FuncId, ) { - self.functions.entry(id).or_default().insert((typ, turbofish_generics), new_id); + self.functions + .entry((id, is_unconstrained)) + .or_default() + .entry(typ) + .or_default() + .insert(turbofish_generics, new_id); } fn compile_main( @@ -275,6 +328,7 @@ impl<'interner> Monomorphizer<'interner> { assert_eq!(new_main_id, Program::main_id()); let location = self.interner.function_meta(&main_id).location; + self.in_unconstrained_function = self.is_unconstrained(main_id); self.function(main_id, new_main_id, location)?; self.return_location = @@ -324,7 +378,7 @@ impl<'interner> Monomorphizer<'interner> { }; let return_type = Self::convert_type(return_type, meta.location)?; - let unconstrained = modifiers.is_unconstrained; + let unconstrained = self.in_unconstrained_function; let attributes = self.interner.function_attributes(&f); let inline_type = InlineType::from(attributes); @@ -641,6 +695,10 @@ impl<'interner> Monomorphizer<'interner> { block, })) } + HirStatement::Loop(block) => { + let block = Box::new(self.expr(block)?); + Ok(ast::Expression::Loop(block)) + } HirStatement::Expression(expr) => self.expr(expr), HirStatement::Semi(expr) => { self.expr(expr).map(|expr| ast::Expression::Semi(Box::new(expr))) @@ -874,7 +932,7 @@ impl<'interner> Monomorphizer<'interner> { *func_id, expr_id, &typ, - generics.unwrap_or_default(), + &generics.unwrap_or_default(), None, ); let typ = Self::convert_type(&typ, ident.location)?; @@ -893,19 +951,7 @@ impl<'interner> Monomorphizer<'interner> { } } DefinitionKind::Global(global_id) => { - let global = self.interner.get_global(*global_id); - - let expr = if let Some(value) = global.value.clone() { - value - .into_hir_expression(self.interner, global.location) - .map_err(MonomorphizationError::InterpreterError)? - } else { - let let_ = self.interner.get_global_let_statement(*global_id).expect( - "Globals should have a corresponding let statement by monomorphization", - ); - let_.expression - }; - self.expr(expr)? + self.global_ident(*global_id, definition.name.clone(), &typ, ident.location)? } DefinitionKind::Local(_) => match self.lookup_captured_expr(ident.id) { Some(expr) => expr, @@ -952,6 +998,66 @@ impl<'interner> Monomorphizer<'interner> { Ok(ident) } + fn global_ident( + &mut self, + global_id: node_interner::GlobalId, + name: String, + typ: &HirType, + location: Location, + ) -> Result { + let global = self.interner.get_global(global_id); + let id = global.id; + let expr = if let Some(seen_global) = self.globals.get(&id) { + let typ = Self::convert_type(typ, location)?; + let ident = ast::Ident { + location: Some(location), + definition: Definition::Global(*seen_global), + mutable: false, + name, + typ, + }; + ast::Expression::Ident(ident) + } else { + let (expr, is_closure) = if let GlobalValue::Resolved(value) = global.value.clone() { + let is_closure = value.is_closure(); + let expr = value + .into_hir_expression(self.interner, global.location) + .map_err(MonomorphizationError::InterpreterError)?; + (expr, is_closure) + } else { + unreachable!("All global values should be resolved at compile time and before monomorphization"); + }; + + let expr = self.expr(expr)?; + + // Globals are meant to be computed at compile time and are stored in their own context to be shared across functions. + // Closures are defined as normal functions among all SSA functions and later need to be defunctionalized. + // Thus, this means we would have to re-define any global closures. + // The effect of defunctionalization would be the same if we were redefining a global closure or a local closure + // just with an extra step of indirection through a global variable. + // For simplicity, we chose to instead inline closures at their callsite as we do not expect + // placing a closure in the global context to change the final result of the program. + if !is_closure { + let new_id = self.next_global_id(); + self.globals.insert(id, new_id); + + self.finished_globals.insert(new_id, expr); + let typ = Self::convert_type(typ, location)?; + let ident = ast::Ident { + location: Some(location), + definition: Definition::Global(new_id), + mutable: false, + name, + typ, + }; + ast::Expression::Ident(ident) + } else { + expr + } + }; + Ok(expr) + } + /// Convert a non-tuple/struct type to a monomorphized type fn convert_type(typ: &HirType, location: Location) -> Result { let typ = typ.follow_bindings_shallow(); @@ -1133,7 +1239,7 @@ impl<'interner> Monomorphizer<'interner> { | HirType::Error | HirType::Quoted(_) => Ok(()), HirType::Constant(_value, kind) => { - if kind.is_error() { + if kind.is_error() || kind.default_type().is_none() { Err(MonomorphizationError::UnknownConstant { location }) } else { Ok(()) @@ -1154,7 +1260,6 @@ impl<'interner> Monomorphizer<'interner> { Ok(()) } - HirType::TypeVariable(ref binding) => { let type_var_kind = match &*binding.borrow() { TypeBinding::Bound(binding) => { @@ -1208,7 +1313,7 @@ impl<'interner> Monomorphizer<'interner> { } HirType::MutableReference(element) => Self::check_type(element, location), - HirType::InfixExpr(lhs, _, rhs) => { + HirType::InfixExpr(lhs, _, rhs, _) => { Self::check_type(lhs, location)?; Self::check_type(rhs, location) } @@ -1282,7 +1387,7 @@ impl<'interner> Monomorphizer<'interner> { .map_err(MonomorphizationError::InterpreterError)?; let func_id = - match self.lookup_function(func_id, expr_id, &function_type, vec![], Some(method)) { + match self.lookup_function(func_id, expr_id, &function_type, &[], Some(method)) { Definition::Function(func_id) => func_id, _ => unreachable!(), }; @@ -1547,12 +1652,19 @@ impl<'interner> Monomorphizer<'interner> { trait_method: Option, ) -> FuncId { let new_id = self.next_function_id(); - self.define_function(id, function_type.clone(), turbofish_generics, new_id); + let is_unconstrained = self.is_unconstrained(id); + self.define_function( + id, + function_type.clone(), + turbofish_generics, + is_unconstrained, + new_id, + ); let location = self.interner.expr_location(&expr_id); let bindings = self.interner.get_instantiation_bindings(expr_id); let bindings = self.follow_bindings(bindings); - self.queue.push_back((id, new_id, bindings, trait_method, location)); + self.queue.push_back((id, new_id, bindings, trait_method, is_unconstrained, location)); new_id } @@ -1639,19 +1751,15 @@ impl<'interner> Monomorphizer<'interner> { let parameters = self.parameters(¶meters)?; let body = self.expr(lambda.body)?; - let id = self.next_function_id(); - let return_type = ret_type.clone(); - let name = lambda_name.to_owned(); - let unconstrained = false; let function = ast::Function { id, - name, + name: lambda_name.to_owned(), parameters, body, - return_type, - unconstrained, + return_type: ret_type.clone(), + unconstrained: self.in_unconstrained_function, inline_type: InlineType::default(), func_sig: FunctionSignature::default(), }; @@ -1774,18 +1882,16 @@ impl<'interner> Monomorphizer<'interner> { typ: lambda_fn_typ.clone(), }); - let mut parameters = vec![]; - parameters.push((env_local_id, true, env_name.to_string(), env_typ.clone())); + let mut parameters = vec![(env_local_id, true, env_name.to_string(), env_typ.clone())]; parameters.append(&mut converted_parameters); - let unconstrained = false; let function = ast::Function { id, name, parameters, body, return_type, - unconstrained, + unconstrained: self.in_unconstrained_function, inline_type: InlineType::default(), func_sig: FunctionSignature::default(), }; @@ -2008,6 +2114,11 @@ impl<'interner> Monomorphizer<'interner> { Ok(ast::Expression::Call(ast::Call { func, arguments, return_type, location })) } + + fn is_unconstrained(&self, func_id: node_interner::FuncId) -> bool { + self.in_unconstrained_function + || self.interner.function_modifiers(&func_id).is_unconstrained + } } fn unwrap_tuple_type(typ: &HirType) -> Vec { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/printer.rs b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/printer.rs index 9c1072a4117..665f4dcd371 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/printer.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/printer.rs @@ -49,6 +49,7 @@ impl AstPrinter { write!(f, " as {})", cast.r#type) } Expression::For(for_expr) => self.print_for(for_expr, f), + Expression::Loop(block) => self.print_loop(block, f), Expression::If(if_expr) => self.print_if(if_expr, f), Expression::Tuple(tuple) => self.print_tuple(tuple, f), Expression::ExtractTupleField(expr, index) => { @@ -209,6 +210,15 @@ impl AstPrinter { write!(f, "}}") } + fn print_loop(&mut self, block: &Expression, f: &mut Formatter) -> Result<(), std::fmt::Error> { + write!(f, "loop {{")?; + self.indent_level += 1; + self.print_expr_expect_block(block, f)?; + self.indent_level -= 1; + self.next_line(f)?; + write!(f, "}}") + } + fn print_if( &mut self, if_expr: &super::ast::If, @@ -293,6 +303,7 @@ impl Display for Definition { fn fmt(&self, f: &mut Formatter) -> std::fmt::Result { match self { Definition::Local(id) => write!(f, "l{}", id.0), + Definition::Global(id) => write!(f, "g{}", id.0), Definition::Function(id) => write!(f, "f{}", id), Definition::Builtin(name) => write!(f, "{name}"), Definition::LowLevel(name) => write!(f, "{name}"), diff --git a/noir/noir-repo/compiler/noirc_frontend/src/node_interner.rs b/noir/noir-repo/compiler/noirc_frontend/src/node_interner.rs index 6d70ea2fd6d..ae2cf224cbd 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/node_interner.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/node_interner.rs @@ -2,7 +2,6 @@ use std::borrow::Cow; use std::fmt; use std::hash::Hash; use std::marker::Copy; -use std::ops::Deref; use fm::FileId; use iter_extended::vecmap; @@ -184,16 +183,13 @@ pub struct NodeInterner { next_type_variable_id: std::cell::Cell, - /// A map from a struct type and method name to a function id for the method. + /// A map from a type and method name to a function id for the method. /// This can resolve to potentially multiple methods if the same method name is /// specialized for different generics on the same type. E.g. for `Struct`, we /// may have both `impl Struct { fn foo(){} }` and `impl Struct { fn foo(){} }`. /// If this happens, the returned Vec will have 2 entries and we'll need to further /// disambiguate them by checking the type of each function. - struct_methods: HashMap>, - - /// Methods on primitive types defined in the stdlib. - primitive_methods: HashMap>, + methods: HashMap>, // For trait implementation functions, this is their self type and trait they belong to func_id_to_trait: HashMap, @@ -271,6 +267,11 @@ pub struct NodeInterner { /// Captures the documentation comments for each module, struct, trait, function, etc. pub(crate) doc_comments: HashMap>, + + /// Only for LSP: a map of trait ID to each module that pub or pub(crate) exports it. + /// In LSP this is used to offer importing the trait via one of these exports if + /// the trait is not visible where it's defined. + trait_reexports: HashMap>, } /// A dependency in the dependency graph may be a type or a definition. @@ -367,6 +368,7 @@ pub struct TraitImplMethod { // under TypeMethodKey::FieldOrInt pub typ: Option, pub method: FuncId, + pub trait_id: TraitId, } /// All the information from a function that is filled out during definition collection rather than @@ -414,7 +416,7 @@ pub struct DefinitionId(usize); impl DefinitionId { //dummy id for error reporting pub fn dummy_id() -> DefinitionId { - DefinitionId(std::usize::MAX) + DefinitionId(usize::MAX) } } @@ -425,7 +427,7 @@ pub struct GlobalId(usize); impl GlobalId { // Dummy id for error reporting pub fn dummy_id() -> Self { - GlobalId(std::usize::MAX) + GlobalId(usize::MAX) } } @@ -496,7 +498,7 @@ pub struct TypeAliasId(pub usize); impl TypeAliasId { pub fn dummy_id() -> TypeAliasId { - TypeAliasId(std::usize::MAX) + TypeAliasId(usize::MAX) } } @@ -609,7 +611,14 @@ pub struct GlobalInfo { pub crate_id: CrateId, pub location: Location, pub let_statement: StmtId, - pub value: Option, + pub value: GlobalValue, +} + +#[derive(Debug, Clone)] +pub enum GlobalValue { + Unresolved, + Resolving, + Resolved(comptime::Value), } #[derive(Debug, Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Hash)] @@ -659,8 +668,7 @@ impl Default for NodeInterner { next_type_variable_id: std::cell::Cell::new(0), globals: Vec::new(), global_attributes: HashMap::default(), - struct_methods: HashMap::default(), - primitive_methods: HashMap::default(), + methods: HashMap::default(), type_alias_ref: Vec::new(), type_ref_locations: Vec::new(), quoted_types: Default::default(), @@ -677,6 +685,7 @@ impl Default for NodeInterner { comptime_scopes: vec![HashMap::default()], trait_impl_associated_types: HashMap::default(), doc_comments: HashMap::default(), + trait_reexports: HashMap::default(), } } } @@ -726,6 +735,7 @@ impl NodeInterner { crate_id: unresolved_trait.crate_id, location: Location::new(unresolved_trait.trait_def.span, unresolved_trait.file_id), generics, + visibility: ItemVisibility::Private, self_type_typevar: TypeVariable::unbound(self.next_type_variable_id(), Kind::Normal), methods: Vec::new(), method_ids: unresolved_trait.method_ids.clone(), @@ -849,6 +859,7 @@ impl NodeInterner { let id = GlobalId(self.globals.len()); let location = Location::new(ident.span(), file); let name = ident.to_string(); + let definition_id = self.push_definition(name, mutable, comptime, DefinitionKind::Global(id), location); @@ -860,7 +871,7 @@ impl NodeInterner { crate_id, let_statement, location, - value: None, + value: GlobalValue::Unresolved, }); self.global_attributes.insert(id, attributes); id @@ -884,6 +895,7 @@ impl NodeInterner { ) -> GlobalId { let statement = self.push_stmt(HirStatement::Error); let span = name.span(); + let id = self .push_global(name, local_id, crate_id, statement, file, attributes, mutable, comptime); self.push_stmt_location(statement, span, file); @@ -1205,27 +1217,8 @@ impl NodeInterner { self.structs[&id].clone() } - pub fn get_struct_methods(&self, id: StructId) -> Option<&HashMap> { - self.struct_methods.get(&id) - } - - fn get_primitive_methods(&self, key: TypeMethodKey) -> Option<&HashMap> { - self.primitive_methods.get(&key) - } - pub fn get_type_methods(&self, typ: &Type) -> Option<&HashMap> { - match typ { - Type::Struct(struct_type, _) => { - let struct_type = struct_type.borrow(); - self.get_struct_methods(struct_type.id) - } - Type::Alias(type_alias, generics) => { - let type_alias = type_alias.borrow(); - let typ = type_alias.get_type(generics); - self.get_type_methods(&typ) - } - _ => get_type_method_key(typ).and_then(|key| self.get_primitive_methods(key)), - } + get_type_method_key(typ).and_then(|key| self.methods.get(&key)) } pub fn get_trait(&self, id: TraitId) -> &Trait { @@ -1382,44 +1375,35 @@ impl NodeInterner { self_type: &Type, method_name: String, method_id: FuncId, - is_trait_method: bool, + trait_id: Option, ) -> Option { match self_type { - Type::Struct(struct_type, _generics) => { - let id = struct_type.borrow().id; - - if let Some(existing) = self.lookup_method(self_type, id, &method_name, true, true) - { - return Some(existing); - } - - self.struct_methods - .entry(id) - .or_default() - .entry(method_name) - .or_default() - .add_method(method_id, None, is_trait_method); - None - } Type::Error => None, Type::MutableReference(element) => { - self.add_method(element, method_name, method_id, is_trait_method) + self.add_method(element, method_name, method_id, trait_id) } - - other => { + _ => { let key = get_type_method_key(self_type).unwrap_or_else(|| { - unreachable!("Cannot add a method to the unsupported type '{}'", other) + unreachable!("Cannot add a method to the unsupported type '{}'", self_type) }); + + if trait_id.is_none() && matches!(self_type, Type::Struct(..)) { + if let Some(existing) = self.lookup_direct_method(self_type, &method_name, true) + { + return Some(existing); + } + } + // Only remember the actual type if it's FieldOrInt, // so later we can disambiguate on calls like `u32::call`. let typ = if key == TypeMethodKey::FieldOrInt { Some(self_type.clone()) } else { None }; - self.primitive_methods + self.methods .entry(key) .or_default() .entry(method_name) .or_default() - .add_method(method_id, typ, is_trait_method); + .add_method(method_id, typ, trait_id); None } } @@ -1469,25 +1453,6 @@ impl NodeInterner { Ok(impl_kind) } - /// Given a `ObjectType: TraitId` pair, find all implementations without taking constraints into account or - /// applying any type bindings. Useful to look for a specific trait in a type that is used in a macro. - pub fn lookup_all_trait_implementations( - &self, - object_type: &Type, - trait_id: TraitId, - ) -> Vec<&TraitImplKind> { - let trait_impl = self.trait_implementation_map.get(&trait_id); - - let trait_impl = trait_impl.map(|trait_impl| { - let impls = trait_impl.iter().filter_map(|(typ, impl_kind)| match &typ { - Type::Forall(_, typ) => (typ.deref() == object_type).then_some(impl_kind), - _ => None, - }); - impls.collect() - }); - trait_impl.unwrap_or_default() - } - /// Similar to `lookup_trait_implementation` but does not apply any type bindings on success. /// On error returns either: /// - 1+ failing trait constraints, including the original. @@ -1776,7 +1741,7 @@ impl NodeInterner { for method in &trait_impl.borrow().methods { let method_name = self.function_name(method).to_owned(); - self.add_method(&object_type, method_name, *method, true); + self.add_method(&object_type, method_name, *method, Some(trait_id)); } // The object type is generalized so that a generic impl will apply @@ -1788,78 +1753,52 @@ impl NodeInterner { Ok(()) } - /// Search by name for a method on the given struct. - /// - /// If `check_type` is true, this will force `lookup_method` to check the type - /// of each candidate instead of returning only the first candidate if there is exactly one. - /// This is generally only desired when declaring new methods to check if they overlap any - /// existing methods. - /// - /// Another detail is that this method does not handle auto-dereferencing through `&mut T`. - /// So if an object is of type `self : &mut T` but a method only accepts `self: T` (or - /// vice-versa), the call will not be selected. If this is ever implemented into this method, - /// we can remove the `methods.len() == 1` check and the `check_type` early return. - pub fn lookup_method( + /// Looks up a method that's directly defined in the given type. + pub fn lookup_direct_method( &self, typ: &Type, - id: StructId, method_name: &str, - force_type_check: bool, has_self_arg: bool, ) -> Option { - let methods = self.struct_methods.get(&id).and_then(|h| h.get(method_name)); - - // If there is only one method, just return it immediately. - // It will still be typechecked later. - if !force_type_check { - if let Some(method) = methods.and_then(|m| m.get_unambiguous()) { - return Some(method); - } - } + let key = get_type_method_key(typ)?; - self.find_matching_method(typ, methods, method_name, has_self_arg) + self.methods + .get(&key) + .and_then(|h| h.get(method_name)) + .and_then(|methods| methods.find_direct_method(typ, has_self_arg, self)) } - /// Select the 1 matching method with an object type matching `typ` - fn find_matching_method( + /// Looks up a methods that apply to the given type but are defined in traits. + pub fn lookup_trait_methods( &self, typ: &Type, - methods: Option<&Methods>, method_name: &str, has_self_arg: bool, - ) -> Option { - if let Some(method) = methods.and_then(|m| m.find_matching_method(typ, has_self_arg, self)) - { - Some(method) + ) -> Vec<(FuncId, TraitId)> { + let key = get_type_method_key(typ); + if let Some(key) = key { + self.methods + .get(&key) + .and_then(|h| h.get(method_name)) + .map(|methods| methods.find_trait_methods(typ, has_self_arg, self)) + .unwrap_or_default() } else { - // Failed to find a match for the type in question, switch to looking at impls - // for all types `T`, e.g. `impl Foo for T` - let global_methods = - self.primitive_methods.get(&TypeMethodKey::Generic)?.get(method_name)?; - global_methods.find_matching_method(typ, has_self_arg, self) + Vec::new() } } - /// Looks up a given method name on the given primitive type. - pub fn lookup_primitive_method( - &self, - typ: &Type, - method_name: &str, - has_self_arg: bool, - ) -> Option { - let key = get_type_method_key(typ)?; - let methods = self.primitive_methods.get(&key)?.get(method_name)?; - self.find_matching_method(typ, Some(methods), method_name, has_self_arg) - } - - pub fn lookup_primitive_trait_method_mut( + /// Looks up methods at impls for all types `T`, e.g. `impl Foo for T` + pub fn lookup_generic_methods( &self, typ: &Type, method_name: &str, has_self_arg: bool, - ) -> Option { - let typ = Type::MutableReference(Box::new(typ.clone())); - self.lookup_primitive_method(&typ, method_name, has_self_arg) + ) -> Vec<(FuncId, TraitId)> { + self.methods + .get(&TypeMethodKey::Generic) + .and_then(|h| h.get(method_name)) + .map(|methods| methods.find_trait_methods(typ, has_self_arg, self)) + .unwrap_or_default() } /// Returns what the next trait impl id is expected to be. @@ -2315,25 +2254,34 @@ impl NodeInterner { pub fn doc_comments(&self, id: ReferenceId) -> Option<&Vec> { self.doc_comments.get(&id) } -} -impl Methods { - /// Get a single, unambiguous reference to a name if one exists. - /// If not, there may be multiple methods of the same name for a given - /// type or there may be no methods at all. - fn get_unambiguous(&self) -> Option { - if self.direct.len() == 1 { - Some(self.direct[0]) - } else if self.direct.is_empty() && self.trait_impl_methods.len() == 1 { - Some(self.trait_impl_methods[0].method) - } else { - None + pub fn get_expr_id_from_index(&self, index: impl Into) -> Option { + let index = index.into(); + match self.nodes.get(index) { + Some(Node::Expression(_)) => Some(ExprId(index)), + _ => None, } } - fn add_method(&mut self, method: FuncId, typ: Option, is_trait_method: bool) { - if is_trait_method { - let trait_impl_method = TraitImplMethod { typ, method }; + pub fn add_trait_reexport( + &mut self, + trait_id: TraitId, + module_id: ModuleId, + name: Ident, + visibility: ItemVisibility, + ) { + self.trait_reexports.entry(trait_id).or_default().push((module_id, name, visibility)); + } + + pub fn get_trait_reexports(&self, trait_id: TraitId) -> &[(ModuleId, Ident, ItemVisibility)] { + self.trait_reexports.get(&trait_id).map_or(&[], |exports| exports) + } +} + +impl Methods { + fn add_method(&mut self, method: FuncId, typ: Option, trait_id: Option) { + if let Some(trait_id) = trait_id { + let trait_impl_method = TraitImplMethod { typ, method, trait_id }; self.trait_impl_methods.push(trait_impl_method); } else { self.direct.push(method); @@ -2341,51 +2289,110 @@ impl Methods { } /// Iterate through each method, starting with the direct methods - pub fn iter(&self) -> impl Iterator)> + '_ { - let trait_impl_methods = self.trait_impl_methods.iter().map(|m| (m.method, &m.typ)); - let direct = self.direct.iter().copied().map(|func_id| { - let typ: &Option = &None; - (func_id, typ) - }); + pub fn iter(&self) -> impl Iterator, Option)> { + let trait_impl_methods = + self.trait_impl_methods.iter().map(|m| (m.method, m.typ.as_ref(), Some(m.trait_id))); + let direct = self.direct.iter().copied().map(|func_id| (func_id, None, None)); direct.chain(trait_impl_methods) } - /// Select the 1 matching method with an object type matching `typ` - pub fn find_matching_method( + pub fn find_matching_methods<'a>( + &'a self, + typ: &'a Type, + has_self_param: bool, + interner: &'a NodeInterner, + ) -> impl Iterator)> + 'a { + self.iter().filter_map(move |(method, method_type, trait_id)| { + if Self::method_matches(typ, has_self_param, method, method_type, interner) { + Some((method, trait_id)) + } else { + None + } + }) + } + + pub fn find_direct_method( &self, typ: &Type, has_self_param: bool, interner: &NodeInterner, ) -> Option { - // When adding methods we always check they do not overlap, so there should be - // at most 1 matching method in this list. - for (method, method_type) in self.iter() { - match interner.function_meta(&method).typ.instantiate(interner).0 { - Type::Function(args, _, _, _) => { - if has_self_param { - if let Some(object) = args.first() { + for method in &self.direct { + if Self::method_matches(typ, has_self_param, *method, None, interner) { + return Some(*method); + } + } + + None + } + + pub fn find_trait_methods( + &self, + typ: &Type, + has_self_param: bool, + interner: &NodeInterner, + ) -> Vec<(FuncId, TraitId)> { + let mut results = Vec::new(); + + for trait_impl_method in &self.trait_impl_methods { + let method = trait_impl_method.method; + let method_type = trait_impl_method.typ.as_ref(); + let trait_id = trait_impl_method.trait_id; + + if Self::method_matches(typ, has_self_param, method, method_type, interner) { + results.push((method, trait_id)); + } + } + + results + } + + fn method_matches( + typ: &Type, + has_self_param: bool, + method: FuncId, + method_type: Option<&Type>, + interner: &NodeInterner, + ) -> bool { + match interner.function_meta(&method).typ.instantiate(interner).0 { + Type::Function(args, _, _, _) => { + if has_self_param { + if let Some(object) = args.first() { + if object.unify(typ).is_ok() { + return true; + } + + // Handle auto-dereferencing `&mut T` into `T` + if let Type::MutableReference(object) = object { if object.unify(typ).is_ok() { - return Some(method); + return true; } } - } else { - // If we recorded the concrete type this trait impl method belongs to, - // and it matches typ, it's an exact match and we return that. - if let Some(method_type) = method_type { + } + } else { + // If we recorded the concrete type this trait impl method belongs to, + // and it matches typ, it's an exact match and we return that. + if let Some(method_type) = method_type { + if method_type.unify(typ).is_ok() { + return true; + } + + // Handle auto-dereferencing `&mut T` into `T` + if let Type::MutableReference(method_type) = method_type { if method_type.unify(typ).is_ok() { - return Some(method); + return true; } - } else { - return Some(method); } + } else { + return true; } } - Type::Error => (), - other => unreachable!("Expected function type, found {other}"), } + Type::Error => (), + other => unreachable!("Expected function type, found {other}"), } - None + false } } @@ -2405,6 +2412,7 @@ enum TypeMethodKey { Function, Generic, Quoted(QuotedType), + Struct(StructId), } fn get_type_method_key(typ: &Type) -> Option { @@ -2432,12 +2440,12 @@ fn get_type_method_key(typ: &Type) -> Option { Type::Quoted(quoted) => Some(Quoted(*quoted)), Type::MutableReference(element) => get_type_method_key(element), Type::Alias(alias, _) => get_type_method_key(&alias.borrow().typ), + Type::Struct(struct_type, _) => Some(Struct(struct_type.borrow().id)), // We do not support adding methods to these types Type::Forall(_, _) | Type::Constant(..) | Type::Error - | Type::Struct(_, _) | Type::InfixExpr(..) | Type::CheckedCast { .. } | Type::TraitAsType(..) => None, diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/errors.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/errors.rs index 899928528e6..508ed33857e 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/errors.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/errors.rs @@ -71,6 +71,8 @@ pub enum ParserErrorReason { PatternInTraitFunctionParameter, #[error("Patterns aren't allowed in a trait impl's associated constants")] PatternInAssociatedConstant, + #[error("Visibility is ignored on a trait method")] + TraitVisibilityIgnored, #[error("Visibility is ignored on a trait impl method")] TraitImplVisibilityIgnored, #[error("comptime keyword is deprecated")] @@ -81,8 +83,8 @@ pub enum ParserErrorReason { "Multiple primary attributes found. Only one function attribute is allowed per function" )] MultipleFunctionAttributesFound, - #[error("A function attribute cannot be placed on a struct")] - NoFunctionAttributesAllowedOnStruct, + #[error("A function attribute cannot be placed on a struct or enum")] + NoFunctionAttributesAllowedOnType, #[error("Assert statements can only accept string literals")] AssertMessageNotString, #[error("Integer bit size {0} isn't supported")] @@ -108,6 +110,12 @@ pub enum ParserErrorReason { WrongNumberOfAttributeArguments { name: String, min: usize, max: usize, found: usize }, #[error("The `deprecated` attribute expects a string argument")] DeprecatedAttributeExpectsAStringArgument, + #[error("Unsafe block must have a safety doc comment above it")] + MissingSafetyComment, + #[error("Unsafe block must start with a safety comment")] + UnsafeDocCommentDoesNotStartWithSafety, + #[error("Missing parameters for function definition")] + MissingParametersForFunctionDefinition, } /// Represents a parsing error, or a parsing error in the making. @@ -181,7 +189,8 @@ impl ParserError { } pub fn is_warning(&self) -> bool { - matches!(self.reason(), Some(ParserErrorReason::ExperimentalFeature(_))) + let diagnostic: Diagnostic = self.into(); + diagnostic.is_warning() } } @@ -258,6 +267,9 @@ impl<'a> From<&'a ParserError> for Diagnostic { ParserErrorReason::ExperimentalFeature(_) => { Diagnostic::simple_warning(reason.to_string(), "".into(), error.span) } + ParserErrorReason::TraitVisibilityIgnored => { + Diagnostic::simple_warning(reason.to_string(), "".into(), error.span) + } ParserErrorReason::TraitImplVisibilityIgnored => { Diagnostic::simple_warning(reason.to_string(), "".into(), error.span) } @@ -272,6 +284,26 @@ impl<'a> From<&'a ParserError> for Diagnostic { "Noir doesn't have immutable references, only mutable references".to_string(), error.span, ), + ParserErrorReason::MissingSafetyComment => Diagnostic::simple_warning( + "Unsafe block must have a safety doc comment above it".into(), + "The doc comment must start with the \"Safety: \" word".into(), + error.span, + ), + ParserErrorReason::UnsafeDocCommentDoesNotStartWithSafety => { + Diagnostic::simple_warning( + "Unsafe block must start with a safety comment".into(), + "The doc comment above this unsafe block must start with the \"Safety: \" word" + .into(), + error.span, + ) + } + ParserErrorReason::MissingParametersForFunctionDefinition => { + Diagnostic::simple_error( + "Missing parameters for function definition".into(), + "Add a parameter list: `()`".into(), + error.span, + ) + } other => Diagnostic::simple_error(format!("{other}"), String::new(), error.span), }, None => { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/mod.rs index 17c156476a7..c433adbfdfb 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/mod.rs @@ -13,7 +13,8 @@ mod parser; use crate::ast::{ Documented, Ident, ImportStatement, ItemVisibility, LetStatement, ModuleDeclaration, - NoirFunction, NoirStruct, NoirTrait, NoirTraitImpl, NoirTypeAlias, TypeImpl, UseTree, + NoirEnumeration, NoirFunction, NoirStruct, NoirTrait, NoirTraitImpl, NoirTypeAlias, TypeImpl, + UseTree, }; use crate::token::SecondaryAttribute; @@ -26,7 +27,8 @@ pub use parser::{parse_program, Parser, StatementOrExpressionOrLValue}; pub struct SortedModule { pub imports: Vec, pub functions: Vec>, - pub types: Vec>, + pub structs: Vec>, + pub enums: Vec>, pub traits: Vec>, pub trait_impls: Vec, pub impls: Vec, @@ -57,7 +59,7 @@ impl std::fmt::Display for SortedModule { write!(f, "{global_const}")?; } - for type_ in &self.types { + for type_ in &self.structs { write!(f, "{type_}")?; } @@ -96,7 +98,8 @@ impl ParsedModule { match item.kind { ItemKind::Import(import, visibility) => module.push_import(import, visibility), ItemKind::Function(func) => module.push_function(func, item.doc_comments), - ItemKind::Struct(typ) => module.push_type(typ, item.doc_comments), + ItemKind::Struct(typ) => module.push_struct(typ, item.doc_comments), + ItemKind::Enum(typ) => module.push_enum(typ, item.doc_comments), ItemKind::Trait(noir_trait) => module.push_trait(noir_trait, item.doc_comments), ItemKind::TraitImpl(trait_impl) => module.push_trait_impl(trait_impl), ItemKind::Impl(r#impl) => module.push_impl(r#impl), @@ -134,6 +137,7 @@ pub enum ItemKind { Import(UseTree, ItemVisibility), Function(NoirFunction), Struct(NoirStruct), + Enum(NoirEnumeration), Trait(NoirTrait), TraitImpl(NoirTraitImpl), Impl(TypeImpl), @@ -147,6 +151,7 @@ pub enum ItemKind { impl std::fmt::Display for ItemKind { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { match self { + ItemKind::Enum(e) => e.fmt(f), ItemKind::Function(fun) => fun.fmt(f), ItemKind::ModuleDecl(m) => m.fmt(f), ItemKind::Import(tree, visibility) => { @@ -222,8 +227,12 @@ impl SortedModule { self.functions.push(Documented::new(func, doc_comments)); } - fn push_type(&mut self, typ: NoirStruct, doc_comments: Vec) { - self.types.push(Documented::new(typ, doc_comments)); + fn push_struct(&mut self, typ: NoirStruct, doc_comments: Vec) { + self.structs.push(Documented::new(typ, doc_comments)); + } + + fn push_enum(&mut self, typ: NoirEnumeration, doc_comments: Vec) { + self.enums.push(Documented::new(typ, doc_comments)); } fn push_trait(&mut self, noir_trait: NoirTrait, doc_comments: Vec) { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser.rs index fcc58c5d833..e554248fb03 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser.rs @@ -13,6 +13,7 @@ use super::{labels::ParsingRuleLabel, ParsedModule, ParserError, ParserErrorReas mod arguments; mod attributes; mod doc_comments; +mod enums; mod expression; mod function; mod generics; @@ -83,6 +84,27 @@ pub struct Parser<'a> { next_token: SpannedToken, current_token_span: Span, previous_token_span: Span, + + /// The current statement's doc comments. + /// This is used to eventually know if an `unsafe { ... }` expression is documented + /// in its containing statement. For example: + /// + /// ```noir + /// /// Safety: test + /// let x = unsafe { call() }; + /// ``` + statement_doc_comments: Option, +} + +#[derive(Debug)] +pub(crate) struct StatementDocComments { + pub(crate) doc_comments: Vec, + pub(crate) start_span: Span, + pub(crate) end_span: Span, + + /// Were these doc comments "read" by an unsafe statement? + /// If not, these doc comments aren't documenting anything and they produce an error. + pub(crate) read: bool, } impl<'a> Parser<'a> { @@ -107,6 +129,7 @@ impl<'a> Parser<'a> { next_token: eof_spanned_token(), current_token_span: Default::default(), previous_token_span: Default::default(), + statement_doc_comments: None, }; parser.read_two_first_tokens(); parser @@ -128,9 +151,12 @@ impl<'a> Parser<'a> { } /// Invokes `parsing_function` (`parsing_function` must be some `parse_*` method of the parser) - /// and returns the result if the parser has no errors, and if the parser consumed all tokens. + /// and returns the result (and any warnings) if the parser has no errors, and if the parser consumed all tokens. /// Otherwise returns the list of errors. - pub fn parse_result(mut self, parsing_function: F) -> Result> + pub fn parse_result( + mut self, + parsing_function: F, + ) -> Result<(T, Vec), Vec> where F: FnOnce(&mut Parser<'a>) -> T, { @@ -140,26 +166,14 @@ impl<'a> Parser<'a> { return Err(self.errors); } - if self.errors.is_empty() { - Ok(item) + let all_warnings = self.errors.iter().all(|error| error.is_warning()); + if all_warnings { + Ok((item, self.errors)) } else { Err(self.errors) } } - /// Invokes `parsing_function` (`parsing_function` must be some `parse_*` method of the parser) - /// and returns the result if the parser has no errors, and if the parser consumed all tokens. - /// Otherwise returns None. - pub fn parse_option(self, parsing_function: F) -> Option - where - F: FnOnce(&mut Parser<'a>) -> Option, - { - match self.parse_result(parsing_function) { - Ok(item) => item, - Err(_) => None, - } - } - /// Bumps this parser by one token. Returns the token that was previously the "current" token. fn bump(&mut self) -> SpannedToken { self.previous_token_span = self.current_token_span; @@ -178,14 +192,10 @@ impl<'a> Parser<'a> { fn read_token_internal(&mut self) -> SpannedToken { loop { - let token = self.tokens.next(); - if let Some(token) = token { - match token { - Ok(token) => return token, - Err(lexer_error) => self.errors.push(lexer_error.into()), - } - } else { - return eof_spanned_token(); + match self.tokens.next() { + Some(Ok(token)) => return token, + Some(Err(lexer_error)) => self.errors.push(lexer_error.into()), + None => return eof_spanned_token(), } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/attributes.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/attributes.rs index 12cb37edb4b..e32e7d3cb23 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/attributes.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/attributes.rs @@ -92,7 +92,7 @@ impl<'a> Parser<'a> { .into_iter() .filter_map(|(attribute, span)| match attribute { Attribute::Function(..) => { - self.push_error(ParserErrorReason::NoFunctionAttributesAllowedOnStruct, span); + self.push_error(ParserErrorReason::NoFunctionAttributesAllowedOnType, span); None } Attribute::Secondary(attr) => Some(attr), diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/enums.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/enums.rs new file mode 100644 index 00000000000..f95c0f8f72b --- /dev/null +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/enums.rs @@ -0,0 +1,265 @@ +use noirc_errors::Span; + +use crate::{ + ast::{Documented, EnumVariant, Ident, ItemVisibility, NoirEnumeration, UnresolvedGenerics}, + parser::ParserErrorReason, + token::{Attribute, SecondaryAttribute, Token}, +}; + +use super::{ + parse_many::{separated_by_comma_until_right_brace, separated_by_comma_until_right_paren}, + Parser, +}; + +impl<'a> Parser<'a> { + /// Enum = 'enum' identifier Generics '{' EnumVariant* '}' + /// + /// EnumField = OuterDocComments identifier ':' Type + pub(crate) fn parse_enum( + &mut self, + attributes: Vec<(Attribute, Span)>, + visibility: ItemVisibility, + start_span: Span, + ) -> NoirEnumeration { + let attributes = self.validate_secondary_attributes(attributes); + + self.push_error(ParserErrorReason::ExperimentalFeature("Enums"), start_span); + + let Some(name) = self.eat_ident() else { + self.expected_identifier(); + return self.empty_enum( + Ident::default(), + attributes, + visibility, + Vec::new(), + start_span, + ); + }; + + let generics = self.parse_generics(); + + if !self.eat_left_brace() { + self.expected_token(Token::LeftBrace); + return self.empty_enum(name, attributes, visibility, generics, start_span); + } + + let comma_separated = separated_by_comma_until_right_brace(); + let variants = self.parse_many("enum variants", comma_separated, Self::parse_enum_variant); + + NoirEnumeration { + name, + attributes, + visibility, + generics, + variants, + span: self.span_since(start_span), + } + } + + fn parse_enum_variant(&mut self) -> Option> { + let mut doc_comments; + let name; + + // Loop until we find an identifier, skipping anything that's not one + loop { + let doc_comments_start_span = self.current_token_span; + doc_comments = self.parse_outer_doc_comments(); + + if let Some(ident) = self.eat_ident() { + name = ident; + break; + } + + if !doc_comments.is_empty() { + self.push_error( + ParserErrorReason::DocCommentDoesNotDocumentAnything, + self.span_since(doc_comments_start_span), + ); + } + + // Though we do have to stop at EOF + if self.at_eof() { + self.expected_token(Token::RightBrace); + return None; + } + + // Or if we find a right brace + if self.at(Token::RightBrace) { + return None; + } + + self.expected_identifier(); + self.bump(); + } + + let mut parameters = Vec::new(); + + if self.eat_left_paren() { + let comma_separated = separated_by_comma_until_right_paren(); + parameters = self.parse_many("variant parameters", comma_separated, Self::parse_type); + } + + Some(Documented::new(EnumVariant { name, parameters }, doc_comments)) + } + + fn empty_enum( + &self, + name: Ident, + attributes: Vec, + visibility: ItemVisibility, + generics: UnresolvedGenerics, + start_span: Span, + ) -> NoirEnumeration { + NoirEnumeration { + name, + attributes, + visibility, + generics, + variants: Vec::new(), + span: self.span_since(start_span), + } + } +} + +#[cfg(test)] +mod tests { + use crate::{ + ast::{IntegerBitSize, NoirEnumeration, Signedness, UnresolvedGeneric, UnresolvedTypeData}, + parser::{ + parser::{ + parse_program, + tests::{expect_no_errors, get_source_with_error_span}, + }, + ItemKind, ParserErrorReason, + }, + }; + + fn parse_enum_no_errors(src: &str) -> NoirEnumeration { + let (mut module, errors) = parse_program(src); + expect_no_errors(&errors); + assert_eq!(module.items.len(), 1); + let item = module.items.remove(0); + let ItemKind::Enum(noir_enum) = item.kind else { + panic!("Expected enum"); + }; + noir_enum + } + + #[test] + fn parse_empty_enum() { + let src = "enum Foo {}"; + let noir_enum = parse_enum_no_errors(src); + assert_eq!("Foo", noir_enum.name.to_string()); + assert!(noir_enum.variants.is_empty()); + assert!(noir_enum.generics.is_empty()); + } + + #[test] + fn parse_empty_enum_with_generics() { + let src = "enum Foo {}"; + let mut noir_enum = parse_enum_no_errors(src); + assert_eq!("Foo", noir_enum.name.to_string()); + assert!(noir_enum.variants.is_empty()); + assert_eq!(noir_enum.generics.len(), 2); + + let generic = noir_enum.generics.remove(0); + let UnresolvedGeneric::Variable(ident) = generic else { + panic!("Expected generic variable"); + }; + assert_eq!("A", ident.to_string()); + + let generic = noir_enum.generics.remove(0); + let UnresolvedGeneric::Numeric { ident, typ } = generic else { + panic!("Expected generic numeric"); + }; + assert_eq!("B", ident.to_string()); + assert_eq!( + typ.typ, + UnresolvedTypeData::Integer(Signedness::Unsigned, IntegerBitSize::ThirtyTwo) + ); + } + + #[test] + fn parse_enum_with_variants() { + let src = "enum Foo { X(i32), y(Field, u32), Z }"; + let mut noir_enum = parse_enum_no_errors(src); + assert_eq!("Foo", noir_enum.name.to_string()); + assert_eq!(noir_enum.variants.len(), 3); + + let variant = noir_enum.variants.remove(0).item; + assert_eq!("X", variant.name.to_string()); + assert!(matches!( + variant.parameters[0].typ, + UnresolvedTypeData::Integer(Signedness::Signed, IntegerBitSize::ThirtyTwo) + )); + + let variant = noir_enum.variants.remove(0).item; + assert_eq!("y", variant.name.to_string()); + assert!(matches!(variant.parameters[0].typ, UnresolvedTypeData::FieldElement)); + assert!(matches!(variant.parameters[1].typ, UnresolvedTypeData::Integer(..))); + + let variant = noir_enum.variants.remove(0).item; + assert_eq!("Z", variant.name.to_string()); + assert_eq!(variant.parameters.len(), 0); + } + + #[test] + fn parse_empty_enum_with_doc_comments() { + let src = "/// Hello\nenum Foo {}"; + let (module, errors) = parse_program(src); + expect_no_errors(&errors); + assert_eq!(module.items.len(), 1); + let item = &module.items[0]; + assert_eq!(item.doc_comments.len(), 1); + let ItemKind::Enum(noir_enum) = &item.kind else { + panic!("Expected enum"); + }; + assert_eq!("Foo", noir_enum.name.to_string()); + } + + #[test] + fn parse_unclosed_enum() { + let src = "enum Foo {"; + let (module, errors) = parse_program(src); + assert_eq!(errors.len(), 2); + assert_eq!(module.items.len(), 1); + let item = &module.items[0]; + let ItemKind::Enum(noir_enum) = &item.kind else { + panic!("Expected enum"); + }; + assert_eq!("Foo", noir_enum.name.to_string()); + } + + #[test] + fn parse_error_no_function_attributes_allowed_on_enum() { + let src = " + #[test] enum Foo {} + ^^^^^^^ + "; + let (src, _) = get_source_with_error_span(src); + let (_, errors) = parse_program(&src); + let reason = errors[0].reason().unwrap(); + assert!(matches!(reason, ParserErrorReason::NoFunctionAttributesAllowedOnType)); + } + + #[test] + fn recovers_on_non_field() { + let src = " + enum Foo { 42 X(i32) } + ^^ + "; + let (src, _) = get_source_with_error_span(src); + let (module, errors) = parse_program(&src); + + assert_eq!(module.items.len(), 1); + let item = &module.items[0]; + let ItemKind::Enum(noir_enum) = &item.kind else { + panic!("Expected enum"); + }; + assert_eq!("Foo", noir_enum.name.to_string()); + assert_eq!(noir_enum.variants.len(), 1); + + let error = &errors[1]; + assert_eq!(error.to_string(), "Expected an identifier but found '42'"); + } +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/expression.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/expression.rs index 526a0c3dd6e..15be2155a15 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/expression.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/expression.rs @@ -267,6 +267,21 @@ impl<'a> Parser<'a> { } fn parse_atom_kind(&mut self, allow_constructors: bool) -> Option { + let span_before_doc_comments = self.current_token_span; + let doc_comments = self.parse_outer_doc_comments(); + let has_doc_comments = !doc_comments.is_empty(); + + if let Some(kind) = self.parse_unsafe_expr(&doc_comments, span_before_doc_comments) { + return Some(kind); + } + + if has_doc_comments { + self.push_error( + ParserErrorReason::DocCommentDoesNotDocumentAnything, + self.span_since(span_before_doc_comments), + ); + } + if let Some(literal) = self.parse_literal() { return Some(literal); } @@ -275,10 +290,6 @@ impl<'a> Parser<'a> { return Some(kind); } - if let Some(kind) = self.parse_unsafe_expr() { - return Some(kind); - } - if let Some(kind) = self.parse_path_expr(allow_constructors) { return Some(kind); } @@ -370,12 +381,43 @@ impl<'a> Parser<'a> { } /// UnsafeExpression = 'unsafe' Block - fn parse_unsafe_expr(&mut self) -> Option { + fn parse_unsafe_expr( + &mut self, + doc_comments: &[String], + span_before_doc_comments: Span, + ) -> Option { + let start_span = self.current_token_span; + if !self.eat_keyword(Keyword::Unsafe) { return None; } - let start_span = self.current_token_span; + if doc_comments.is_empty() { + if let Some(statement_doc_comments) = &mut self.statement_doc_comments { + statement_doc_comments.read = true; + + let doc_comments = &statement_doc_comments.doc_comments; + let span_before_doc_comments = statement_doc_comments.start_span; + let span_after_doc_comments = statement_doc_comments.end_span; + + if !doc_comments[0].trim().to_lowercase().starts_with("safety:") { + self.push_error( + ParserErrorReason::UnsafeDocCommentDoesNotStartWithSafety, + Span::from( + span_before_doc_comments.start()..span_after_doc_comments.start(), + ), + ); + } + } else { + self.push_error(ParserErrorReason::MissingSafetyComment, start_span); + } + } else if !doc_comments[0].trim().to_lowercase().starts_with("safety:") { + self.push_error( + ParserErrorReason::UnsafeDocCommentDoesNotStartWithSafety, + self.span_since(span_before_doc_comments), + ); + } + if let Some(block) = self.parse_block() { Some(ExpressionKind::Unsafe(block, self.span_since(start_span))) } else { @@ -389,9 +431,7 @@ impl<'a> Parser<'a> { /// /// VariableExpression = Path fn parse_path_expr(&mut self, allow_constructors: bool) -> Option { - let Some(path) = self.parse_path() else { - return None; - }; + let path = self.parse_path()?; if allow_constructors && self.eat_left_brace() { let typ = UnresolvedType::from_path(path); @@ -421,9 +461,7 @@ impl<'a> Parser<'a> { } fn parse_constructor_field(&mut self) -> Option<(Ident, Expression)> { - let Some(ident) = self.eat_ident() else { - return None; - }; + let ident = self.eat_ident()?; Some(if self.eat_colon() { let expression = self.parse_expression_or_error(); @@ -534,9 +572,7 @@ impl<'a> Parser<'a> { /// TypePathExpression = PrimitiveType '::' identifier ( '::' GenericTypeArgs )? fn parse_type_path_expr(&mut self) -> Option { let start_span = self.current_token_span; - let Some(typ) = self.parse_primitive_type() else { - return None; - }; + let typ = self.parse_primitive_type()?; let typ = UnresolvedType { typ, span: self.span_since(start_span) }; self.eat_or_error(Token::DoubleColon); @@ -963,7 +999,21 @@ mod tests { #[test] fn parses_unsafe_expression() { - let src = "unsafe { 1 }"; + let src = " + /// Safety: test + unsafe { 1 }"; + let expr = parse_expression_no_errors(src); + let ExpressionKind::Unsafe(block, _) = expr.kind else { + panic!("Expected unsafe expression"); + }; + assert_eq!(block.statements.len(), 1); + } + + #[test] + fn parses_unsafe_expression_with_doc_comment() { + let src = " + /// Safety: test + unsafe { 1 }"; let expr = parse_expression_no_errors(src); let ExpressionKind::Unsafe(block, _) = expr.kind else { panic!("Expected unsafe expression"); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/function.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/function.rs index 438757b5d79..29e864200f3 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/function.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/function.rs @@ -94,6 +94,17 @@ impl<'a> Parser<'a> { let generics = self.parse_generics(); let parameters = self.parse_function_parameters(allow_self); + let parameters = match parameters { + Some(parameters) => parameters, + None => { + self.push_error( + ParserErrorReason::MissingParametersForFunctionDefinition, + name.span(), + ); + Vec::new() + } + }; + let (return_type, return_visibility) = if self.eat(Token::Arrow) { let visibility = self.parse_visibility(); (FunctionReturnType::Ty(self.parse_type_or_error()), visibility) @@ -131,14 +142,14 @@ impl<'a> Parser<'a> { /// FunctionParametersList = FunctionParameter ( ',' FunctionParameter )* ','? /// /// FunctionParameter = Visibility PatternOrSelf ':' Type - fn parse_function_parameters(&mut self, allow_self: bool) -> Vec { + fn parse_function_parameters(&mut self, allow_self: bool) -> Option> { if !self.eat_left_paren() { - return Vec::new(); + return None; } - self.parse_many("parameters", separated_by_comma_until_right_paren(), |parser| { + Some(self.parse_many("parameters", separated_by_comma_until_right_paren(), |parser| { parser.parse_function_parameter(allow_self) - }) + })) } fn parse_function_parameter(&mut self, allow_self: bool) -> Option { @@ -490,4 +501,16 @@ mod tests { assert!(noir_function.def.is_unconstrained); assert_eq!(noir_function.def.visibility, ItemVisibility::Public); } + + #[test] + fn parse_function_without_parentheses() { + let src = " + fn foo {} + ^^^ + "; + let (src, span) = get_source_with_error_span(src); + let (_, errors) = parse_program(&src); + let reason = get_single_error_reason(&errors, span); + assert!(matches!(reason, ParserErrorReason::MissingParametersForFunctionDefinition)); + } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/global.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/global.rs index 5ece510022a..3a4f6d922a4 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/global.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/global.rs @@ -23,6 +23,7 @@ impl<'a> Parser<'a> { // and throw the error in name resolution. let attributes = self.validate_secondary_attributes(attributes); + let is_global_let = true; let Some(ident) = self.eat_ident() else { self.eat_semicolons(); @@ -35,6 +36,7 @@ impl<'a> Parser<'a> { expression: Expression { kind: ExpressionKind::Error, span: Span::default() }, attributes, comptime, + is_global_let, }; }; @@ -53,7 +55,7 @@ impl<'a> Parser<'a> { self.expected_token(Token::Semicolon); } - LetStatement { pattern, r#type: typ, expression, attributes, comptime } + LetStatement { pattern, r#type: typ, expression, attributes, comptime, is_global_let } } } @@ -105,6 +107,7 @@ mod tests { assert_eq!("foo", name.to_string()); assert!(matches!(let_statement.r#type.typ, UnresolvedTypeData::Unspecified)); assert!(!let_statement.comptime); + assert!(let_statement.is_global_let); assert_eq!(visibility, ItemVisibility::Private); } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/impls.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/impls.rs index 8e6b3bae0e9..278c20e1e27 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/impls.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/impls.rs @@ -245,7 +245,7 @@ impl<'a> Parser<'a> { let noir_function = self.parse_function( attributes, - ItemVisibility::Public, + modifiers.visibility, modifiers.comptime.is_some(), modifiers.unconstrained.is_some(), true, // allow_self @@ -482,7 +482,7 @@ mod tests { panic!("Expected function"); }; assert_eq!(function.def.name.to_string(), "foo"); - assert_eq!(function.def.visibility, ItemVisibility::Public); + assert_eq!(function.def.visibility, ItemVisibility::Private); } #[test] diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/item.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/item.rs index ce712b559d8..d928d8e82d3 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/item.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/item.rs @@ -107,6 +107,7 @@ impl<'a> Parser<'a> { /// ( Use /// | ModOrContract /// | Struct + /// | Enum /// | Impl /// | Trait /// | Global @@ -148,6 +149,16 @@ impl<'a> Parser<'a> { ))]; } + if self.eat_keyword(Keyword::Enum) { + self.comptime_mutable_and_unconstrained_not_applicable(modifiers); + + return vec![ItemKind::Enum(self.parse_enum( + attributes, + modifiers.visibility, + start_span, + ))]; + } + if self.eat_keyword(Keyword::Impl) { self.comptime_mutable_and_unconstrained_not_applicable(modifiers); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/module.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/module.rs index 263338863c0..da733168099 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/module.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/module.rs @@ -74,7 +74,6 @@ mod tests { fn parse_submodule() { let src = "mod foo { mod bar; }"; let (module, errors) = parse_program(src); - dbg!(&errors); expect_no_errors(&errors); assert_eq!(module.items.len(), 1); let item = &module.items[0]; @@ -90,7 +89,6 @@ mod tests { fn parse_contract() { let src = "contract foo {}"; let (module, errors) = parse_program(src); - dbg!(&errors); expect_no_errors(&errors); assert_eq!(module.items.len(), 1); let item = &module.items[0]; diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/pattern.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/pattern.rs index c4dcab55d73..50779e9ccff 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/pattern.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/pattern.rs @@ -154,9 +154,7 @@ impl<'a> Parser<'a> { /// InternedPattern = interned_pattern fn parse_interned_pattern(&mut self) -> Option { - let Some(token) = self.eat_kind(TokenKind::InternedPattern) else { - return None; - }; + let token = self.eat_kind(TokenKind::InternedPattern)?; match token.into_token() { Token::InternedPattern(pattern) => { diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/statement.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/statement.rs index ff6117c9348..8c6d18d90ef 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/statement.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/statement.rs @@ -10,7 +10,7 @@ use crate::{ token::{Attribute, Keyword, Token, TokenKind}, }; -use super::Parser; +use super::{Parser, StatementDocComments}; impl<'a> Parser<'a> { pub(crate) fn parse_statement_or_error(&mut self) -> Statement { @@ -25,10 +25,38 @@ impl<'a> Parser<'a> { /// Statement = Attributes StatementKind ';'? pub(crate) fn parse_statement(&mut self) -> Option<(Statement, (Option, Span))> { loop { + let span_before_doc_comments = self.current_token_span; + let doc_comments = self.parse_outer_doc_comments(); + let span_after_doc_comments = self.current_token_span; + if doc_comments.is_empty() { + self.statement_doc_comments = None; + } else { + self.statement_doc_comments = Some(StatementDocComments { + doc_comments, + start_span: span_before_doc_comments, + end_span: span_after_doc_comments, + read: false, + }); + } + let attributes = self.parse_attributes(); let start_span = self.current_token_span; let kind = self.parse_statement_kind(attributes); + if let Some(statement_doc_comments) = &self.statement_doc_comments { + if !statement_doc_comments.read { + self.push_error( + ParserErrorReason::DocCommentDoesNotDocumentAnything, + Span::from( + statement_doc_comments.start_span.start() + ..statement_doc_comments.end_span.start(), + ), + ); + } + } + + self.statement_doc_comments = None; + let (semicolon_token, semicolon_span) = if self.at(Token::Semicolon) { let token = self.token.clone(); self.bump(); @@ -64,6 +92,7 @@ impl<'a> Parser<'a> { /// | ConstrainStatement /// | ComptimeStatement /// | ForStatement + /// | LoopStatement /// | IfStatement /// | BlockStatement /// | AssignStatement @@ -128,6 +157,10 @@ impl<'a> Parser<'a> { return Some(StatementKind::For(for_loop)); } + if let Some(block) = self.parse_loop() { + return Some(StatementKind::Loop(block)); + } + if let Some(kind) = self.parse_if_expr() { return Some(StatementKind::Expression(Expression { kind, @@ -259,6 +292,29 @@ impl<'a> Parser<'a> { Some(ForLoopStatement { identifier, range, block, span: self.span_since(start_span) }) } + /// LoopStatement = 'loop' Block + fn parse_loop(&mut self) -> Option { + let start_span = self.current_token_span; + if !self.eat_keyword(Keyword::Loop) { + return None; + } + + self.push_error(ParserErrorReason::ExperimentalFeature("loops"), start_span); + + let block_start_span = self.current_token_span; + let block = if let Some(block) = self.parse_block() { + Expression { + kind: ExpressionKind::Block(block), + span: self.span_since(block_start_span), + } + } else { + self.expected_token(Token::LeftBrace); + Expression { kind: ExpressionKind::Error, span: self.span_since(block_start_span) } + }; + + Some(block) + } + /// ForRange /// = ExpressionExceptConstructor /// | ExpressionExceptConstructor '..' ExpressionExceptConstructor @@ -364,7 +420,14 @@ impl<'a> Parser<'a> { Expression { kind: ExpressionKind::Error, span: self.current_token_span } }; - Some(LetStatement { pattern, r#type, expression, attributes, comptime: false }) + Some(LetStatement { + pattern, + r#type, + expression, + attributes, + comptime: false, + is_global_let: false, + }) } /// ConstrainStatement @@ -373,9 +436,7 @@ impl<'a> Parser<'a> { /// | 'assert_eq' Arguments fn parse_constrain_statement(&mut self) -> Option { let start_span = self.current_token_span; - let Some(kind) = self.parse_constrain_kind() else { - return None; - }; + let kind = self.parse_constrain_kind()?; Some(match kind { ConstrainKind::Assert | ConstrainKind::AssertEq => { @@ -476,6 +537,17 @@ mod tests { assert!(!let_statement.comptime); } + #[test] + fn parses_let_statement_with_unsafe() { + let src = "/// Safety: doc comment + let x = unsafe { 1 };"; + let statement = parse_statement_no_errors(src); + let StatementKind::Let(let_statement) = statement.kind else { + panic!("Expected let statement"); + }; + assert_eq!(let_statement.pattern.to_string(), "x"); + } + #[test] fn parses_assert() { let src = "assert(true, \"good\")"; @@ -628,6 +700,20 @@ mod tests { }; } + #[test] + fn parses_assignment_with_unsafe() { + let src = "/// Safety: test + x = unsafe { 1 }"; + let statement = parse_statement_no_errors(src); + let StatementKind::Assign(assign) = statement.kind else { + panic!("Expected assign"); + }; + let LValue::Ident(ident) = assign.lvalue else { + panic!("Expected ident"); + }; + assert_eq!(ident.to_string(), "x"); + } + #[test] fn parses_op_assignment() { let src = "x += 1"; @@ -648,6 +734,16 @@ mod tests { assert_eq!(assign.to_string(), "x = (x >> 1)"); } + #[test] + fn parses_op_assignment_with_unsafe() { + let src = "/// Safety: comment + x += unsafe { 1 }"; + let statement = parse_statement_no_errors(src); + let StatementKind::Assign(_) = statement.kind else { + panic!("Expected assign"); + }; + } + #[test] fn parses_if_statement_followed_by_tuple() { // This shouldn't be parsed as a call @@ -722,4 +818,32 @@ mod tests { assert!(statement.is_none()); assert_eq!(parser.errors.len(), 2); } + + #[test] + fn parses_empty_loop() { + let src = "loop { }"; + let mut parser = Parser::for_str(src); + let statement = parser.parse_statement_or_error(); + let StatementKind::Loop(block) = statement.kind else { + panic!("Expected loop"); + }; + let ExpressionKind::Block(block) = block.kind else { + panic!("Expected block"); + }; + assert!(block.statements.is_empty()); + } + + #[test] + fn parses_loop_with_statements() { + let src = "loop { 1; 2 }"; + let mut parser = Parser::for_str(src); + let statement = parser.parse_statement_or_error(); + let StatementKind::Loop(block) = statement.kind else { + panic!("Expected loop"); + }; + let ExpressionKind::Block(block) = block.kind else { + panic!("Expected block"); + }; + assert_eq!(block.statements.len(), 2); + } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/structs.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/structs.rs index da8ac64e021..b066565e680 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/structs.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/structs.rs @@ -251,7 +251,7 @@ mod tests { let (src, span) = get_source_with_error_span(src); let (_, errors) = parse_program(&src); let reason = get_single_error_reason(&errors, span); - assert!(matches!(reason, ParserErrorReason::NoFunctionAttributesAllowedOnStruct)); + assert!(matches!(reason, ParserErrorReason::NoFunctionAttributesAllowedOnType)); } #[test] diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/tests.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/tests.rs index ea8b1fc638d..7308458e948 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/tests.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/tests.rs @@ -44,7 +44,7 @@ pub(super) fn get_single_error_reason( } pub(super) fn expect_no_errors(errors: &[ParserError]) { - if errors.is_empty() { + if errors.is_empty() || errors.iter().all(|error| error.is_warning()) { return; } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/traits.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/traits.rs index e03b629e9ea..6f6a9bab960 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/traits.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/traits.rs @@ -225,6 +225,10 @@ impl<'a> Parser<'a> { false, // allow mut ); + if modifiers.visibility != ItemVisibility::Private { + self.push_error(ParserErrorReason::TraitVisibilityIgnored, modifiers.visibility_span); + } + if !self.eat_keyword(Keyword::Fn) { self.modifiers_not_followed_by_an_item(modifiers); return None; @@ -285,7 +289,11 @@ mod tests { use crate::{ ast::{NoirTrait, NoirTraitImpl, TraitItem}, parser::{ - parser::{parse_program, tests::expect_no_errors, ParserErrorReason}, + parser::{ + parse_program, + tests::{expect_no_errors, get_single_error, get_source_with_error_span}, + ParserErrorReason, + }, ItemKind, }, }; @@ -513,7 +521,19 @@ mod tests { } #[test] - fn parse_trait_inheirtance() { + fn parse_trait_function_with_visibility() { + let src = " + trait Foo { pub fn foo(); } + ^^^ + "; + let (src, span) = get_source_with_error_span(src); + let (_module, errors) = parse_program(&src); + let error = get_single_error(&errors, span); + assert!(error.to_string().contains("Visibility is ignored on a trait method")); + } + + #[test] + fn parse_trait_inheritance() { let src = "trait Foo: Bar + Baz {}"; let noir_trait = parse_trait_no_errors(src); assert_eq!(noir_trait.bounds.len(), 2); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/type_expression.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/type_expression.rs index 7dd59aedb45..83b04bb157a 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/type_expression.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/type_expression.rs @@ -158,9 +158,7 @@ impl<'a> Parser<'a> { /// ConstantTypeExpression = int fn parse_constant_type_expression(&mut self) -> Option { - let Some(int) = self.eat_int() else { - return None; - }; + let int = self.eat_int()?; Some(UnresolvedTypeExpression::Constant(int, self.previous_token_span)) } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/types.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/types.rs index 884db763698..f325c7e60ca 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/types.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/types.rs @@ -208,6 +208,9 @@ impl<'a> Parser<'a> { if self.eat_keyword(Keyword::StructDefinition) { return Some(UnresolvedTypeData::Quoted(QuotedType::StructDefinition)); } + if self.eat_keyword(Keyword::EnumDefinition) { + return Some(UnresolvedTypeData::Quoted(QuotedType::EnumDefinition)); + } if self.eat_keyword(Keyword::TraitConstraint) { return Some(UnresolvedTypeData::Quoted(QuotedType::TraitConstraint)); } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/where_clause.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/where_clause.rs index a753ffb6fd2..8945e6f29f5 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/where_clause.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/parser/where_clause.rs @@ -36,9 +36,7 @@ impl<'a> Parser<'a> { } fn parse_single_where_clause(&mut self) -> Option<(UnresolvedType, Vec)> { - let Some(typ) = self.parse_type() else { - return None; - }; + let typ = self.parse_type()?; self.eat_or_error(Token::Colon); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/tests.rs b/noir/noir-repo/compiler/noirc_frontend/src/tests.rs index 8ddf1b571e6..29ba7b0fab1 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/tests.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/tests.rs @@ -65,6 +65,9 @@ pub(crate) fn get_program(src: &str) -> (ParsedModule, Context, Vec<(Compilation ) } +/// Compile a program. +/// +/// The stdlib is not available for these snippets. pub(crate) fn get_program_with_maybe_parser_errors( src: &str, allow_parser_errors: bool, @@ -114,7 +117,7 @@ pub(crate) fn get_program_with_maybe_parser_errors( }; let debug_comptime_in_file = None; - let error_on_unused_imports = true; + let pedantic_solving = true; // Now we want to populate the CrateDefMap using the DefCollector errors.extend(DefCollector::collect_crate_and_dependencies( @@ -123,7 +126,7 @@ pub(crate) fn get_program_with_maybe_parser_errors( program.clone().into_sorted(), root_file_id, debug_comptime_in_file, - error_on_unused_imports, + pedantic_solving, )); } (program, context, errors) @@ -900,6 +903,7 @@ fn find_lambda_captures(stmts: &[StmtId], interner: &NodeInterner, result: &mut HirStatement::Constrain(constr_stmt) => constr_stmt.0, HirStatement::Semi(semi_expr) => semi_expr, HirStatement::For(for_loop) => for_loop.block, + HirStatement::Loop(block) => block, HirStatement::Error => panic!("Invalid HirStatement!"), HirStatement::Break => panic!("Unexpected break"), HirStatement::Continue => panic!("Unexpected continue"), @@ -1244,7 +1248,7 @@ fn resolve_fmt_strings() { fn monomorphize_program(src: &str) -> Result { let (_program, mut context, _errors) = get_program(src); let main_func_id = context.def_interner.find_function("main").unwrap(); - monomorphize(main_func_id, &mut context.def_interner) + monomorphize(main_func_id, &mut context.def_interner, false) } fn get_monomorphization_error(src: &str) -> Option { @@ -1287,11 +1291,15 @@ fn lambda$f1(mut env$l1: (Field)) -> Field { check_rewrite(src, expected_rewrite); } +// TODO(https://github.com/noir-lang/noir/issues/6780): currently failing +// with a stack overflow #[test] +#[ignore] fn deny_cyclic_globals() { let src = r#" global A: u32 = B; global B: u32 = A; + fn main() {} "#; @@ -1991,9 +1999,6 @@ fn numeric_generic_u16_array_size() { )); } -// TODO(https://github.com/noir-lang/noir/issues/6238): -// The EvaluatedGlobalIsntU32 warning is a stopgap -// (originally from https://github.com/noir-lang/noir/issues/6125) #[test] fn numeric_generic_field_larger_than_u32() { let src = r#" @@ -2006,29 +2011,16 @@ fn numeric_generic_field_larger_than_u32() { } "#; let errors = get_program_errors(src); - assert_eq!(errors.len(), 2); - assert!(matches!( - errors[0].0, - CompilationError::TypeError(TypeCheckError::EvaluatedGlobalIsntU32 { .. }), - )); - assert!(matches!( - errors[1].0, - CompilationError::ResolverError(ResolverError::IntegerTooLarge { .. }) - )); + assert_eq!(errors.len(), 0); } -// TODO(https://github.com/noir-lang/noir/issues/6238): -// The EvaluatedGlobalIsntU32 warning is a stopgap -// (originally from https://github.com/noir-lang/noir/issues/6126) #[test] fn numeric_generic_field_arithmetic_larger_than_u32() { let src = r#" struct Foo {} - impl Foo { - fn size(self) -> Field { - F - } + fn size(_x: Foo) -> Field { + F } // 2^32 - 1 @@ -2039,21 +2031,11 @@ fn numeric_generic_field_arithmetic_larger_than_u32() { } fn main() { - let _ = foo::().size(); + let _ = size(foo::()); } "#; let errors = get_program_errors(src); - assert_eq!(errors.len(), 2); - - assert!(matches!( - errors[0].0, - CompilationError::TypeError(TypeCheckError::EvaluatedGlobalIsntU32 { .. }), - )); - - assert!(matches!( - errors[1].0, - CompilationError::ResolverError(ResolverError::UnusedVariable { .. }) - )); + assert_eq!(errors.len(), 0); } #[test] @@ -2180,25 +2162,11 @@ fn numeric_generics_type_kind_mismatch() { } "#; let errors = get_program_errors(src); - assert_eq!(errors.len(), 3); - - // TODO(https://github.com/noir-lang/noir/issues/6238): - // The EvaluatedGlobalIsntU32 warning is a stopgap + assert_eq!(errors.len(), 1); assert!(matches!( errors[0].0, - CompilationError::TypeError(TypeCheckError::EvaluatedGlobalIsntU32 { .. }), - )); - - assert!(matches!( - errors[1].0, CompilationError::TypeError(TypeCheckError::TypeKindMismatch { .. }), )); - - // TODO(https://github.com/noir-lang/noir/issues/6238): see above - assert!(matches!( - errors[2].0, - CompilationError::TypeError(TypeCheckError::EvaluatedGlobalIsntU32 { .. }), - )); } #[test] @@ -2880,6 +2848,21 @@ fn trait_constraint_on_tuple_type() { assert_no_errors(src); } +#[test] +fn trait_constraint_on_tuple_type_pub_crate() { + let src = r#" + pub(crate) trait Foo { + fn foo(self, x: A) -> bool; + } + + pub fn bar(x: (T, U), y: V) -> bool where (T, U): Foo { + x.foo(y) + } + + fn main() {}"#; + assert_no_errors(src); +} + #[test] fn incorrect_generic_count_on_struct_impl() { let src = r#" @@ -2978,7 +2961,7 @@ fn uses_self_type_inside_trait() { fn uses_self_type_in_trait_where_clause() { let src = r#" pub trait Trait { - fn trait_func() -> bool; + fn trait_func(self) -> bool; } pub trait Foo where Self: Trait { @@ -3033,13 +3016,13 @@ fn do_not_eagerly_error_on_cast_on_type_variable() { #[test] fn error_on_cast_over_type_variable() { let src = r#" - pub fn foo(x: T, f: fn(T) -> U) -> U { + pub fn foo(f: fn(T) -> U, x: T, ) -> U { f(x) } fn main() { let x = "a"; - let _: Field = foo(x, |x| x as Field); + let _: Field = foo(|x| x as Field, x); } "#; @@ -3219,20 +3202,20 @@ fn dont_infer_globals_to_u32_from_type_use() { } "#; - let errors = get_program_errors(src); - assert_eq!(errors.len(), 3); - assert!(matches!( - errors[0].0, - CompilationError::ResolverError(ResolverError::UnspecifiedGlobalType { .. }) - )); - assert!(matches!( - errors[1].0, - CompilationError::ResolverError(ResolverError::UnspecifiedGlobalType { .. }) - )); - assert!(matches!( - errors[2].0, - CompilationError::ResolverError(ResolverError::UnspecifiedGlobalType { .. }) - )); + let mut errors = get_program_errors(src); + assert_eq!(errors.len(), 6); + for (error, _file_id) in errors.drain(0..3) { + assert!(matches!( + error, + CompilationError::ResolverError(ResolverError::UnspecifiedGlobalType { .. }) + )); + } + for (error, _file_id) in errors { + assert!(matches!( + error, + CompilationError::TypeError(TypeCheckError::TypeKindMismatch { .. }) + )); + } } #[test] @@ -3242,7 +3225,8 @@ fn dont_infer_partial_global_types() { pub global NESTED_ARRAY: [[Field; _]; 3] = [[]; 3]; pub global STR: str<_> = "hi"; pub global NESTED_STR: [str<_>] = &["hi"]; - pub global FMT_STR: fmtstr<_, _> = f"hi {ARRAY}"; + pub global FORMATTED_VALUE: str<5> = "there"; + pub global FMT_STR: fmtstr<_, _> = f"hi {FORMATTED_VALUE}"; pub global TUPLE_WITH_MULTIPLE: ([str<_>], [[Field; _]; 3]) = (&["hi"], [[]; 3]); fn main() { } @@ -3318,13 +3302,9 @@ fn non_u32_as_array_length() { "#; let errors = get_program_errors(src); - assert_eq!(errors.len(), 2); + assert_eq!(errors.len(), 1); assert!(matches!( errors[0].0, - CompilationError::TypeError(TypeCheckError::EvaluatedGlobalIsntU32 { .. }) - )); - assert!(matches!( - errors[1].0, CompilationError::TypeError(TypeCheckError::TypeKindMismatch { .. }) )); } @@ -3894,3 +3874,197 @@ fn disallows_underscore_on_right_hand_side() { assert_eq!(name, "_"); } + +#[test] +fn errors_on_cyclic_globals() { + let src = r#" + pub comptime global A: u32 = B; + pub comptime global B: u32 = A; + + fn main() { } + "#; + let errors = get_program_errors(src); + + assert!(errors.iter().any(|(error, _)| matches!( + error, + CompilationError::InterpreterError(InterpreterError::GlobalsDependencyCycle { .. }) + ))); + assert!(errors.iter().any(|(error, _)| matches!( + error, + CompilationError::ResolverError(ResolverError::DependencyCycle { .. }) + ))); +} + +#[test] +fn warns_on_unneeded_unsafe() { + let src = r#" + fn main() { + /// Safety: test + unsafe { + foo() + } + } + + fn foo() {} + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + assert!(matches!( + &errors[0].0, + CompilationError::TypeError(TypeCheckError::UnnecessaryUnsafeBlock { .. }) + )); +} + +#[test] +fn warns_on_nested_unsafe() { + let src = r#" + fn main() { + /// Safety: test + unsafe { + /// Safety: test + unsafe { + foo() + } + } + } + + unconstrained fn foo() {} + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + assert!(matches!( + &errors[0].0, + CompilationError::TypeError(TypeCheckError::NestedUnsafeBlock { .. }) + )); +} + +#[test] +fn mutable_self_call() { + let src = r#" + fn main() { + let mut bar = Bar {}; + let _ = bar.bar(); + } + + struct Bar {} + + impl Bar { + fn bar(&mut self) { + let _ = self; + } + } + "#; + assert_no_errors(src); +} + +#[test] +fn checks_visibility_of_trait_related_to_trait_impl_on_method_call() { + let src = r#" + mod moo { + pub struct Bar {} + } + + trait Foo { + fn foo(self); + } + + impl Foo for moo::Bar { + fn foo(self) {} + } + + fn main() { + let bar = moo::Bar {}; + bar.foo(); + } + "#; + assert_no_errors(src); +} + +#[test] +fn infers_lambda_argument_from_method_call_function_type() { + let src = r#" + struct Foo { + value: Field, + } + + impl Foo { + fn foo(self) -> Field { + self.value + } + } + + struct Box { + value: T, + } + + impl Box { + fn map(self, f: fn(T) -> U) -> Box { + Box { value: f(self.value) } + } + } + + fn main() { + let box = Box { value: Foo { value: 1 } }; + let _ = box.map(|foo| foo.foo()); + } + "#; + assert_no_errors(src); +} + +#[test] +fn infers_lambda_argument_from_call_function_type() { + let src = r#" + struct Foo { + value: Field, + } + + fn call(f: fn(Foo) -> Field) -> Field { + f(Foo { value: 1 }) + } + + fn main() { + let _ = call(|foo| foo.value); + } + "#; + assert_no_errors(src); +} + +#[test] +fn infers_lambda_argument_from_call_function_type_in_generic_call() { + let src = r#" + struct Foo { + value: Field, + } + + fn call(t: T, f: fn(T) -> Field) -> Field { + f(t) + } + + fn main() { + let _ = call(Foo { value: 1 }, |foo| foo.value); + } + "#; + assert_no_errors(src); +} + +#[test] +fn regression_7088() { + // A test for code that initially broke when implementing inferring + // lambda parameter types from the function type related to the call + // the lambda is in (PR #7088). + let src = r#" + struct U60Repr {} + + impl U60Repr { + fn new(_: [Field; N * NumFieldSegments]) -> Self { + U60Repr {} + } + } + + fn main() { + let input: [Field; 6] = [0; 6]; + let _: U60Repr<3, 6> = U60Repr::new(input); + } + "#; + assert_no_errors(src); +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/tests/aliases.rs b/noir/noir-repo/compiler/noirc_frontend/src/tests/aliases.rs index 8d3433299f6..1a5621c8adb 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/tests/aliases.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/tests/aliases.rs @@ -47,3 +47,31 @@ fn alias_in_let_pattern() { "#; assert_no_errors(src); } + +#[test] +fn double_alias_in_path() { + let src = r#" + struct Foo {} + + impl Foo { + fn new() -> Self { + Self {} + } + } + + type FooAlias1 = Foo; + type FooAlias2 = FooAlias1; + + fn main() { + let _ = FooAlias2::new(); + } + "#; + assert_no_errors(src); +} + +#[test] +fn double_generic_alias_in_path() { + let src = r#" + "#; + assert_no_errors(src); +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/tests/arithmetic_generics.rs b/noir/noir-repo/compiler/noirc_frontend/src/tests/arithmetic_generics.rs index 3328fe439ae..83de9c077ab 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/tests/arithmetic_generics.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/tests/arithmetic_generics.rs @@ -153,3 +153,49 @@ fn arithmetic_generics_checked_cast_indirect_zeros() { panic!("unexpected error: {:?}", monomorphization_error); } } + +#[test] +fn global_numeric_generic_larger_than_u32() { + // Regression test for https://github.com/noir-lang/noir/issues/6125 + let source = r#" + global A: Field = 4294967297; + + fn foo() { } + + fn main() { + let _ = foo::(); + } + "#; + let errors = get_program_errors(source); + assert_eq!(errors.len(), 0); +} + +#[test] +fn global_arithmetic_generic_larger_than_u32() { + // Regression test for https://github.com/noir-lang/noir/issues/6126 + let source = r#" + struct Foo {} + + impl Foo { + fn size(self) -> Field { + let _ = self; + F + } + } + + // 2^32 - 1 + global A: Field = 4294967295; + + // Avoiding overflow succeeds: + // fn foo() -> Foo { + fn foo() -> Foo { + Foo {} + } + + fn main() { + let _ = foo::().size(); + } + "#; + let errors = get_program_errors(source); + assert_eq!(errors.len(), 0); +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/tests/metaprogramming.rs b/noir/noir-repo/compiler/noirc_frontend/src/tests/metaprogramming.rs index 89a049ebc9d..8256744e18f 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/tests/metaprogramming.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/tests/metaprogramming.rs @@ -10,6 +10,7 @@ use crate::{ resolution::errors::ResolverError, type_check::TypeCheckError, }, + parser::ParserErrorReason, }; use super::{assert_no_errors, get_program_errors}; @@ -161,3 +162,33 @@ fn uses_correct_type_for_attribute_arguments() { "#; assert_no_errors(src); } + +#[test] +fn does_not_fail_to_parse_macro_on_parser_warning() { + let src = r#" + #[make_bar] + pub unconstrained fn foo() {} + + comptime fn make_bar(_: FunctionDefinition) -> Quoted { + quote { + pub fn bar() { + unsafe { + foo(); + } + } + } + } + + fn main() { + bar() + } + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ParseError(parser_error) = &errors[0].0 else { + panic!("Expected a ParseError, got {:?}", errors[0].0); + }; + + assert!(matches!(parser_error.reason(), Some(ParserErrorReason::MissingSafetyComment))); +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/tests/references.rs b/noir/noir-repo/compiler/noirc_frontend/src/tests/references.rs index ce72240d146..956b2c5f05e 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/tests/references.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/tests/references.rs @@ -87,6 +87,7 @@ fn constrained_reference_to_unconstrained() { fn main(mut x: u32, y: pub u32) { let x_ref = &mut x; if x == 5 { + /// Safety: test context unsafe { mut_ref_input(x_ref, y); } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/tests/traits.rs b/noir/noir-repo/compiler/noirc_frontend/src/tests/traits.rs index 811a32bab86..8d3a63f65cf 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/tests/traits.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/tests/traits.rs @@ -1,5 +1,6 @@ use crate::hir::def_collector::dc_crate::CompilationError; use crate::hir::resolution::errors::ResolverError; +use crate::hir::resolution::import::PathResolutionError; use crate::hir::type_check::TypeCheckError; use crate::tests::{get_program_errors, get_program_with_maybe_parser_errors}; @@ -592,7 +593,7 @@ fn trait_bounds_which_are_dependent_on_generic_types_are_resolved_correctly() { // Regression test for https://github.com/noir-lang/noir/issues/6420 let src = r#" trait Foo { - fn foo() -> Field; + fn foo(self) -> Field; } trait Bar: Foo { @@ -613,7 +614,8 @@ fn trait_bounds_which_are_dependent_on_generic_types_are_resolved_correctly() { where T: MarkerTrait, { - fn foo() -> Field { + fn foo(self) -> Field { + let _ = self; 42 } } @@ -652,3 +654,657 @@ fn does_not_crash_on_as_trait_path_with_empty_path() { ); assert!(!errors.is_empty()); } + +#[test] +fn warns_if_trait_is_not_in_scope_for_function_call_and_there_is_only_one_trait_method() { + let src = r#" + fn main() { + let _ = Bar::foo(); + } + + pub struct Bar { + } + + mod private_mod { + pub trait Foo { + fn foo() -> i32; + } + + impl Foo for super::Bar { + fn foo() -> i32 { + 42 + } + } + } + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::TraitMethodNotInScope { ident, trait_name }, + )) = &errors[0].0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + assert_eq!(trait_name, "private_mod::Foo"); +} + +#[test] +fn calls_trait_function_if_it_is_in_scope() { + let src = r#" + use private_mod::Foo; + + fn main() { + let _ = Bar::foo(); + } + + pub struct Bar { + } + + mod private_mod { + pub trait Foo { + fn foo() -> i32; + } + + impl Foo for super::Bar { + fn foo() -> i32 { + 42 + } + } + } + "#; + assert_no_errors(src); +} + +#[test] +fn calls_trait_function_if_it_is_only_candidate_in_scope() { + let src = r#" + use private_mod::Foo; + + fn main() { + let _ = Bar::foo(); + } + + pub struct Bar { + } + + mod private_mod { + pub trait Foo { + fn foo() -> i32; + } + + impl Foo for super::Bar { + fn foo() -> i32 { + 42 + } + } + + pub trait Foo2 { + fn foo() -> i32; + } + + impl Foo2 for super::Bar { + fn foo() -> i32 { + 42 + } + } + } + "#; + assert_no_errors(src); +} + +#[test] +fn calls_trait_function_if_it_is_only_candidate_in_scope_in_nested_module_using_super() { + let src = r#" + mod moo { + use super::public_mod::Foo; + + pub fn method() { + let _ = super::Bar::foo(); + } + } + + fn main() {} + + pub struct Bar {} + + pub mod public_mod { + pub trait Foo { + fn foo() -> i32; + } + + impl Foo for super::Bar { + fn foo() -> i32 { + 42 + } + } + } + "#; + assert_no_errors(src); +} + +#[test] +fn errors_if_trait_is_not_in_scope_for_function_call_and_there_are_multiple_candidates() { + let src = r#" + fn main() { + let _ = Bar::foo(); + } + + pub struct Bar { + } + + mod private_mod { + pub trait Foo { + fn foo() -> i32; + } + + impl Foo for super::Bar { + fn foo() -> i32 { + 42 + } + } + + pub trait Foo2 { + fn foo() -> i32; + } + + impl Foo2 for super::Bar { + fn foo() -> i32 { + 42 + } + } + } + "#; + let mut errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::UnresolvedWithPossibleTraitsToImport { ident, mut traits }, + )) = errors.remove(0).0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + traits.sort(); + assert_eq!(traits, vec!["private_mod::Foo", "private_mod::Foo2"]); +} + +#[test] +fn errors_if_multiple_trait_methods_are_in_scope_for_function_call() { + let src = r#" + use private_mod::Foo; + use private_mod::Foo2; + + fn main() { + let _ = Bar::foo(); + } + + pub struct Bar { + } + + mod private_mod { + pub trait Foo { + fn foo() -> i32; + } + + impl Foo for super::Bar { + fn foo() -> i32 { + 42 + } + } + + pub trait Foo2 { + fn foo() -> i32; + } + + impl Foo2 for super::Bar { + fn foo() -> i32 { + 42 + } + } + } + "#; + let mut errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::MultipleTraitsInScope { ident, mut traits }, + )) = errors.remove(0).0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + traits.sort(); + assert_eq!(traits, vec!["private_mod::Foo", "private_mod::Foo2"]); +} + +#[test] +fn warns_if_trait_is_not_in_scope_for_method_call_and_there_is_only_one_trait_method() { + let src = r#" + fn main() { + let bar = Bar { x: 42 }; + let _ = bar.foo(); + } + + pub struct Bar { + x: i32, + } + + mod private_mod { + pub trait Foo { + fn foo(self) -> i32; + } + + impl Foo for super::Bar { + fn foo(self) -> i32 { + self.x + } + } + } + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::TraitMethodNotInScope { ident, trait_name }, + )) = &errors[0].0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + assert_eq!(trait_name, "private_mod::Foo"); +} + +#[test] +fn calls_trait_method_if_it_is_in_scope() { + let src = r#" + use private_mod::Foo; + + fn main() { + let bar = Bar { x: 42 }; + let _ = bar.foo(); + } + + pub struct Bar { + x: i32, + } + + mod private_mod { + pub trait Foo { + fn foo(self) -> i32; + } + + impl Foo for super::Bar { + fn foo(self) -> i32 { + self.x + } + } + } + "#; + assert_no_errors(src); +} + +#[test] +fn errors_if_trait_is_not_in_scope_for_method_call_and_there_are_multiple_candidates() { + let src = r#" + fn main() { + let bar = Bar { x: 42 }; + let _ = bar.foo(); + } + + pub struct Bar { + x: i32, + } + + mod private_mod { + pub trait Foo { + fn foo(self) -> i32; + } + + impl Foo for super::Bar { + fn foo(self) -> i32 { + self.x + } + } + + pub trait Foo2 { + fn foo(self) -> i32; + } + + impl Foo2 for super::Bar { + fn foo(self) -> i32 { + self.x + } + } + } + "#; + let mut errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::UnresolvedWithPossibleTraitsToImport { ident, mut traits }, + )) = errors.remove(0).0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + traits.sort(); + assert_eq!(traits, vec!["private_mod::Foo", "private_mod::Foo2"]); +} + +#[test] +fn errors_if_multiple_trait_methods_are_in_scope_for_method_call() { + let src = r#" + use private_mod::Foo; + use private_mod::Foo2; + + fn main() { + let bar = Bar { x : 42 }; + let _ = bar.foo(); + } + + pub struct Bar { + x: i32, + } + + mod private_mod { + pub trait Foo { + fn foo(self) -> i32; + } + + impl Foo for super::Bar { + fn foo(self) -> i32 { + self.x + } + } + + pub trait Foo2 { + fn foo(self) -> i32; + } + + impl Foo2 for super::Bar { + fn foo(self) -> i32 { + self.x + } + } + } + "#; + let mut errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::MultipleTraitsInScope { ident, mut traits }, + )) = errors.remove(0).0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + traits.sort(); + assert_eq!(traits, vec!["private_mod::Foo", "private_mod::Foo2"]); +} + +#[test] +fn calls_trait_method_if_it_is_in_scope_with_multiple_candidates_but_only_one_decided_by_generics() +{ + let src = r#" + struct Foo { + inner: Field, + } + + trait Converter { + fn convert(self) -> N; + } + + impl Converter for Foo { + fn convert(self) -> Field { + self.inner + } + } + + impl Converter for Foo { + fn convert(self) -> u32 { + self.inner as u32 + } + } + + fn main() { + let foo = Foo { inner: 42 }; + let _: u32 = foo.convert(); + } + "#; + assert_no_errors(src); +} + +#[test] +fn type_checks_trait_default_method_and_errors() { + let src = r#" + pub trait Foo { + fn foo(self) -> i32 { + let _ = self; + true + } + } + + fn main() {} + "#; + + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::TypeError(TypeCheckError::TypeMismatchWithSource { + expected, + actual, + .. + }) = &errors[0].0 + else { + panic!("Expected a type mismatch error, got {:?}", errors[0].0); + }; + + assert_eq!(expected.to_string(), "i32"); + assert_eq!(actual.to_string(), "bool"); +} + +#[test] +fn type_checks_trait_default_method_and_does_not_error() { + let src = r#" + pub trait Foo { + fn foo(self) -> i32 { + let _ = self; + 1 + } + } + + fn main() {} + "#; + assert_no_errors(src); +} + +#[test] +fn type_checks_trait_default_method_and_does_not_error_using_self() { + let src = r#" + pub trait Foo { + fn foo(self) -> i32 { + self.bar() + } + + fn bar(self) -> i32 { + let _ = self; + 1 + } + } + + fn main() {} + "#; + assert_no_errors(src); +} + +#[test] +fn warns_if_trait_is_not_in_scope_for_primitive_function_call_and_there_is_only_one_trait_method() { + let src = r#" + fn main() { + let _ = Field::foo(); + } + + mod private_mod { + pub trait Foo { + fn foo() -> i32; + } + + impl Foo for Field { + fn foo() -> i32 { + 42 + } + } + } + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::TraitMethodNotInScope { ident, trait_name }, + )) = &errors[0].0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + assert_eq!(trait_name, "private_mod::Foo"); +} + +#[test] +fn warns_if_trait_is_not_in_scope_for_primitive_method_call_and_there_is_only_one_trait_method() { + let src = r#" + fn main() { + let x: Field = 1; + let _ = x.foo(); + } + + mod private_mod { + pub trait Foo { + fn foo(self) -> i32; + } + + impl Foo for Field { + fn foo(self) -> i32 { + self as i32 + } + } + } + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::TraitMethodNotInScope { ident, trait_name }, + )) = &errors[0].0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + assert_eq!(trait_name, "private_mod::Foo"); +} + +#[test] +fn warns_if_trait_is_not_in_scope_for_generic_function_call_and_there_is_only_one_trait_method() { + let src = r#" + fn main() { + let x: i32 = 1; + let _ = x.foo(); + } + + mod private_mod { + pub trait Foo { + fn foo(self) -> i32; + } + + impl Foo for T { + fn foo(self) -> i32 { + 42 + } + } + } + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 1); + + let CompilationError::ResolverError(ResolverError::PathResolutionError( + PathResolutionError::TraitMethodNotInScope { ident, trait_name }, + )) = &errors[0].0 + else { + panic!("Expected a 'trait method not in scope' error"); + }; + assert_eq!(ident.to_string(), "foo"); + assert_eq!(trait_name, "private_mod::Foo"); +} + +// See https://github.com/noir-lang/noir/issues/6530 +#[test] +fn regression_6530() { + let src = r#" + pub trait From { + fn from(input: T) -> Self; + } + + pub trait Into { + fn into(self) -> T; + } + + impl Into for U + where + T: From, + { + fn into(self) -> T { + T::from(self) + } + } + + struct Foo { + inner: Field, + } + + impl Into for Foo { + fn into(self) -> Field { + self.inner + } + } + + fn main() { + let foo = Foo { inner: 0 }; + + // This works: + let _: Field = Into::::into(foo); + + // This was failing with 'No matching impl': + let _: Field = foo.into(); + } + "#; + let errors = get_program_errors(src); + assert_eq!(errors.len(), 0); +} + +// See https://github.com/noir-lang/noir/issues/7090 +#[test] +#[should_panic] +fn calls_trait_method_using_struct_name_when_multiple_impls_exist() { + let src = r#" + trait From2 { + fn from2(input: T) -> Self; + } + struct U60Repr {} + impl From2<[Field; 3]> for U60Repr { + fn from2(_: [Field; 3]) -> Self { + U60Repr {} + } + } + impl From2 for U60Repr { + fn from2(_: Field) -> Self { + U60Repr {} + } + } + fn main() { + let _ = U60Repr::from2([1, 2, 3]); + let _ = U60Repr::from2(1); + } + "#; + assert_no_errors(src); +} diff --git a/noir/noir-repo/compiler/noirc_frontend/src/tests/visibility.rs b/noir/noir-repo/compiler/noirc_frontend/src/tests/visibility.rs index 824a1de4c37..917394316cf 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/tests/visibility.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/tests/visibility.rs @@ -229,6 +229,29 @@ fn errors_if_pub_trait_returns_private_struct() { assert_type_is_more_private_than_item_error(src, "Bar", "foo"); } +#[test] +fn does_not_error_if_trait_with_default_visibility_returns_struct_with_default_visibility() { + let src = r#" + struct Foo {} + + trait Bar { + fn bar(self) -> Foo; + } + + impl Bar for Foo { + fn bar(self) -> Foo { + self + } + } + + fn main() { + let foo = Foo {}; + let _ = foo.bar(); + } + "#; + assert_no_errors(src); +} + #[test] fn errors_if_trying_to_access_public_function_inside_private_module() { let src = r#" diff --git a/noir/noir-repo/compiler/noirc_printable_type/Cargo.toml b/noir/noir-repo/compiler/noirc_printable_type/Cargo.toml index 8d0574aad64..a1eae750b1f 100644 --- a/noir/noir-repo/compiler/noirc_printable_type/Cargo.toml +++ b/noir/noir-repo/compiler/noirc_printable_type/Cargo.toml @@ -13,10 +13,6 @@ workspace = true [dependencies] acvm.workspace = true -iter-extended.workspace = true serde.workspace = true -serde_json.workspace = true -thiserror.workspace = true -jsonrpc.workspace = true [dev-dependencies] diff --git a/noir/noir-repo/compiler/noirc_printable_type/src/lib.rs b/noir/noir-repo/compiler/noirc_printable_type/src/lib.rs index 6ae187da27f..eb74d2470fb 100644 --- a/noir/noir-repo/compiler/noirc_printable_type/src/lib.rs +++ b/noir/noir-repo/compiler/noirc_printable_type/src/lib.rs @@ -1,10 +1,13 @@ +#![forbid(unsafe_code)] +#![warn(unused_crate_dependencies, unused_extern_crates)] +#![warn(unreachable_pub)] +#![warn(clippy::semicolon_if_nothing_returned)] + use std::{collections::BTreeMap, str}; -use acvm::{acir::AcirField, brillig_vm::brillig::ForeignCallParam}; -use iter_extended::vecmap; +use acvm::AcirField; use serde::{Deserialize, Serialize}; -use thiserror::Error; #[derive(Clone, Debug, PartialEq, Eq, Hash, Serialize, Deserialize)] #[serde(tag = "kind", rename_all = "lowercase")] @@ -66,96 +69,23 @@ pub enum PrintableValueDisplay { Plain(PrintableValue, PrintableType), FmtString(String, Vec<(PrintableValue, PrintableType)>), } - -#[derive(Debug, Error)] -pub enum ForeignCallError { - #[error("No handler could be found for foreign call `{0}`")] - NoHandler(String), - - #[error("Foreign call inputs needed for execution are missing")] - MissingForeignCallInputs, - - #[error("Could not parse PrintableType argument. {0}")] - ParsingError(#[from] serde_json::Error), - - #[error("Failed calling external resolver. {0}")] - ExternalResolverError(#[from] jsonrpc::Error), - - #[error("Assert message resolved after an unsatisified constrain. {0}")] - ResolvedAssertMessage(String), -} - -impl TryFrom<&[ForeignCallParam]> for PrintableValueDisplay { - type Error = ForeignCallError; - - fn try_from(foreign_call_inputs: &[ForeignCallParam]) -> Result { - let (is_fmt_str, foreign_call_inputs) = - foreign_call_inputs.split_last().ok_or(ForeignCallError::MissingForeignCallInputs)?; - - if is_fmt_str.unwrap_field().is_one() { - convert_fmt_string_inputs(foreign_call_inputs) - } else { - convert_string_inputs(foreign_call_inputs) +impl std::fmt::Display for PrintableValueDisplay { + fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + match self { + Self::Plain(value, typ) => { + let output_string = to_string(value, typ).ok_or(std::fmt::Error)?; + write!(fmt, "{output_string}") + } + Self::FmtString(template, values) => { + let mut values_iter = values.iter(); + write_template_replacing_interpolations(template, fmt, || { + values_iter.next().and_then(|(value, typ)| to_string(value, typ)) + }) + } } } } -fn convert_string_inputs( - foreign_call_inputs: &[ForeignCallParam], -) -> Result, ForeignCallError> { - // Fetch the PrintableType from the foreign call input - // The remaining input values should hold what is to be printed - let (printable_type_as_values, input_values) = - foreign_call_inputs.split_last().ok_or(ForeignCallError::MissingForeignCallInputs)?; - let printable_type = fetch_printable_type(printable_type_as_values)?; - - // We must use a flat map here as each value in a struct will be in a separate input value - let mut input_values_as_fields = input_values.iter().flat_map(|param| param.fields()); - - let value = decode_value(&mut input_values_as_fields, &printable_type); - - Ok(PrintableValueDisplay::Plain(value, printable_type)) -} - -fn convert_fmt_string_inputs( - foreign_call_inputs: &[ForeignCallParam], -) -> Result, ForeignCallError> { - let (message, input_and_printable_types) = - foreign_call_inputs.split_first().ok_or(ForeignCallError::MissingForeignCallInputs)?; - - let message_as_fields = message.fields(); - let message_as_string = decode_string_value(&message_as_fields); - - let (num_values, input_and_printable_types) = input_and_printable_types - .split_first() - .ok_or(ForeignCallError::MissingForeignCallInputs)?; - - let mut output = Vec::new(); - let num_values = num_values.unwrap_field().to_u128() as usize; - - let types_start_at = input_and_printable_types.len() - num_values; - let mut input_iter = - input_and_printable_types[0..types_start_at].iter().flat_map(|param| param.fields()); - for printable_type in input_and_printable_types.iter().skip(types_start_at) { - let printable_type = fetch_printable_type(printable_type)?; - let value = decode_value(&mut input_iter, &printable_type); - - output.push((value, printable_type)); - } - - Ok(PrintableValueDisplay::FmtString(message_as_string, output)) -} - -fn fetch_printable_type( - printable_type: &ForeignCallParam, -) -> Result { - let printable_type_as_fields = printable_type.fields(); - let printable_type_as_string = decode_string_value(&printable_type_as_fields); - let printable_type: PrintableType = serde_json::from_str(&printable_type_as_string)?; - - Ok(printable_type) -} - fn to_string(value: &PrintableValue, typ: &PrintableType) -> Option { let mut output = String::new(); match (value, typ) { @@ -193,7 +123,7 @@ fn to_string(value: &PrintableValue, typ: &PrintableType) -> Op (PrintableValue::Vec { array_elements, is_slice }, PrintableType::Array { typ, .. }) | (PrintableValue::Vec { array_elements, is_slice }, PrintableType::Slice { typ }) => { if *is_slice { - output.push('&') + output.push('&'); } output.push('['); let mut values = array_elements.iter().peekable(); @@ -253,23 +183,6 @@ fn to_string(value: &PrintableValue, typ: &PrintableType) -> Op Some(output) } -impl std::fmt::Display for PrintableValueDisplay { - fn fmt(&self, fmt: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { - match self { - Self::Plain(value, typ) => { - let output_string = to_string(value, typ).ok_or(std::fmt::Error)?; - write!(fmt, "{output_string}") - } - Self::FmtString(template, values) => { - let mut values_iter = values.iter(); - write_template_replacing_interpolations(template, fmt, || { - values_iter.next().and_then(|(value, typ)| to_string(value, typ)) - }) - } - } - } -} - fn write_template_replacing_interpolations( template: &str, fmt: &mut std::fmt::Formatter<'_>, @@ -346,94 +259,11 @@ fn format_field_string(field: F) -> String { "0x".to_owned() + &trimmed_field } -/// Assumes that `field_iterator` contains enough field elements in order to decode the [PrintableType] -pub fn decode_value( - field_iterator: &mut impl Iterator, - typ: &PrintableType, -) -> PrintableValue { - match typ { - PrintableType::Field - | PrintableType::SignedInteger { .. } - | PrintableType::UnsignedInteger { .. } - | PrintableType::Boolean => { - let field_element = field_iterator.next().unwrap(); - - PrintableValue::Field(field_element) - } - PrintableType::Array { length, typ } => { - let length = *length as usize; - let mut array_elements = Vec::with_capacity(length); - for _ in 0..length { - array_elements.push(decode_value(field_iterator, typ)); - } - - PrintableValue::Vec { array_elements, is_slice: false } - } - PrintableType::Slice { typ } => { - let length = field_iterator - .next() - .expect("not enough data to decode variable array length") - .to_u128() as usize; - let mut array_elements = Vec::with_capacity(length); - for _ in 0..length { - array_elements.push(decode_value(field_iterator, typ)); - } - - PrintableValue::Vec { array_elements, is_slice: true } - } - PrintableType::Tuple { types } => PrintableValue::Vec { - array_elements: vecmap(types, |typ| decode_value(field_iterator, typ)), - is_slice: false, - }, - PrintableType::String { length } => { - let field_elements: Vec = field_iterator.take(*length as usize).collect(); - - PrintableValue::String(decode_string_value(&field_elements)) - } - PrintableType::Struct { fields, .. } => { - let mut struct_map = BTreeMap::new(); - - for (field_key, param_type) in fields { - let field_value = decode_value(field_iterator, param_type); - - struct_map.insert(field_key.to_owned(), field_value); - } - - PrintableValue::Struct(struct_map) - } - PrintableType::Function { env, .. } => { - let field_element = field_iterator.next().unwrap(); - let func_ref = PrintableValue::Field(field_element); - // we want to consume the fields from the environment, but for now they are not actually printed - decode_value(field_iterator, env); - func_ref - } - PrintableType::MutableReference { typ } => { - // we decode the reference, but it's not really used for printing - decode_value(field_iterator, typ) - } - PrintableType::Unit => PrintableValue::Field(F::zero()), - } -} - -pub fn decode_string_value(field_elements: &[F]) -> String { - // TODO: Replace with `into` when Char is supported - let string_as_slice = vecmap(field_elements, |e| { - let mut field_as_bytes = e.to_be_bytes(); - let char_byte = field_as_bytes.pop().unwrap(); // A character in a string is represented by a u8, thus we just want the last byte of the element - assert!(field_as_bytes.into_iter().all(|b| b == 0)); // Assert that the rest of the field element's bytes are empty - char_byte - }); - - let final_string = str::from_utf8(&string_as_slice).unwrap(); - final_string.to_owned() -} - #[cfg(test)] mod tests { use acvm::FieldElement; - use crate::{PrintableType, PrintableValue, PrintableValueDisplay}; + use super::{PrintableType, PrintableValue, PrintableValueDisplay}; #[test] fn printable_value_display_to_string_without_interpolations() { diff --git a/noir/noir-repo/compiler/wasm/Cargo.toml b/noir/noir-repo/compiler/wasm/Cargo.toml index 9951b23f609..5cde0dfbbcf 100644 --- a/noir/noir-repo/compiler/wasm/Cargo.toml +++ b/noir/noir-repo/compiler/wasm/Cargo.toml @@ -17,9 +17,9 @@ workspace = true crate-type = ["cdylib"] [dependencies] + acvm = { workspace = true, features = ["bn254"] } fm.workspace = true -nargo.workspace = true noirc_driver.workspace = true noirc_frontend = { workspace = true, features = ["bn254"] } noirc_errors.workspace = true @@ -33,6 +33,10 @@ gloo-utils.workspace = true tracing-subscriber.workspace = true tracing-web.workspace = true +# Cannot use the `rpc` feature because the HTTP dependency wouldn't compile to Wasm. +# We could use `path` if `rpc` was a default feature, but we made it opt-in so we don't get any problems when publishing the workspace. +nargo.workspace = true + # This is an unused dependency, we are adding it # so that we can enable the js feature in getrandom. getrandom = { workspace = true, features = ["js"] } diff --git a/noir/noir-repo/compiler/wasm/package.json b/noir/noir-repo/compiler/wasm/package.json index 946ba8dc699..84250dfa7da 100644 --- a/noir/noir-repo/compiler/wasm/package.json +++ b/noir/noir-repo/compiler/wasm/package.json @@ -3,7 +3,7 @@ "contributors": [ "The Noir Team " ], - "version": "1.0.0-beta.0", + "version": "1.0.0-beta.1", "license": "(MIT OR Apache-2.0)", "main": "dist/main.js", "types": "./dist/types/src/index.d.cts", diff --git a/noir/noir-repo/cspell.json b/noir/noir-repo/cspell.json index 5c707e92e21..ed9f7427c6f 100644 --- a/noir/noir-repo/cspell.json +++ b/noir/noir-repo/cspell.json @@ -32,6 +32,7 @@ "boilerplates", "bridgekeeper", "brillig", + "bunx", "bytecount", "cachix", "callsite", @@ -125,6 +126,7 @@ "jmpifs", "jmps", "jsdoc", + "jsonrpsee", "Jubjub", "keccak", "keccakf", @@ -165,6 +167,7 @@ "nomicfoundation", "noncanonical", "nouner", + "oneshot", "overflowing", "pedersen", "peekable", @@ -201,6 +204,7 @@ "Secpr", "signedness", "signorecello", + "smallvec", "smol", "splitn", "srem", @@ -242,9 +246,11 @@ "walkdir", "wasi", "wasmer", + "webapps", "Weierstraß", "whitespace", "whitespaces", + "YOLO", "zkhash", "zshell" ], diff --git a/noir/noir-repo/deny.toml b/noir/noir-repo/deny.toml index 661c8095281..48628fb0045 100644 --- a/noir/noir-repo/deny.toml +++ b/noir/noir-repo/deny.toml @@ -66,14 +66,19 @@ exceptions = [ # so we prefer to not have dependencies using it # https://tldrlegal.com/license/creative-commons-cc0-1.0-universal { allow = ["CC0-1.0"], name = "more-asserts" }, - { allow = ["CC0-1.0"], name = "jsonrpc" }, { allow = ["CC0-1.0"], name = "notify" }, { allow = ["CC0-1.0"], name = "tiny-keccak" }, { allow = ["MPL-2.0"], name = "sized-chunks" }, { allow = ["MPL-2.0"], name = "webpki-roots" }, { allow = ["CDDL-1.0"], name = "inferno" }, + { allow = ["OpenSSL"], name = "ring" }, ] +[[licenses.clarify]] +crate = "ring" +expression = "ISC" +license-files = [{ path = "LICENSE", hash = 0xbd0eed23 }] + # This section is considered when running `cargo deny check sources`. # More documentation about the 'sources' section can be found here: # https://embarkstudios.github.io/cargo-deny/checks/sources/cfg.html diff --git a/noir/noir-repo/docs/docs/explainers/explainer-writing-noir.md b/noir/noir-repo/docs/docs/explainers/explainer-writing-noir.md index 3ce4245dc45..b4265a14dbf 100644 --- a/noir/noir-repo/docs/docs/explainers/explainer-writing-noir.md +++ b/noir/noir-repo/docs/docs/explainers/explainer-writing-noir.md @@ -26,9 +26,9 @@ The equivalent optimization task when writing zk circuits is affectionately refe ### Coding for circuits - a paradigm shift -In zero knowledge cryptography, code is compiled to "circuits" consisting of arithmetic gates, and gate count is the significant cost. Depending on the proving system this is linearly proportionate to proving time, and so from a product point this should be kept as low as possible. +In zero knowledge cryptography, code is compiled to "circuits" consisting of arithmetic gates, and gate count is the significant cost. Depending on the proving system this is linearly proportionate to proof size and proving time, so from a product point of view this should be kept as low as possible. -Whilst writing efficient code for web apps and Solidity has a few key differences, writing efficient circuits have a different set of considerations. It is a bit of a paradigm shift, like writing code for GPUs for the first time... +Whilst writing efficient code for web apps and Solidity have some differences, writing efficient circuits have a different set of considerations. It is a bit of a paradigm shift, like writing code for GPUs for the first time... For example, drawing a circle at (0, 0) of radius `r`: - For a single CPU thread, @@ -57,7 +57,7 @@ For those coming from a primarily web app background, this article will explain ## Translating from Rust -For some applications using Noir, existing code might be a convenient starting point to then proceed to optimize the gate count of. +Programs written in anything from pseudo code to C, can be translated into Noir. A Rust program written for execution can be readily ported to Noir thanks to the similarities in syntax. :::note Many valuable functions and algorithms have been written in more established languages (C/C++), and converted to modern ones (like Rust). @@ -93,23 +93,42 @@ A Noir program compiles to an Abstract Circuit Intermediate Representation which :::tip The command `nargo info` shows the programs circuit size, and is useful to compare the value of changes made. -You can dig deeper and use the `--print-acir` param to take a closer look at individual ACIR opcodes, and the proving backend to see its gate count (eg for barretenberg, `bb gates -b ./target/program.json`). +You can dig deeper and use the `--print-acir` param to take a closer look at individual ACIR opcodes, and the proving backend to see its gate count (eg for barretenberg, the `bb` binary has a `gates` option). ::: -### Use the `Field` type +### Numerical types -Since the native type of values in circuits are `Field`s, using them for variables in Noir means less gates converting them under the hood. -Some things to be mindful of when using a Field type for a regular integer value: -- A variable of type `Field` can be cast `as` an integer type (eg `u8`, `u64`) - - Note: this retains only the bits of the integer type. Eg a Field value of 260 as a `u8` becomes 4 -- For Field types arithmetic operations meaningfully overflow/underflow, yet for integer types they are checked according to their size -- Comparisons and bitwise operations do not exist for `Field`s, cast to an appropriately sized integer type when you need to +As mentioned earlier Noir has many familiar integer types (eg `i8`, `u64`). Ideally after bringing a program into Noir, proving/verifying of its execution just works where needed: client/server side, on an evm, or on the Aztec network. + +A program optimized for execution may leverage the binary representations of integers, reducing the number of clock cycles, and thus time of execution. +The cryptography in a proving backend makes use of a `Field` type, and leveraging this lower level type correctly can reduce gate count, and thus proof size and proving time. + +In some instances simply replacing the integer type with a `Field` could save on some range checks (and hence gates). +Note: when casting a `Field` to an integer type, the value is converted based on the integer binary representation. Eg a Field variable with a value of 260 `as u8` becomes 4 + +### `Field`s for efficiency + +`Field` types have their own underlying representation that is efficient for cryptography, which is different to binary representations efficient for CPUs. So, mathematically speaking, things like bitwise operations do not directly translate to fields. That said, the same outcome can be achieved if wanting to use the Field type as a number with lower overhead. + +For instance shift (`<<`) and or (`|`) work seamlessly with integer types (bit-packing `u8`'s into a `u16`): +``` + high as u16 << 8 | low as u16 +``` + +More efficiently with `Field` types, the equivalent is: +``` + low.assert_max_bit_size::<8>(); // ensure Field values could be represented as 8 bit numbers + high.assert_max_bit_size::<8>(); + (high * 2.pow_32(8) + low) +``` +(Note, the power of two can instead be a constant (256) or global evaluated at compile time) + +The first snippet is good for compatibility when using existing code, converting to the latter can help optimize frequently used functions. :::tip -Where possible, use `Field` type for values. Using smaller value types, and bit-packing strategies, will result in MORE gates +Where possible, use the `Field` type for values. Writing code with smaller value types and bit-packing strategies will result in MORE gates ::: - ### Use Arithmetic over non-arithmetic operations Since circuits are made of arithmetic gates, the cost of arithmetic operations tends to be one gate. Whereas for procedural code, they represent several clock cycles. @@ -140,6 +159,18 @@ Use arrays and indices that are known at compile time where possible. Using `assert_constant(i);` before an index, `i`, is used in an array will give a compile error if `i` is NOT known at compile time. ::: +### Reduce what is inside loops and conditional logic + +Putting less logic inside an `if` (`else`, etc) paths, or inside a loop, translates to less gates required to represent the program. The compiler should mostly take care of this. + +A loop duplicates the gates for each iteration of the loop, or put another way, "unrolls" the loop. Any calculations/calls that are unchanged in the loop should be calculated once before, and the result used in the loop. + +An `if` statement is "flattened" and gates created for each path even if execution uses only one path. Furthermore, there are additional operations required for each path. Sometimes this can have a multiplying effect on the operations in the `if` and `else` etc. + +:::tip +Only have essential computation inside conditional logic and loops, and calculate anything else once (before, or after, depending). +::: + ### Leverage unconstrained execution Constrained verification can leverage unconstrained execution, this is especially useful for operations that are represented by many gates. diff --git a/noir/noir-repo/docs/docs/getting_started/quick_start.md b/noir/noir-repo/docs/docs/getting_started/quick_start.md index 7deeae12fd9..c980b5e7ffc 100644 --- a/noir/noir-repo/docs/docs/getting_started/quick_start.md +++ b/noir/noir-repo/docs/docs/getting_started/quick_start.md @@ -98,7 +98,7 @@ bb prove -b ./target/hello_world.json -w ./target/hello_world.gz -o ./target/pro :::tip -Naming can be confusing, specially as you pass them to the `bb` commands. If unsure, it won't hurt to delete the target folder and start anew to make sure you're using the most recent versions of the compiled circuit and witness. +Naming can be confusing, specially as you pass them to the `bb` commands. If unsure, it won't hurt to delete the target folder and start fresh to make sure you're using the most recent versions of the compiled circuit and witness. ::: diff --git a/noir/noir-repo/docs/docs/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/docs/how_to/how-to-solidity-verifier.mdx similarity index 64% rename from noir/noir-repo/docs/docs/how_to/how-to-solidity-verifier.md rename to noir/noir-repo/docs/docs/how_to/how-to-solidity-verifier.mdx index 2cc0f8e57ce..da36b60920d 100644 --- a/noir/noir-repo/docs/docs/how_to/how-to-solidity-verifier.md +++ b/noir/noir-repo/docs/docs/how_to/how-to-solidity-verifier.mdx @@ -20,24 +20,37 @@ sidebar_position: 0 pagination_next: tutorials/noirjs_app --- -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +Noir is universal. The witness and the compiled program can be fed into a proving backend such as Aztec's [Barretenberg](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg), which can then generate a verifier contract for deployment on blockchains. This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: +Although not strictly in the domain of Noir itself, this guide shows how to generate a Solidity Verifier with Barretenberg and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: +- You will be using Barretenberg as your proving backend +- You will be using an EVM blockchain to verify your proof - You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network - You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/quick_start.md) with Nargo and the example Hello Noir circuit - You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. ## Rundown -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: +Generating a Solidity Verifier with Barretenberg contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: 1. How to generate a solidity smart contract 2. How to compile the smart contract in the RemixIDE 3. How to deploy it to a testnet +:::info[Which proving system to use?] + +Barretenberg currently provides two provers: `UltraPlonk` and `UltraHonk`. In a nutshell, `UltraHonk` is faster and uses less RAM, but its verifier contract is much more expensive. `UltraPlonk` is optimized for on-chain verification, but proving is more expensive. + +In any case, we provide instructions for both. Choose your poison ☠️ + +::: + ## Step 1 - Generate a contract This is by far the most straightforward step. Just run: @@ -46,25 +59,31 @@ This is by far the most straightforward step. Just run: nargo compile ``` -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: +This will compile your source code into a Noir build artifact to be stored in the `./target` directory. From here on, it's Barretenberg's work. You can generate the smart contract using the commands: + + + + +```sh +bb write_vk_ultra_keccak_honk -b ./target/.json +bb contract_ultra_honk +``` + + + ```sh -# Here we pass the path to the newly generated Noir artifact. bb write_vk -b ./target/.json bb contract ``` -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. + + -You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/quick_start.md#proving-backend). +replacing `` with the name of your Noir project. A `Verifier.sol` contract is now in the target folder and can be deployed to any EVM blockchain acting as a verifier smart contract. -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. +You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/quick_start.md#proving-backend). -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: ## Step 2 - Compiling @@ -85,17 +104,12 @@ To compile our the verifier, we can navigate to the compilation tab: ![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: +Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely tell you the contract is too big to deploy on mainnet, or complain about a stack too deep: -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) +![Contract code too big](@site/static/img/how-tos/solidity_verifier_6.png) +![Stack too deep](@site/static/img/how-tos/solidity_verifier_8.png) -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: +To avoid this, you can just use some optimization. Open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. ![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) @@ -103,54 +117,81 @@ This time we will see a warning about an unused function parameter. This is expe At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": +Looking closely, we will notice that our "Solidity Verifier" is composed on multiple contracts working together. Remix will take care of the dependencies for us so we can simply deploy the Verifier contract by selecting it and hitting "deploy": -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) + + -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. +![Deploying HonkVerifier](@site/static/img/how-tos/solidity_verifier_7.png) -:::note + + -Why "UltraVerifier"? +![Deploying PlonkVerifier](@site/static/img/how-tos/solidity_verifier_9.png) -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. + + -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: +A contract will show up in the "Deployed Contracts" section. ## Step 4 - Verifying -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: +To verify a proof using the Solidity verifier contract, we call the `verify` function: ```solidity function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) ``` -When using the default example in the [Hello Noir](../getting_started/quick_start.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. +First generate a proof with `bb`. We need a `Prover.toml` file for our inputs. Run: -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/quick_start.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the +```bash +nargo check +``` + +This will generate a `Prover.toml` you can fill with the values you want to prove. We can now execute the circuit with `nargo` and then use the proving backend to prove: + + + + +```bash +nargo execute +bb prove_ultra_keccak_honk -b ./target/.json -w ./target/ -o ./target/proof +``` + +:::tip[Public inputs] +Barretenberg attaches the public inputs to the proof, which in this case it's not very useful. If you're up for some JS, `bb.js` has [a method for it](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/ts/src/proof/index.ts), but in the CLI you can use this ugly snippet: + +```bash +cat ./target/proof | od -An -v -t x1 | tr -d $' \n' | sed 's/^.\{8\}//' | (read hex; echo "${hex:0:192}${hex:256}") +``` + +Beautiful. This assumes a circuit with one public input (32 bytes, for Barretenberg). For more inputs, you can just increment `hex:256` with 32 more bytes for each public input. + +::: + + + ```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') +nargo execute +bb prove -b ./target/.json -w ./target/ -o ./target/proof +``` + -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS +:::tip[Public inputs] +Barretenberg attaches the public inputs to the proof, which in this case it's not very useful. If you're up for some JS, `bb.js` has [a method for it](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/ts/src/proof/index.ts), but in the CLI you can use this ugly snippet: -echo "Proof:" -echo "0x$HEX_PROOF" +```bash +tail -c +33 ./target/proof | od -An -v -t x1 | tr -d $' \n' ``` +Beautiful. This assumes a circuit with one public input (32 bytes, for Barretenberg). For more inputs, you can just add 32 more bytes for each public input to the `tail` command. + +::: + + + + Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): @@ -188,7 +229,7 @@ the `verify` function will expect the public inputs array (second function param Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. +In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return]`. ::: diff --git a/noir/noir-repo/docs/docs/noir/concepts/control_flow.md b/noir/noir-repo/docs/docs/noir/concepts/control_flow.md index b365bb22728..a11db545e32 100644 --- a/noir/noir-repo/docs/docs/noir/concepts/control_flow.md +++ b/noir/noir-repo/docs/docs/noir/concepts/control_flow.md @@ -29,10 +29,9 @@ if a == 0 { assert(x == 2); ``` -## Loops +## For loops -Noir has one kind of loop: the `for` loop. `for` loops allow you to repeat a block of code multiple -times. +`for` loops allow you to repeat a block of code multiple times. The following block of code between the braces is run 10 times. @@ -48,7 +47,7 @@ The index for loops is of type `u64`. ### Break and Continue -In unconstrained code, `break` and `continue` are also allowed in `for` loops. These are only allowed +In unconstrained code, `break` and `continue` are also allowed in `for` and `loop` loops. These are only allowed in unconstrained code since normal constrained code requires that Noir knows exactly how many iterations a loop may have. `break` and `continue` can be used like so: @@ -77,3 +76,22 @@ above, `continue` will jump to `println("Iteration start")` when used. Note that The iteration variable `i` is still increased by one as normal when `continue` is used. `break` and `continue` cannot currently be used to jump out of more than a single loop at a time. + +## Loops + +In unconstrained code, `loop` is allowed for loops that end with a `break`. This is only allowed +in unconstrained code since normal constrained code requires that Noir knows exactly how many iterations +a loop may have. + +```rust +let mut i = 10 +loop { + println(i); + i -= 1; + + if i == 0 { + break; + } +} +``` + diff --git a/noir/noir-repo/docs/docs/noir/concepts/data_types/fields.md b/noir/noir-repo/docs/docs/noir/concepts/data_types/fields.md index b5af20f7d7e..c339e3d05e4 100644 --- a/noir/noir-repo/docs/docs/noir/concepts/data_types/fields.md +++ b/noir/noir-repo/docs/docs/noir/concepts/data_types/fields.md @@ -132,7 +132,7 @@ example: ```rust fn main() { let field = 2 - field.assert_max_bit_size(32); + field.assert_max_bit_size::<32>(); } ``` diff --git a/noir/noir-repo/docs/docs/noir/concepts/data_types/strings.md b/noir/noir-repo/docs/docs/noir/concepts/data_types/strings.md index 1fdee42425e..b2257e8bdbb 100644 --- a/noir/noir-repo/docs/docs/noir/concepts/data_types/strings.md +++ b/noir/noir-repo/docs/docs/noir/concepts/data_types/strings.md @@ -77,3 +77,38 @@ let s = r#"Simon says "hello world""#; // Any number of hashes may be used (>= 1) as long as the string also terminates with the same number of hashes let s = r#####"One "#, Two "##, Three "###, Four "####, Five will end the string."#####; ``` + +## Format strings + +A format string begins with the letter `f` and allows inserting the value of local and global variables in it. + +Example: + +```rust +let four = 2 + 2; +let s = f"Two plus two is: {four}"; +println(s); +``` + +The output of the above program is: + +```text +Two plus two is: 4 +``` + +To insert the value of a local or global variable, put it inside `{...}` in the string. + +If you need to write the `{` or `}` characters, use `{{` and `}}` respectively: + +```rust +let four = 2 + 2; + +// Prints "This is not expanded: {four}" +println(f"This is not expanded: {{four}}"); +``` + +More complex expressions are not allowed inside `{...}`: + +```rust +let s = f"Two plus two is: {2 + 2}" // Error: invalid format string +``` \ No newline at end of file diff --git a/noir/noir-repo/docs/docs/noir/concepts/traits.md b/noir/noir-repo/docs/docs/noir/concepts/traits.md index b6c0a886eb0..13818ecffac 100644 --- a/noir/noir-repo/docs/docs/noir/concepts/traits.md +++ b/noir/noir-repo/docs/docs/noir/concepts/traits.md @@ -252,36 +252,33 @@ impl MyTrait for Field { Since associated constants can also be used in a type position, its values are limited to only other expression kinds allowed in numeric generics. -Note that currently all associated types and constants must be explicitly specified in a trait constraint. -If we leave out any, we'll get an error that we're missing one: +When writing a trait constraint, you can specify all associated types and constants explicitly if +you wish: ```rust -// Error! Constraint is missing associated constant for `Bar` -fn foo(x: T) where T: MyTrait { +fn foo(x: T) where T: MyTrait { ... } ``` -Because all associated types and constants must be explicitly specified, they are essentially named generics, -although this is set to change in the future. Future versions of Noir will allow users to elide associated types -in trait constraints similar to Rust. When this is done, you may still refer to their value with the `::AssociatedType` -syntax: +Or you can also elide them since there should only be one `Foo` and `Bar` for a given implementation +of `MyTrait` for a type: ```rust -// Only valid in future versions of Noir: fn foo(x: T) where T: MyTrait { - let _: ::Foo = ...; + ... } ``` -The type as trait syntax is possible in Noir today but is less useful when each type must be explicitly specified anyway: +If you elide associated types, you can still refer to them via the type as trait syntax ``: ```rust -fn foo(x: T) where T: MyTrait { - // Works, but could just use F directly - let _: >::Foo = ...; - - let _: F = ...; +fn foo(x: T) where + T: MyTrait, + ::Foo: Default + Eq +{ + let foo_value: ::Foo = Default::default(); + assert_eq(foo_value, foo_value); } ``` @@ -582,3 +579,5 @@ pub trait Trait {} // This trait alias is now public pub trait Baz = Foo + Bar; ``` + +Trait methods have the same visibility as the trait they are in. diff --git a/noir/noir-repo/docs/docs/noir/concepts/unconstrained.md b/noir/noir-repo/docs/docs/noir/concepts/unconstrained.md index b5221b8d2dd..26a5df0be3e 100644 --- a/noir/noir-repo/docs/docs/noir/concepts/unconstrained.md +++ b/noir/noir-repo/docs/docs/noir/concepts/unconstrained.md @@ -66,9 +66,8 @@ We can then run `u72_to_u8` as unconstrained brillig code in order to calculate ```rust fn main(num: u72) -> pub [u8; 8] { - let out = unsafe { - u72_to_u8(num) - }; + /// Safety: 'out' is properly constrained below in 'assert(num == reconstructed_num);' + let out = unsafe { u72_to_u8(num) }; let mut reconstructed_num: u72 = 0; for i in 0..8 { @@ -96,6 +95,7 @@ This ends up taking off another ~250 gates from our circuit! We've ended up with Note that in order to invoke unconstrained functions we need to wrap them in an `unsafe` block, to make it clear that the call is unconstrained. +Furthermore, a warning is emitted unless the `unsafe` block is documented with a `/// Safety: ...` doc comment explaining why it is fine to call the unconstrained function. Note that either the `unsafe` block can be documented this way or the statement it exists in (like in the `let` example above). Generally we want to use brillig whenever there's something that's easy to verify but hard to compute within the circuit. For example, if you wanted to calculate a square root of a number it'll be a much better idea to calculate this in brillig and then assert that if you square the result you get back your number. diff --git a/noir/noir-repo/docs/docs/noir/standard_library/meta/struct_def.md b/noir/noir-repo/docs/docs/noir/standard_library/meta/struct_def.md index 535708e0353..18e722af31e 100644 --- a/noir/noir-repo/docs/docs/noir/standard_library/meta/struct_def.md +++ b/noir/noir-repo/docs/docs/noir/standard_library/meta/struct_def.md @@ -42,23 +42,28 @@ any generics, the generics are also included as-is. #include_code generics noir_stdlib/src/meta/struct_def.nr rust -Returns each generic on this struct. +Returns each generic on this struct. Each generic is represented as a tuple containing the type, +and an optional containing the numeric type if it's a numeric generic. Example: ``` #[example] -struct Foo { - bar: [T; 2], +struct Foo { + bar: [T; K], baz: Baz, } comptime fn example(foo: StructDefinition) { - assert_eq(foo.generics().len(), 2); + assert_eq(foo.generics().len(), 3); // Fails because `T` isn't in scope // let t = quote { T }.as_type(); - // assert_eq(foo.generics()[0], t); + // assert_eq(foo.generics()[0].0, t); + assert(foo.generics()[0].1.is_none()); + + // Last generic is numeric, so we have the numeric type available to us + assert(foo.generics()[2].1.is_some()); } ``` @@ -66,7 +71,18 @@ comptime fn example(foo: StructDefinition) { #include_code fields noir_stdlib/src/meta/struct_def.nr rust -Returns each field of this struct as a pair of (field name, field type). +Returns (name, type) pairs of each field in this struct. +Any generic types used in each field type is automatically substituted with the +provided generic arguments. + +### fields_as_written + +#include_code fields_as_written noir_stdlib/src/meta/struct_def.nr rust + +Returns (name, type) pairs of each field in this struct. Each type is as-is +with any generic arguments unchanged. Unless the field types are not needed, +users should generally prefer to use `StructDefinition::fields` over this +function if possible. ### has_named_attribute diff --git a/noir/noir-repo/docs/docs/noir/standard_library/meta/trait_impl.md b/noir/noir-repo/docs/docs/noir/standard_library/meta/trait_impl.md index 659c6aad719..9134bf548b0 100644 --- a/noir/noir-repo/docs/docs/noir/standard_library/meta/trait_impl.md +++ b/noir/noir-repo/docs/docs/noir/standard_library/meta/trait_impl.md @@ -25,8 +25,10 @@ comptime { let generics = my_impl.trait_generic_args(); assert_eq(generics.len(), 2); - assert_eq(generics[0], quote { i32 }.as_type()); - assert_eq(generics[1], quote { Field }.as_type()); + assert_eq(generics[0].0, quote { i32 }.as_type()); + assert(generics[0].1.is_none()); + assert_eq(generics[1].0, quote { Field }.as_type()); + assert(generics[1].1.is_none()); } ``` diff --git a/noir/noir-repo/docs/docs/tutorials/noirjs_app.md b/noir/noir-repo/docs/docs/tutorials/noirjs_app.md index 6e69ea0bbed..cc61fff8405 100644 --- a/noir/noir-repo/docs/docs/tutorials/noirjs_app.md +++ b/noir/noir-repo/docs/docs/tutorials/noirjs_app.md @@ -1,186 +1,91 @@ --- -title: Building a web app with NoirJS +title: Building a web app with Noir and Barretenberg description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] sidebar_position: 0 pagination_next: noir/concepts/data_types/index --- -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! +NoirJS is a Typescript package meant to work both in a browser and a server environment. -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/quick_start.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` +In this tutorial, we will combine NoirJS with Aztec's Barretenberg backend to build a simple web app. From here, you should get an idea on how to proceed with your own Noir projects! -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! +You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). -### Nargo +## Dependencies -Run: +Before we start, we want to make sure we have Node installed. For convenience (and speed), we can just install [Bun](https://bun.sh) as our package manager, and Node will work out-of-the-box: ```bash -nargo new circuit +curl -fsSL https://bun.sh/install | bash ``` -And... That's about it. Your program is ready to be compiled and run. +Let's go barebones. Doing the bare minimum is not only simple, but also allows you to easily adapt it to almost any frontend framework. -To compile, let's `cd` into the `circuit` folder to enter our project, and call: +Barebones means we can immediately start with the dependencies even on an empty folder 😈: ```bash -nargo compile +bun i @noir-lang/noir_wasm@1.0.0-beta.1 @noir-lang/noir_js@1.0.0-beta.1 @aztec/bb.js@0.63.1 ``` -This compiles our circuit into `json` format and add it to a new `target` folder. +Wait, what are these dependencies? -:::info +- `noir_wasm` is the `wasm` version of the Noir compiler. Although most developers prefer to use `nargo` for compiling, there's nothing wrong with `noir_wasm`. We like `noir_wasm`. +- `noir_js` is the main Noir package. It will execute our program, and generate the witness that will be sent to the backend. +- `bb.js` is the Typescript interface for Aztec's Barretenberg proving backend. It also uses the `wasm` version in order to run on the browser. -At this point in the tutorial, your folder structure should look like this: +:::info -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` +In this guide, we will install versions pinned to 1.0.0-beta.1. These work with Barretenberg version 0.63.1, so we are using that one version too. Feel free to try with older or later versions, though! ::: -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/quick_start.md) guide. However, we want our app to run on the browser, so we need Vite. +## Setting up our Noir program -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. +ZK is a powerful technology. An app that reveals computational correctness but doesn't reveal some of its inputs is almost unbelievable, yet Noir makes it as easy as a single line of code. -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". +:::tip -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: +It's not just you. We also enjoy syntax highlighting. [Check out the Language Server](../tooling/language_server.md) -```bash -cd vite-project -``` +::: -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; +All you need is a `main.nr` and a `Nargo.toml` file. You can follow the [noirup](../getting_started/noir_installation.md) installation and just run `noirup -v 1.0.0-beta.1`, or just create them by hand: -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); +```bash +mkdir -p circuit/src +touch circuit/src/main.nr circuit/Nargo.toml ``` -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: +To make our program interesting, let's give it a real use-case scenario: Bob wants to prove he is older than 18, without disclosing his age. Open `main.nr` and write: -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev +```rust +fn main(age: u8) { + assert(age >= 18); +} ``` -:::info - -At this point in the tutorial, your folder structure should look like this: +This program accepts a private input called age, and simply proves this number is higher than 18. But to run this code, we need to give the compiler a `Nargo.toml` with at least a name and a type: -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... +```toml +[package] +name = "circuit" +type = "bin" ``` -::: +This is all that we need to get started with Noir. -#### Some cleanup +![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. +## Setting up our app -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) +Remember when apps only had one `html` and one `js` file? Well, that's enough for Noir webapps. Let's create them: -## HTML +```bash +touch index.html index.js +``` -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: +And add something useful to our HTML file: ```html @@ -200,11 +105,11 @@ Our app won't run like this, of course. We need some working HTML, at least. Let - +

Noir app

- - + +

Logs

@@ -214,32 +119,26 @@ Our app won't run like this, of course. We need some working HTML, at least. Let ``` -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). +It _could_ be a beautiful UI... Depending on which universe you live in. In any case, we're using some scary CSS to make two boxes that will show cool things on the screen. -Start by pasting in this boilerplate code: +As for the JS, real madmen could just `console.log` everything, but let's say we want to see things happening (the true initial purpose of JS... right?). Here's some boilerplate for that. Just paste it in `index.js`: ```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} +const show = (id, content) => { + const container = document.getElementById(id); + container.appendChild(document.createTextNode(content)); + container.appendChild(document.createElement("br")); +}; -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } +document.getElementById("submit").addEventListener("click", async () => { + try { + // noir goes here + } catch { + show("logs", "Oh 💔"); + } }); -``` -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 +``` :::info @@ -248,30 +147,56 @@ At this point in the tutorial, your folder structure should look like this: ```tree . └── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html + └── src + └── main.nr + Nargo.toml + index.js + package.json + index.html + ...etc ``` -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - ::: -## Some NoirJS +## Compile compile compile -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: +Finally we're up for something cool. But before we can execute a Noir program, we need to compile it into ACIR: an abstract representation. Here's where `noir_wasm` comes in. -```ts -import circuit from '../circuit/target/circuit.json'; +`noir_wasm` expects a filesystem so it can resolve dependencies. While we could use the `public` folder, let's just import those using the nice `?url` syntax provided by vite. At the top of the file: + +```js +import { compile, createFileManager } from "@noir-lang/noir_wasm" + +import main from "./circuit/src/main.nr?url"; +import nargoToml from "./circuit/Nargo.toml?url"; ``` -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: +Compiling on the browser is common enough that `createFileManager` already gives us a nice in-memory filesystem we can use. So all we need to compile is fetching these files, writing them to our filesystem, and compile. Add this function: ```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; +export async function getCircuit() { + const fm = createFileManager("/"); + const { body } = await fetch(main); + const { body: nargoTomlBody } = await fetch(nargoToml); + + fm.writeFile("./src/main.nr", body); + fm.writeFile("./Nargo.toml", nargoTomlBody); + return await compile(fm); +} +``` + +:::tip + +As you can imagine, with `node` it's all conveniently easier since you get native access to `fs`... + +::: + +## Some more JS + +We're starting with the good stuff now. We want to execute our circuit to get the witness, and then feed that witness to Barretenberg. Luckily, both packages are quite easy to work with. Let's import them at the top of the file: + +```js +import { UltraHonkBackend } from '@aztec/bb.js'; import { Noir } from '@noir-lang/noir_js'; ``` @@ -279,88 +204,103 @@ And instantiate them inside our try-catch block: ```ts // try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); +const { program } = await getCircuit(); +const noir = new Noir(program); +const backend = new UltraHonkBackend(program.bytecode); // } ``` -:::note +:::warning -For the remainder of the tutorial, everything will be happening inside the `try` block +WASMs are not always easy to work with. In our case, `vite` likes serving them with the wrong MIME type. There are different fixes but we found the easiest one is just YOLO instantiating the WASMs manually. Paste this at the top of the file, just below the other imports, and it will work just fine: + +```js +import initNoirC from "@noir-lang/noirc_abi"; +import initACVM from "@noir-lang/acvm_js"; +import acvm from "@noir-lang/acvm_js/web/acvm_js_bg.wasm?url"; +import noirc from "@noir-lang/noirc_abi/web/noirc_abi_wasm_bg.wasm?url"; +await Promise.all([initACVM(fetch(acvm)), initNoirC(fetch(noirc))]); +``` ::: -## Our app +## Executing and proving -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: +Now for the app itself. We're capturing whatever is in the input when people press the submit button. Inside our `try` block, let's just grab that input and get its value. Noir will gladly execute it, and give us a witness: ```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; +const age = document.getElementById("age").value; +show("logs", "Generating witness... ⏳"); +const { witness } = await noir.execute({ age }); +show("logs", "Generated witness... ✅"); + ``` +:::note + +For the remainder of the tutorial, everything will be happening inside the `try` block + +::: + Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: ```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); +show("logs", "Generating proof... ⏳"); const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); +show("logs", "Generated proof... ✅"); +show("results", proof.proof); ``` -You're probably eager to see stuff happening, so go and run your app now! +Our program is technically **done** . You're probably eager to see stuff happening! To serve this in a convenient way, we can use a bundler like `vite` by creating a `vite.config.js` file: + +```bash +touch vite.config.js +``` -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. +`vite` helps us with a little catch: `bb.js` in particular uses top-level awaits which aren't supported everywhere. So we can add this to the `vite.config.js` to make the bundler optimize them: -![Getting Started 0](@site/static/img/noir_getting_started_1.png) +```js +export default { optimizeDeps: { esbuildOptions: { target: "esnext" } } }; +``` + +This should be enough for vite. We don't even need to install it, just run: + +```bash +bunx vite +``` -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! +If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. +![Noir Webapp UI](@site/static/img/tutorials/noirjs_webapp/webapp1.png) + +Now, our circuit requires a private input `fn main(age: u8)`, and fails if it is less than 18. Let's see if it works. Submit any number above 18 (as long as it fits in 8 bits) and you should get a valid proof. Otherwise the proof won't even generate correctly. + +By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. ## Verifying Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: ```js -display('logs', 'Verifying proof... ⌛'); +show('logs', 'Verifying proof... ⌛'); const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); +show("logs", `Proof is ${isValid ? "valid" : "invalid"}... ✅`); ``` You have successfully generated a client-side Noir web app! ![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. +## Next steps -## UltraHonk Backend +At this point, you have a working ZK app that works on the browser. Actually, it works on a mobile phone too! -Barretenberg has recently exposed a new UltraHonk backend. We can use UltraHonk in NoirJS after version 0.33.0. Everything will be the same as the tutorial above, except that the class we need to import will change: +If you want to continue learning by doing, here are some challenges for you: -```js -import { UltraHonkBackend, UltraHonkVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -``` - -The backend will then be instantiated as such: - -```js -const backend = new UltraHonkBackend(circuit); -``` +- Install [nargo](https://noir-lang.org/docs/getting_started/noir_installation) and write [Noir tests](../tooling/testing) +- Change the circuit to accept a [public input](../noir/concepts/data_types/#private--public-types) as the cutoff age. It could be different depending on the purpose, for example! +- Enjoy Noir's Rust-like syntax and write a struct `Country` that implements a trait `MinAge` with a method `get_min_age`. Then, make a struct `Person` have an `u8` as its age and a country of type `Country`. You can pass a `person` in JS just like a JSON object `person: { age, country: { min_age: 18 }}` -Then all the commands to prove and verify your circuit will be same. +The world is your stage, just have fun with ZK! You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. -The only feature currently unsupported with UltraHonk are [recursive proofs](../explainers/explainer-recursion.md). +Check out other starters, tools, or just cool projects in the [awesome noir repository](https://github.com/noir-lang/awesome-noir). diff --git a/noir/noir-repo/docs/package.json b/noir/noir-repo/docs/package.json index 39807588eaa..fcfdd3c9c14 100644 --- a/noir/noir-repo/docs/package.json +++ b/noir/noir-repo/docs/package.json @@ -13,6 +13,7 @@ "version": "yarn version::stables && ./scripts/cut_version.sh" }, "dependencies": { + "@cookbookdev/docsbot": "^4.24.9", "@docusaurus/core": "^3.5.2", "@docusaurus/preset-classic": "^3.5.2", "@easyops-cn/docusaurus-search-local": "^0.35.0", diff --git a/noir/noir-repo/docs/src/theme/SearchBar/index.js b/noir/noir-repo/docs/src/theme/SearchBar/index.js new file mode 100644 index 00000000000..41dd822a352 --- /dev/null +++ b/noir/noir-repo/docs/src/theme/SearchBar/index.js @@ -0,0 +1,14 @@ +import React from 'react'; +import SearchBar from '@theme-original/SearchBar'; +import AskCookbook from '@cookbookdev/docsbot/react' +import BrowserOnly from '@docusaurus/BrowserOnly'; +/** It's a public API key, so it's safe to expose it here */ +const COOKBOOK_PUBLIC_API_KEY = "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJzdWIiOiI2NzVhMGVkODQ4MGI5OThmNzlhNjNhOTciLCJpYXQiOjE3MzM5NTUyODgsImV4cCI6MjA0OTUzMTI4OH0.NzQlH2sfhJkjvnYxoaRgSY6nRyNClxHg57n3-JueN9Q"; +export default function SearchBarWrapper(props) { + return ( + <> + + {() => } + + ); +} diff --git a/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_6.png b/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_6.png new file mode 100644 index 00000000000..c92e874740a Binary files /dev/null and b/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_6.png differ diff --git a/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_7.png b/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_7.png new file mode 100644 index 00000000000..c8541fa544c Binary files /dev/null and b/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_7.png differ diff --git a/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_8.png b/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_8.png new file mode 100644 index 00000000000..125c140acec Binary files /dev/null and b/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_8.png differ diff --git a/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_9.png b/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_9.png new file mode 100644 index 00000000000..150812e3320 Binary files /dev/null and b/noir/noir-repo/docs/static/img/how-tos/solidity_verifier_9.png differ diff --git a/noir/noir-repo/docs/static/img/tutorials/noirjs_webapp/webapp1.png b/noir/noir-repo/docs/static/img/tutorials/noirjs_webapp/webapp1.png new file mode 100644 index 00000000000..7591cd827e3 Binary files /dev/null and b/noir/noir-repo/docs/static/img/tutorials/noirjs_webapp/webapp1.png differ diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/backend/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/backend/_category_.json deleted file mode 100644 index b82e92beb0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/backend/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "position": 1, - "label": "Install Proving Backend", - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/backend/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/backend/index.md deleted file mode 100644 index 7192d954877..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/backend/index.md +++ /dev/null @@ -1,31 +0,0 @@ ---- -title: Proving Backend Installation -description: Proving backends offer command line tools for proving and verifying Noir programs. This page describes how to install `bb` as an example. -keywords: [ - Proving - Backend - Barretenberg - bb - bbup - Installation - Terminal - Command - CLI - Version -] -pagination_next: getting_started/hello_noir/index ---- - -Proving backends each provide their own tools for working with Noir programs, providing functionality like proof generation, proof verification, and verifier smart contract generation. - -For the latest information on tooling provided by each proving backend, installation instructions, Noir version compatibility... you may refer to the proving backends' own documentation. - -You can find the full list of proving backends compatible with Noir in [Awesome Noir](https://github.com/noir-lang/awesome-noir/?tab=readme-ov-file#proving-backends). - -## Example: Installing `bb` - -`bb` is the CLI tool provided by the [Barretenberg proving backend](https://github.com/AztecProtocol/barretenberg) developed by Aztec Labs. - -You can find the instructions for installation in [`bb`'s documentation](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/cpp/src/barretenberg/bb/readme.md#installation). - -Once installed, we are ready to start working on [our first Noir program](../hello_noir/index.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/_category_.json deleted file mode 100644 index 976a2325de0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/index.md deleted file mode 100644 index 3baae217eb3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/index.md +++ /dev/null @@ -1,152 +0,0 @@ ---- -title: Creating a Project -description: - Learn how to create and verify your first Noir program using Nargo, a programming language for - zero-knowledge proofs. -keywords: - [ - Nargo, - Noir, - zero-knowledge proofs, - programming language, - create Noir program, - verify Noir program, - step-by-step guide, - ] -sidebar_position: 1 - ---- - -Now that we have installed Nargo and a proving backend, it is time to make our first hello world program! - -### 1. Create a new project directory - -Noir code can live anywhere on your computer. Let us create a _projects_ folder in the home -directory to house our first Noir program. - -Create the directory and change directory into it by running: - -```sh -mkdir ~/projects -cd ~/projects -``` - -## Nargo - -Nargo provides the ability to initiate and execute Noir projects. Read the [Nargo installation](../installation/index.md) section to learn more about Nargo and how to install it. - -### 2. Create a new Noir project - -Now that we are in the projects directory, create a new Nargo project by running: - -```sh -nargo new hello_world -``` - -`hello_world` can be any arbitrary project name, we are simply using `hello_world` for demonstration. - -In production, it is common practice to name the project folder, `circuits`, for clarity amongst other folders in the codebase (like: `contracts`, `scripts`, and `test`). - -A `hello_world` folder would be created. Similar to Rust, the folder houses _src/main.nr_ and -_Nargo.toml_ which contain the source code and environmental options of your Noir program -respectively. - -#### Intro to Noir Syntax - -Let us take a closer look at _main.nr_. The default _main.nr_ generated should look like this: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -The first line of the program specifies the program's inputs: - -```rust -x : Field, y : pub Field -``` - -Program inputs in Noir are private by default (e.g. `x`), but can be labeled public using the -keyword `pub` (e.g. `y`). To learn more about private and public values, check the -[Data Types](../../noir/concepts/data_types/index.md) section. - -The next line of the program specifies its body: - -```rust -assert(x != y); -``` - -The Noir syntax `assert` can be interpreted as something similar to constraints in other zk-contract languages. - -For more Noir syntax, check the [Language Concepts](../../noir/concepts/comments.md) chapter. - -### 3. Build in/output files - -Change directory into _hello_world_ and build in/output files for your Noir program by running: - -```sh -cd hello_world -nargo check -``` - -A _Prover.toml_ file will be generated in your project directory, to allow specifying input values to the program. - -### 4. Execute the Noir program - -Now that the project is set up, we can execute our Noir program. - -Fill in input values for execution in the _Prover.toml_ file. For example: - -```toml -x = "1" -y = "2" -``` - -Execute your Noir program: - -```sh -nargo execute witness-name -``` - -The witness corresponding to this execution will then be written to the file `./target/witness-name.gz`. - -The command also automatically compiles your Noir program if it was not already / was edited, which you may notice the compiled artifacts being written to the file `./target/hello_world.json`. - -## Proving Backend - -Proving backends provide the ability to generate and verify proofs of executing Noir programs, following Noir's tooling that compiles and executes the programs. Read the [proving backend installation](../backend/index.md) section to learn more about proving backends and how to install them. - -Barretenberg is used as an example here to demonstrate how proving and verifying could be implemented and used. Read the [`bb` installation](../backend/index.md#example-installing-bb) section for how to install Barretenberg's CLI tool; refer to [`bb`'s documentation](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/cpp/src/barretenberg/bb/readme.md) for full details about the tool. - -### 5. Prove an execution of the Noir program - -Using Barretenberg as an example, prove the valid execution of your Noir program running: - -```sh -bb prove -b ./target/hello_world.json -w ./target/witness-name.gz -o ./target/proof -``` - -The proof generated will then be written to the file `./target/proof`. - -### 6. Verify the execution proof - -Once a proof is generated, we can verify correct execution of our Noir program by verifying the proof file. - -Using Barretenberg as an example, compute the verification key for the Noir program by running: - -```sh -bb write_vk -b ./target/hello_world.json -o ./target/vk -``` - -And verify your proof by running: - -```sh -bb verify -k ./target/vk -p ./target/proof -``` - -If successful, the verification will complete in silence; if unsuccessful, the command will trigger logging of the corresponding error. - -Congratulations, you have now created and verified a proof for your very first Noir program! - -In the [next section](./project_breakdown.md), we will go into more detail on each step performed. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/project_breakdown.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/project_breakdown.md deleted file mode 100644 index 96e653f6c08..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/hello_noir/project_breakdown.md +++ /dev/null @@ -1,159 +0,0 @@ ---- -title: Project Breakdown -description: - Learn about the anatomy of a Nargo project, including the purpose of the Prover TOML - file, and how to prove and verify your program. -keywords: - [Nargo, Nargo project, Prover.toml, proof verification, private asset transfer] -sidebar_position: 2 ---- - -This section breaks down our hello world program from the previous section. - -## Anatomy of a Nargo Project - -Upon creating a new project with `nargo new` and building the in/output files with `nargo check` -commands, you would get a minimal Nargo project of the following structure: - - - src - - Prover.toml - - Nargo.toml - -The source directory _src_ holds the source code for your Noir program. By default only a _main.nr_ -file will be generated within it. - -### Prover.toml - -_Prover.toml_ is used for specifying the input values for executing and proving the program. You can specify `toml` files with different names by using the `--prover-name` or `-p` flags, see the [Prover](#provertoml) section below. Optionally you may specify expected output values for prove-time checking as well. - -### Nargo.toml - -_Nargo.toml_ contains the environmental options of your project. It contains a "package" section and a "dependencies" section. - -Example Nargo.toml: - -```toml -[package] -name = "noir_starter" -type = "bin" -authors = ["Alice"] -compiler_version = "0.9.0" -description = "Getting started with Noir" -entry = "circuit/main.nr" -license = "MIT" - -[dependencies] -ecrecover = {tag = "v0.9.0", git = "https://github.com/colinnielsen/ecrecover-noir.git"} -``` - -Nargo.toml for a [workspace](../../noir/modules_packages_crates/workspaces.md) will look a bit different. For example: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -#### Package section - -The package section defines a number of fields including: - -- `name` (**required**) - the name of the package -- `type` (**required**) - can be "bin", "lib", or "contract" to specify whether its a binary, library or Aztec contract -- `authors` (optional) - authors of the project -- `compiler_version` - specifies the version of the compiler to use. This is enforced by the compiler and follow's [Rust's versioning](https://doc.rust-lang.org/cargo/reference/manifest.html#the-version-field), so a `compiler_version = 0.18.0` will enforce Nargo version 0.18.0, `compiler_version = ^0.18.0` will enforce anything above 0.18.0 but below 0.19.0, etc. For more information, see how [Rust handles these operators](https://docs.rs/semver/latest/semver/enum.Op.html) -- `description` (optional) -- `entry` (optional) - a relative filepath to use as the entry point into your package (overrides the default of `src/lib.nr` or `src/main.nr`) -- `backend` (optional) -- `license` (optional) -- `expression_width` (optional) - Sets the default backend expression width. This field will override the default backend expression width specified by the Noir compiler (currently set to width 4). - -#### Dependencies section - -This is where you will specify any dependencies for your project. See the [Dependencies page](../../noir/modules_packages_crates/dependencies.md) for more info. - -`./proofs/` and `./contract/` directories will not be immediately visible until you create a proof or -verifier contract respectively. - -### main.nr - -The _main.nr_ file contains a `main` method, this method is the entry point into your Noir program. - -In our sample program, _main.nr_ looks like this: - -```rust -fn main(x : Field, y : Field) { - assert(x != y); -} -``` - -The parameters `x` and `y` can be seen as the API for the program and must be supplied by the prover. Since neither `x` nor `y` is marked as public, the verifier does not supply any inputs, when verifying the proof. - -The prover supplies the values for `x` and `y` in the _Prover.toml_ file. - -As for the program body, `assert` ensures that the condition to be satisfied (e.g. `x != y`) is constrained by the proof of the execution of said program (i.e. if the condition was not met, the verifier would reject the proof as an invalid proof). - -### Prover.toml - -The _Prover.toml_ file is a file which the prover uses to supply the inputs to the Noir program (both private and public). - -In our hello world program the _Prover.toml_ file looks like this: - -```toml -x = "1" -y = "2" -``` - -When the command `nargo execute` is executed, nargo will execute the Noir program using the inputs specified in `Prover.toml`, aborting if it finds that these do not satisfy the constraints defined by `main`. In this example, `x` and `y` must satisfy the inequality constraint `assert(x != y)`. - -If an output name is specified such as `nargo execute foo`, the witness generated by this execution will be written to `./target/foo.gz`. This can then be used to generate a proof of the execution. - -#### Arrays of Structs - -The following code shows how to pass an array of structs to a Noir program to generate a proof. - -```rust -// main.nr -struct Foo { - bar: Field, - baz: Field, -} - -fn main(foos: [Foo; 3]) -> pub Field { - foos[2].bar + foos[2].baz -} -``` - -Prover.toml: - -```toml -[[foos]] # foos[0] -bar = 0 -baz = 0 - -[[foos]] # foos[1] -bar = 0 -baz = 0 - -[[foos]] # foos[2] -bar = 1 -baz = 2 -``` - -#### Custom toml files - -You can specify a `toml` file with a different name to use for execution by using the `--prover-name` or `-p` flags. - -This command looks for proof inputs in the default **Prover.toml** and generates the witness and saves it at `./target/foo.gz`: - -```bash -nargo execute foo -``` - -This command looks for proof inputs in the custom **OtherProver.toml** and generates the witness and saves it at `./target/bar.gz`: - -```bash -nargo execute -p OtherProver bar -``` - -Now that you understand the concepts, you'll probably want some editor feedback while you are writing more complex code. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/_category_.json deleted file mode 100644 index 0c02fb5d4d7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "position": 0, - "label": "Install Nargo", - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/index.md deleted file mode 100644 index 53ea9c7891c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/index.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Nargo Installation -description: - nargo is a command line tool for interacting with Noir programs. This page is a quick guide on how to install Nargo through the most common and easy method, noirup -keywords: [ - Nargo - Noir - Rust - Cargo - Noirup - Installation - Terminal Commands - Version Check - Nightlies - Specific Versions - Branches - Noirup Repository -] -pagination_next: getting_started/hello_noir/index ---- - -`nargo` is a tool for working with Noir programs on the CLI, providing you with the ability to start new projects, compile, execute and test Noir programs from the terminal. - -The name is inspired by Rust's package manager `cargo`; and similar to Rust's `rustup`, Noir also has an easy installation script `noirup`. - -## Installing Noirup - -Open a terminal on your machine, and write: - -```bash -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Close the terminal, open another one, and run - -```bash -noirup -``` - -Done. That's it. You should have the latest version working. You can check with `nargo --version`. - -You can also install nightlies, specific versions -or branches. Check out the [noirup repository](https://github.com/noir-lang/noirup) for more -information. - -Now we're ready to start working on [our first Noir program!](../hello_noir/index.md) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/other_install_methods.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/other_install_methods.md deleted file mode 100644 index 3634723562b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/installation/other_install_methods.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Alternative Installations -description: There are different ways to install Nargo, the one-stop shop and command-line tool for developing Noir programs. This guide explains how to specify which version to install when using noirup, and using WSL for windows. -keywords: [ - Installation - Nargo - Noirup - Binaries - Compiling from Source - WSL for Windows - macOS - Linux - Nix - Direnv - Uninstalling Nargo - ] -sidebar_position: 1 ---- - -## Encouraged Installation Method: Noirup - -Noirup is the endorsed method for installing Nargo, streamlining the process of fetching binaries or compiling from source. It supports a range of options to cater to your specific needs, from nightly builds and specific versions to compiling from various sources. - -### Installing Noirup - -First, ensure you have `noirup` installed: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -### Fetching Binaries - -With `noirup`, you can easily switch between different Nargo versions, including nightly builds: - -- **Nightly Version**: Install the latest nightly build. - - ```sh - noirup --version nightly - ``` - -- **Specific Version**: Install a specific version of Nargo. - ```sh - noirup --version - ``` - -### Compiling from Source - -`noirup` also enables compiling Nargo from various sources: - -- **From a Specific Branch**: Install from the latest commit on a branch. - - ```sh - noirup --branch - ``` - -- **From a Fork**: Install from the main branch of a fork. - - ```sh - noirup --repo - ``` - -- **From a Specific Branch in a Fork**: Install from a specific branch in a fork. - - ```sh - noirup --repo --branch - ``` - -- **From a Specific Pull Request**: Install from a specific PR. - - ```sh - noirup --pr - ``` - -- **From a Specific Commit**: Install from a specific commit. - - ```sh - noirup -C - ``` - -- **From Local Source**: Compile and install from a local directory. - ```sh - noirup --path ./path/to/local/source - ``` - -## Installation on Windows - -The default backend for Noir (Barretenberg) doesn't provide Windows binaries at this time. For that reason, Noir cannot be installed natively. However, it is available by using Windows Subsystem for Linux (WSL). - -Step 1: Follow the instructions [here](https://learn.microsoft.com/en-us/windows/wsl/install) to install and run WSL. - -step 2: Follow the [Noirup instructions](#encouraged-installation-method-noirup). - -## Uninstalling Nargo - -If you installed Nargo with `noirup`, you can uninstall Nargo by removing the files in `~/.nargo`, `~/nargo`, and `~/noir_cache`. This ensures that all installed binaries, configurations, and cache related to Nargo are fully removed from your system. - -```bash -rm -r ~/.nargo -rm -r ~/nargo -rm -r ~/noir_cache -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/tooling/noir_codegen.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/tooling/noir_codegen.md deleted file mode 100644 index f7505bef7ab..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/tooling/noir_codegen.md +++ /dev/null @@ -1,114 +0,0 @@ ---- -title: Noir Codegen for TypeScript -description: Learn how to use Noir codegen to generate TypeScript bindings -keywords: [Nargo, Noir, compile, TypeScript] -sidebar_position: 3 ---- - -When using TypeScript, it is extra work to interpret Noir program outputs in a type-safe way. Third party libraries may exist for popular Noir programs, but they are either hard to find or unmaintained. - -Now you can generate TypeScript bindings for your Noir programs in two steps: -1. Exporting Noir functions using `nargo export` -2. Using the TypeScript module `noir_codegen` to generate TypeScript binding - -**Note:** you can only export functions from a Noir *library* (not binary or contract program types). - -## Installation - -### Your TypeScript project - -If you don't already have a TypeScript project you can add the module with `yarn` (or `npm`), then initialize it: - -```bash -yarn add typescript -D -npx tsc --init -``` - -### Add TypeScript module - `noir_codegen` - -The following command will add the module to your project's devDependencies: - -```bash -yarn add @noir-lang/noir_codegen -D -``` - -### Nargo library -Make sure you have Nargo, v0.25.0 or greater, installed. If you don't, follow the [installation guide](../installation/index.md). - -If you're in a new project, make a `circuits` folder and create a new Noir library: - -```bash -mkdir circuits && cd circuits -nargo new --lib myNoirLib -``` - -## Usage - -### Export ABI of specified functions - -First go to the `.nr` files in your Noir library, and add the `#[export]` macro to each function that you want to use in TypeScript. - -```rust -#[export] -fn your_function(... -``` - -From your Noir library (where `Nargo.toml` is), run the following command: - -```bash -nargo export -``` - -You will now have an `export` directory with a .json file per exported function. - -You can also specify the directory of Noir programs using `--program-dir`, for example: - -```bash -nargo export --program-dir=./circuits/myNoirLib -``` - -### Generate TypeScript bindings from exported functions - -To use the `noir-codegen` package we added to the TypeScript project: - -```bash -yarn noir-codegen ./export/your_function.json -``` - -This creates an `exports` directory with an `index.ts` file containing all exported functions. - -**Note:** adding `--out-dir` allows you to specify an output dir for your TypeScript bindings to go. Eg: - -```bash -yarn noir-codegen ./export/*.json --out-dir ./path/to/output/dir -``` - -## Example .nr function to .ts output - -Consider a Noir library with this function: - -```rust -#[export] -fn not_equal(x: Field, y: Field) -> bool { - x != y -} -``` - -After the export and codegen steps, you should have an `index.ts` like: - -```typescript -export type Field = string; - - -export const is_equal_circuit: CompiledCircuit = -{"abi":{"parameters":[{"name":"x","type":{"kind":"field"},"visibility":"private"},{"name":"y","type":{"kind":"field"},"visibility":"private"}],"return_type":{"abi_type":{"kind":"boolean"},"visibility":"private"}},"bytecode":"H4sIAAAAAAAA/7WUMQ7DIAxFQ0Krrr2JjSGYLVcpKrn/CaqqDQN12WK+hPBgmWd/wEyHbF1SS923uhOs3pfoChI+wKXMAXzIKyNj4PB0TFTYc0w5RUjoqeAeEu1wqK0F54RGkWvW44LPzExnlkbMEs4JNZmN8PxS42uHv82T8a3Jeyn2Ks+VLPcO558HmyLMCDOXAXXtpPt4R/Rt9T36ss6dS9HGPx/eG17nGegKBQAA"}; - -export async function is_equal(x: Field, y: Field, foreignCallHandler?: ForeignCallHandler): Promise { - const program = new Noir(is_equal_circuit); - const args: InputMap = { x, y }; - const { returnValue } = await program.execute(args, foreignCallHandler); - return returnValue as boolean; -} -``` - -Now the `is_equal()` function and relevant types are readily available for use in TypeScript. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-oracles.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-oracles.md deleted file mode 100644 index 94a89710e66..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-oracles.md +++ /dev/null @@ -1,269 +0,0 @@ ---- -title: How to use Oracles -description: Learn how to use oracles in your Noir program with examples in both Nargo and NoirJS. This guide also covers writing a JSON RPC server and providing custom foreign call handlers for NoirJS. -keywords: - - Noir Programming - - Oracles - - Nargo - - NoirJS - - JSON RPC Server - - Foreign Call Handlers -sidebar_position: 1 ---- - -This guide shows you how to use oracles in your Noir program. For the sake of clarity, it assumes that: - -- You have read the [explainer on Oracles](../explainers/explainer-oracle.md) and are comfortable with the concept. -- You have a Noir program to add oracles to. You can create one using the [vite-hardhat starter](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) as a boilerplate. -- You understand the concept of a JSON-RPC server. Visit the [JSON-RPC website](https://www.jsonrpc.org/) if you need a refresher. -- You are comfortable with server-side JavaScript (e.g. Node.js, managing packages, etc.). - -## Rundown - -This guide has 3 major steps: - -1. How to modify our Noir program to make use of oracle calls as unconstrained functions -2. How to write a JSON RPC Server to resolve these oracle calls with Nargo -3. How to use them in Nargo and how to provide a custom resolver in NoirJS - -## Step 1 - Modify your Noir program - -An oracle is defined in a Noir program by defining two methods: - -- An unconstrained method - This tells the compiler that it is executing an [unconstrained functions](../noir/concepts//unconstrained.md). -- A decorated oracle method - This tells the compiler that this method is an RPC call. - -An example of an oracle that returns a `Field` would be: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(number: Field) -> Field { } - -unconstrained fn get_sqrt(number: Field) -> Field { - sqrt(number) -} -``` - -In this example, we're wrapping our oracle function in an unconstrained method, and decorating it with `oracle(getSqrt)`. We can then call the unconstrained function as we would call any other function: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); -} -``` - -In the next section, we will make this `getSqrt` (defined on the `sqrt` decorator) be a method of the RPC server Noir will use. - -:::danger - -As explained in the [Oracle Explainer](../explainers/explainer-oracle.md), this `main` function is unsafe unless you constrain its return value. For example: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); - assert(sqrt.pow_32(2) as u64 == input as u64); // <---- constrain the return of an oracle! -} -``` - -::: - -:::info - -Currently, oracles only work with single params or array params. For example: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt([Field; 2]) -> [Field; 2] { } -``` - -::: - -## Step 2 - Write an RPC server - -Brillig will call *one* RPC server. Most likely you will have to write your own, and you can do it in whatever language you prefer. In this guide, we will do it in Javascript. - -Let's use the above example of an oracle that consumes an array with two `Field` and returns their square roots: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(input: [Field; 2]) -> [Field; 2] { } - -unconstrained fn get_sqrt(input: [Field; 2]) -> [Field; 2] { - sqrt(input) -} - -fn main(input: [Field; 2]) { - let sqrt = get_sqrt(input); - assert(sqrt[0].pow_32(2) as u64 == input[0] as u64); - assert(sqrt[1].pow_32(2) as u64 == input[1] as u64); -} -``` - -:::info - -Why square root? - -In general, computing square roots is computationally more expensive than multiplications, which takes a toll when speaking about ZK applications. In this case, instead of calculating the square root in Noir, we are using our oracle to offload that computation to be made in plain. In our circuit we can simply multiply the two values. - -::: - -Now, we should write the correspondent RPC server, starting with the [default JSON-RPC 2.0 boilerplate](https://www.npmjs.com/package/json-rpc-2.0#example): - -```js -import { JSONRPCServer } from "json-rpc-2.0"; -import express from "express"; -import bodyParser from "body-parser"; - -const app = express(); -app.use(bodyParser.json()); - -const server = new JSONRPCServer(); -app.post("/", (req, res) => { - const jsonRPCRequest = req.body; - server.receive(jsonRPCRequest).then((jsonRPCResponse) => { - if (jsonRPCResponse) { - res.json(jsonRPCResponse); - } else { - res.sendStatus(204); - } - }); -}); - -app.listen(5555); -``` - -Now, we will add our `getSqrt` method, as expected by the `#[oracle(getSqrt)]` decorator in our Noir code. It maps through the params array and returns their square roots: - -```js -server.addMethod("resolve_foreign_call", async (params) => { - if (params[0].function !== "getSqrt") { - throw Error("Unexpected foreign call") - }; - const values = params[0].inputs[0].map((field) => { - return `${Math.sqrt(parseInt(field, 16))}`; - }); - return { values: [values] }; -}); -``` - -If you're using Typescript, the following types may be helpful in understanding the expected return value and making sure they're easy to follow: - -```js -export type ForeignCallSingle = string; - -export type ForeignCallArray = string[]; - -export type ForeignCallResult = { - values: (ForeignCallSingle | ForeignCallArray)[]; -}; -``` - -:::info Multidimensional Arrays - -If the Oracle function is returning an array containing other arrays, such as `[['1','2],['3','4']]`, you need to provide the values in JSON as flattened values. In the previous example, it would be `['1', '2', '3', '4']`. In the Noir program, the Oracle signature can use a nested type, the flattened values will be automatically converted to the nested type. - -::: - -## Step 3 - Usage with Nargo - -Using the [`nargo` CLI tool](../getting_started/installation/index.md), you can use oracles in the `nargo test` and `nargo execute` commands by passing a value to `--oracle-resolver`. For example: - -```bash -nargo test --oracle-resolver http://localhost:5555 -``` - -This tells `nargo` to use your RPC Server URL whenever it finds an oracle decorator. - -## Step 4 - Usage with NoirJS - -In a JS environment, an RPC server is not strictly necessary, as you may want to resolve your oracles without needing any JSON call at all. NoirJS simply expects that you pass a callback function when you generate proofs, and that callback function can be anything. - -For example, if your Noir program expects the host machine to provide CPU pseudo-randomness, you could simply pass it as the `foreignCallHandler`. You don't strictly need to create an RPC server to serve pseudo-randomness, as you may as well get it directly in your app: - -```js -const foreignCallHandler = (name, inputs) => crypto.randomBytes(16) // etc - -await noir.execute(inputs, foreignCallHandler) -``` - -As one can see, in NoirJS, the [`foreignCallHandler`](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) function simply means "a callback function that returns a value of type [`ForeignCallOutput`](../reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md). It doesn't have to be an RPC call like in the case for Nargo. - -:::tip - -Does this mean you don't have to write an RPC server like in [Step #2](#step-2---write-an-rpc-server)? - -You don't technically have to, but then how would you run `nargo test`? To use both `Nargo` and `NoirJS` in your development flow, you will have to write a JSON RPC server. - -::: - -In this case, let's make `foreignCallHandler` call the JSON RPC Server we created in [Step #2](#step-2---write-an-rpc-server), by making it a JSON RPC Client. - -For example, using the same `getSqrt` program in [Step #1](#step-1---modify-your-noir-program) (comments in the code): - -```js -import { JSONRPCClient } from "json-rpc-2.0"; - -// declaring the JSONRPCClient -const client = new JSONRPCClient((jsonRPCRequest) => { -// hitting the same JSON RPC Server we coded above - return fetch("http://localhost:5555", { - method: "POST", - headers: { - "content-type": "application/json", - }, - body: JSON.stringify(jsonRPCRequest), - }).then((response) => { - if (response.status === 200) { - return response - .json() - .then((jsonRPCResponse) => client.receive(jsonRPCResponse)); - } else if (jsonRPCRequest.id !== undefined) { - return Promise.reject(new Error(response.statusText)); - } - }); -}); - -// declaring a function that takes the name of the foreign call (getSqrt) and the inputs -const foreignCallHandler = async (name, input) => { - const inputs = input[0].map((i) => i.toString("hex")) - // notice that the "inputs" parameter contains *all* the inputs - // in this case we to make the RPC request with the first parameter "numbers", which would be input[0] - const oracleReturn = await client.request("resolve_foreign_call", [ - { - function: name, - inputs: [inputs] - }, - ]); - return [oracleReturn.values[0]]; -}; - -// the rest of your NoirJS code -const input = { input: [4, 16] }; -const { witness } = await noir.execute(input, foreignCallHandler); -``` - -:::tip - -If you're in a NoirJS environment running your RPC server together with a frontend app, you'll probably hit a familiar problem in full-stack development: requests being blocked by [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) policy. For development only, you can simply install and use the [`cors` npm package](https://www.npmjs.com/package/cors) to get around the problem: - -```bash -yarn add cors -``` - -and use it as a middleware: - -```js -import cors from "cors"; - -const app = express(); -app.use(cors()) -``` - -::: - -## Conclusion - -Hopefully by the end of this guide, you should be able to: - -- Write your own logic around Oracles and how to write a JSON RPC server to make them work with your Nargo commands. -- Provide custom foreign call handlers for NoirJS. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-recursion.md deleted file mode 100644 index c8c4dc9f5b4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-recursion.md +++ /dev/null @@ -1,180 +0,0 @@ ---- -title: How to use recursion on NoirJS -description: Learn how to implement recursion with NoirJS, a powerful tool for creating smart contracts on the EVM blockchain. This guide assumes familiarity with NoirJS, solidity verifiers, and the Barretenberg proving backend. Discover how to generate both final and intermediate proofs using `noir_js` and `backend_barretenberg`. -keywords: - [ - "NoirJS", - "EVM blockchain", - "smart contracts", - "recursion", - "solidity verifiers", - "Barretenberg backend", - "noir_js", - "backend_barretenberg", - "intermediate proofs", - "final proofs", - "nargo compile", - "json import", - "recursive circuit", - "recursive app" - ] -sidebar_position: 1 ---- - -This guide shows you how to use recursive proofs in your NoirJS app. For the sake of clarity, it is assumed that: - -- You already have a NoirJS app. If you don't, please visit the [NoirJS tutorial](../tutorials/noirjs_app.md) and the [reference](../reference/NoirJS/noir_js/index.md). -- You are familiar with what are recursive proofs and you have read the [recursion explainer](../explainers/explainer-recursion.md) -- You already built a recursive circuit following [the reference](../noir/standard_library/recursion.mdx), and understand how it works. - -It is also assumed that you're not using `noir_wasm` for compilation, and instead you've used [`nargo compile`](../reference/nargo_commands.md) to generate the `json` you're now importing into your project. However, the guide should work just the same if you're using `noir_wasm`. - -:::info - -As you've read in the [explainer](../explainers/explainer-recursion.md), a recursive proof is an intermediate proof. This means that it doesn't necessarily generate the final step that makes it verifiable in a smart contract. However, it is easy to verify within another circuit. - -While "standard" usage of NoirJS packages abstracts final proofs, it currently lacks the necessary interface to abstract away intermediate proofs. This means that these proofs need to be created by using the backend directly. - -In short: - -- `noir_js` generates *only* final proofs -- `backend_barretenberg` generates both types of proofs - -::: - -In a standard recursive app, you're also dealing with at least two circuits. For the purpose of this guide, we will assume the following: - -- `main`: a circuit of type `assert(x != y)`, where `main` is marked with a `#[recursive]` attribute. This attribute states that the backend should generate proofs that are friendly for verification within another circuit. -- `recursive`: a circuit that verifies `main` - -For a full example of how recursive proofs work, please refer to the [noir-examples](https://github.com/noir-lang/noir-examples) repository. We will *not* be using it as a reference for this guide. - -## Step 1: Setup - -In a common NoirJS app, you need to instantiate a backend with something like `const backend = new Backend(circuit)`. Then you feed it to the `noir_js` interface. - -For recursion, this doesn't happen, and the only need for `noir_js` is only to `execute` a circuit and get its witness and return value. Everything else is not interfaced, so it needs to happen on the `backend` object. - -It is also recommended that you instantiate the backend with as many threads as possible, to allow for maximum concurrency: - -```js -const backend = new Backend(circuit, { threads: 8 }) -``` - -:::tip -You can use the [`os.cpus()`](https://nodejs.org/api/os.html#oscpus) object in `nodejs` or [`navigator.hardwareConcurrency`](https://developer.mozilla.org/en-US/docs/Web/API/Navigator/hardwareConcurrency) on the browser to make the most out of those glorious cpu cores -::: - -## Step 2: Generating the witness and the proof for `main` - -After instantiating the backend, you should also instantiate `noir_js`. We will use it to execute the circuit and get the witness. - -```js -const noir = new Noir(circuit) -const { witness } = noir.execute(input) -``` - -With this witness, you are now able to generate the intermediate proof for the main circuit: - -```js -const { proof, publicInputs } = await backend.generateProof(witness) -``` - -:::warning - -Always keep in mind what is actually happening on your development process, otherwise you'll quickly become confused about what circuit we are actually running and why! - -In this case, you can imagine that Alice (running the `main` circuit) is proving something to Bob (running the `recursive` circuit), and Bob is verifying her proof within his proof. - -With this in mind, it becomes clear that our intermediate proof is the one *meant to be verified within another circuit*, so it must be Alice's. Actually, the only final proof in this theoretical scenario would be the last one, sent on-chain. - -::: - -## Step 3 - Verification and proof artifacts - -Optionally, you are able to verify the intermediate proof: - -```js -const verified = await backend.verifyProof({ proof, publicInputs }) -``` - -This can be useful to make sure our intermediate proof was correctly generated. But the real goal is to do it within another circuit. For that, we need to generate recursive proof artifacts that will be passed to the circuit that is verifying the proof we just generated. Instead of passing the proof and verification key as a byte array, we pass them as fields which makes it cheaper to verify in a circuit: - -```js -const { proofAsFields, vkAsFields, vkHash } = await backend.generateRecursiveProofArtifacts( { publicInputs, proof }, publicInputsCount) -``` - -This call takes the public inputs and the proof, but also the public inputs count. While this is easily retrievable by simply counting the `publicInputs` length, the backend interface doesn't currently abstract it away. - -:::info - -The `proofAsFields` has a constant size `[Field; 93]` and verification keys in Barretenberg are always `[Field; 114]`. - -::: - -:::warning - -One common mistake is to forget *who* makes this call. - -In a situation where Alice is generating the `main` proof, if she generates the proof artifacts and sends them to Bob, which gladly takes them as true, this would mean Alice could prove anything! - -Instead, Bob needs to make sure *he* extracts the proof artifacts, using his own instance of the `main` circuit backend. This way, Alice has to provide a valid proof for the correct `main` circuit. - -::: - -## Step 4 - Recursive proof generation - -With the artifacts, generating a recursive proof is no different from a normal proof. You simply use the `backend` (with the recursive circuit) to generate it: - -```js -const recursiveInputs = { - verification_key: vkAsFields, // array of length 114 - proof: proofAsFields, // array of length 93 + size of public inputs - publicInputs: [mainInput.y], // using the example above, where `y` is the only public input - key_hash: vkHash, -} - -const { witness, returnValue } = noir.execute(recursiveInputs) // we're executing the recursive circuit now! -const { proof, publicInputs } = backend.generateProof(witness) -const verified = backend.verifyProof({ proof, publicInputs }) -``` - -You can obviously chain this proof into another proof. In fact, if you're using recursive proofs, you're probably interested of using them this way! - -:::tip - -Managing circuits and "who does what" can be confusing. To make sure your naming is consistent, you can keep them in an object. For example: - -```js -const circuits = { - main: mainJSON, - recursive: recursiveJSON -} -const backends = { - main: new BarretenbergBackend(circuits.main), - recursive: new BarretenbergBackend(circuits.recursive) -} -const noir_programs = { - main: new Noir(circuits.main), - recursive: new Noir(circuits.recursive) -} -``` - -This allows you to neatly call exactly the method you want without conflicting names: - -```js -// Alice runs this 👇 -const { witness: mainWitness } = await noir_programs.main.execute(input) -const proof = await backends.main.generateProof(mainWitness) - -// Bob runs this 👇 -const verified = await backends.main.verifyProof(proof) -const { proofAsFields, vkAsFields, vkHash } = await backends.main.generateRecursiveProofArtifacts( - proof, - numPublicInputs, -); -const { witness: recursiveWitness } = await noir_programs.recursive.execute(recursiveInputs) -const recursiveProof = await backends.recursive.generateProof(recursiveWitness); -``` - -::: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-solidity-verifier.md deleted file mode 100644 index a8169595b3d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/how-to-solidity-verifier.md +++ /dev/null @@ -1,259 +0,0 @@ ---- -title: Generate a Solidity Verifier -description: - Learn how to run the verifier as a smart contract on the blockchain. Compile a Solidity verifier - contract for your Noir program and deploy it on any EVM blockchain acting as a verifier smart - contract. Read more to find out -keywords: - [ - solidity verifier, - smart contract, - blockchain, - compiler, - plonk_vk.sol, - EVM blockchain, - verifying Noir programs, - proving backend, - Barretenberg, - ] -sidebar_position: 0 -pagination_next: tutorials/noirjs_app ---- - -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. - -This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. - -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: - -- You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network -- You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/installation/index.md) with Nargo and the example Hello Noir circuit -- You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. - -## Rundown - -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: - -1. How to generate a solidity smart contract -2. How to compile the smart contract in the RemixIDE -3. How to deploy it to a testnet - -## Step 1 - Generate a contract - -This is by far the most straightforward step. Just run: - -```sh -nargo compile -``` - -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: - -```sh -# Here we pass the path to the newly generated Noir artifact. -bb write_vk -b ./target/.json -bb contract -``` - -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. - -You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/hello_noir/index.md#proving-backend). - -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. - -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: - -## Step 2 - Compiling - -We will mostly skip the details of RemixIDE, as the UI can change from version to version. For now, we can just open -
Remix and create a blank workspace. - -![Create Workspace](@site/static/img/how-tos/solidity_verifier_1.png) - -We will create a new file to contain the contract Nargo generated, and copy-paste its content. - -:::warning - -You'll likely see a warning advising you to not trust pasted code. While it is an important warning, it is irrelevant in the context of this guide and can be ignored. We will not be deploying anywhere near a mainnet. - -::: - -To compile our the verifier, we can navigate to the compilation tab: - -![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) - -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: - -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) - -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: - -![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) - -## Step 3 - Deploying - -At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. - -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": - -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) - -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. - -:::note - -Why "UltraVerifier"? - -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. - -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: - -## Step 4 - Verifying - -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: - -```solidity -function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) -``` - -When using the default example in the [Hello Noir](../getting_started/hello_noir/index.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. - -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/hello_noir/index.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the - -```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') - -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS - -echo "Proof:" -echo "0x$HEX_PROOF" -``` - -Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. - -A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): - -```solidity -function castVote(bytes calldata proof, uint proposalId, uint vote, bytes32 nullifierHash) public returns (bool) { - // ... - bytes32[] memory publicInputs = new bytes32[](4); - publicInputs[0] = merkleRoot; - publicInputs[1] = bytes32(proposalId); - publicInputs[2] = bytes32(vote); - publicInputs[3] = nullifierHash; - require(verifier.verify(proof, publicInputs), "Invalid proof"); -``` - -:::info[Return Values] - -A circuit doesn't have the concept of a return value. Return values are just syntactic sugar in Noir. - -Under the hood, the return value is passed as an input to the circuit and is checked at the end of the circuit program. - -For example, if you have Noir program like this: - -```rust -fn main( - // Public inputs - pubkey_x: pub Field, - pubkey_y: pub Field, - // Private inputs - priv_key: Field, -) -> pub Field -``` - -the `verify` function will expect the public inputs array (second function parameter) to be of length 3, the two inputs and the return value. - -Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. - -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. - -::: - -:::tip[Structs] - -You can pass structs to the verifier contract. They will be flattened so that the array of inputs is 1-dimensional array. - -For example, consider the following program: - -```rust -struct Type1 { - val1: Field, - val2: Field, -} - -struct Nested { - t1: Type1, - is_true: bool, -} - -fn main(x: pub Field, nested: pub Nested, y: pub Field) { - //... -} -``` - -The order of these inputs would be flattened to: `[x, nested.t1.val1, nested.t1.val2, nested.is_true, y]` - -::: - -The other function you can call is our entrypoint `verify` function, as defined above. - -:::tip - -It's worth noticing that the `verify` function is actually a `view` function. A `view` function does not alter the blockchain state, so it doesn't need to be distributed (i.e. it will run only on the executing node), and therefore doesn't cost any gas. - -This can be particularly useful in some situations. If Alice generated a proof and wants Bob to verify its correctness, Bob doesn't need to run Nargo, NoirJS, or any Noir specific infrastructure. He can simply make a call to the blockchain with the proof and verify it is correct without paying any gas. - -It would be incorrect to say that a Noir proof verification costs any gas at all. However, most of the time the result of `verify` is used to modify state (for example, to update a balance, a game state, etc). In that case the whole network needs to execute it, which does incur gas costs (calldata and execution, but not storage). - -::: - -## A Note on EVM chains - -Noir proof verification requires the ecMul, ecAdd and ecPairing precompiles. Not all EVM chains support EC Pairings, notably some of the ZK-EVMs. This means that you won't be able to use the verifier contract in all of them. You can find an incomplete list of which EVM chains support these precompiles [here](https://www.evmdiff.com/features?feature=precompiles). - -For example, chains like `zkSync ERA` and `Polygon zkEVM` do not currently support these precompiles, so proof verification via Solidity verifier contracts won't work. Here's a quick list of EVM chains that have been tested and are known to work: - -- Optimism -- Arbitrum -- Polygon PoS -- Scroll -- Celo -- BSC -- Blast L2 -- Avalanche C-Chain -- Mode -- Linea -- Moonbeam - -If you test any other chains, please open a PR on this page to update the list. See [this doc](https://github.com/noir-lang/noir-starter/tree/main/with-foundry#testing-on-chain) for more info about testing verifier contracts on different EVM chains. - -## What's next - -Now that you know how to call a Noir Solidity Verifier on a smart contract using Remix, you should be comfortable with using it with some programmatic frameworks, such as [hardhat](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) and [foundry](https://github.com/noir-lang/noir-starter/tree/main/with-foundry). - -You can find other tools, examples, boilerplates and libraries in the [awesome-noir](https://github.com/noir-lang/awesome-noir) repository. - -You should also be ready to write and deploy your first NoirJS app and start generating proofs on websites, phones, and NodeJS environments! Head on to the [NoirJS tutorial](../tutorials/noirjs_app.md) to learn how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/index.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/index.mdx deleted file mode 100644 index a6bd306f91d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/index.mdx +++ /dev/null @@ -1,67 +0,0 @@ ---- -title: Noir Lang -hide_title: true -description: - Learn about the public alpha release of Noir, a domain specific language heavily influenced by Rust that compiles to - an intermediate language which can be compiled to an arithmetic circuit or a rank-1 constraint system. -keywords: - [Noir, - Domain Specific Language, - Rust, - Intermediate Language, - Arithmetic Circuit, - Rank-1 Constraint System, - Ethereum Developers, - Protocol Developers, - Blockchain Developers, - Proving System, - Smart Contract Language] -sidebar_position: 0 ---- - -import Tabs from '@theme/Tabs'; -import TabItem from '@theme/TabItem'; - -Noir Logo - -Noir is an open-source Domain-Specific Language for safe and seamless construction of privacy-preserving Zero-Knowledge programs, requiring no previous knowledge on the underlying mathematics or cryptography. - -ZK programs are programs that can generate short proofs of statements without revealing all inputs to the statements. You can read more about Zero-Knowledge Proofs [here](https://dev.to/spalladino/a-beginners-intro-to-coding-zero-knowledge-proofs-c56). - -## What's new about Noir? - -Noir works differently from most ZK languages by taking a two-pronged path. First, it compiles the program to an adaptable intermediate language known as ACIR. From there, depending on a given project's needs, ACIR can be further compiled into an arithmetic circuit for integration with the proving backend. - -:::info - -Noir is backend agnostic, which means it makes no assumptions on which proving backend powers the ZK proof. Being the language that powers [Aztec Contracts](https://docs.aztec.network/developers/contracts/main), it defaults to Aztec's Barretenberg proving backend. - -However, the ACIR output can be transformed to be compatible with other PLONK-based backends, or into a [rank-1 constraint system](https://www.rareskills.io/post/rank-1-constraint-system) suitable for backends such as Arkwork's Marlin. - -::: - -## Who is Noir for? - -Noir can be used both in complex cloud-based backends and in user's smartphones, requiring no knowledge on the underlying math or cryptography. From authorization systems that keep a password in the user's device, to complex on-chain verification of recursive proofs, Noir is designed to abstract away complexity without any significant overhead. Here are some examples of situations where Noir can be used: - - - - Noir Logo - - Aztec Contracts leverage Noir to allow for the storage and execution of private information. Writing an Aztec Contract is as easy as writing Noir, and Aztec developers can easily interact with the network storage and execution through the [Aztec.nr](https://docs.aztec.network/developers/contracts/main) library. - - - Soliditry Verifier Example - Noir can auto-generate Solidity verifier contracts that verify Noir proofs. This allows for non-interactive verification of proofs containing private information in an immutable system. This feature powers a multitude of use-case scenarios, from P2P chess tournaments, to [Aztec Layer-2 Blockchain](https://docs.aztec.network/) - - - Aztec Labs developed NoirJS, an easy interface to generate and verify Noir proofs in a Javascript environment. This allows for Noir to be used in webpages, mobile apps, games, and any other environment supporting JS execution in a standalone manner. - - - - -## Libraries - -Noir is meant to be easy to extend by simply importing Noir libraries just like in Rust. -The [awesome-noir repo](https://github.com/noir-lang/awesome-noir#libraries) is a collection of libraries developed by the Noir community. -Writing a new library is easy and makes code be composable and easy to reuse. See the section on [dependencies](noir/modules_packages_crates/dependencies.md) for more information. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/control_flow.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/control_flow.md deleted file mode 100644 index 045d3c3a5f5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/control_flow.md +++ /dev/null @@ -1,77 +0,0 @@ ---- -title: Control Flow -description: - Learn how to use loops and if expressions in the Noir programming language. Discover the syntax - and examples for for loops and if-else statements. -keywords: [Noir programming language, loops, for loop, if-else statements, Rust syntax] -sidebar_position: 2 ---- - -## If Expressions - -Noir supports `if-else` statements. The syntax is most similar to Rust's where it is not required -for the statement's conditional to be surrounded by parentheses. - -```rust -let a = 0; -let mut x: u32 = 0; - -if a == 0 { - if a != 0 { - x = 6; - } else { - x = 2; - } -} else { - x = 5; - assert(x == 5); -} -assert(x == 2); -``` - -## Loops - -Noir has one kind of loop: the `for` loop. `for` loops allow you to repeat a block of code multiple -times. - -The following block of code between the braces is run 10 times. - -```rust -for i in 0..10 { - // do something -} -``` - -The index for loops is of type `u64`. - -### Break and Continue - -In unconstrained code, `break` and `continue` are also allowed in `for` loops. These are only allowed -in unconstrained code since normal constrained code requires that Noir knows exactly how many iterations -a loop may have. `break` and `continue` can be used like so: - -```rust -for i in 0 .. 10 { - println("Iteration start") - - if i == 2 { - continue; - } - - if i == 5 { - break; - } - - println(i); -} -println("Loop end") -``` - -When used, `break` will end the current loop early and jump to the statement after the for loop. In the example -above, the `break` will stop the loop and jump to the `println("Loop end")`. - -`continue` will stop the current iteration of the loop, and jump to the start of the next iteration. In the example -above, `continue` will jump to `println("Iteration start")` when used. Note that the loop continues as normal after this. -The iteration variable `i` is still increased by one as normal when `continue` is used. - -`break` and `continue` cannot currently be used to jump out of more than a single loop at a time. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/arrays.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/arrays.md deleted file mode 100644 index d26f6dff070..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/arrays.md +++ /dev/null @@ -1,253 +0,0 @@ ---- -title: Arrays -description: - Dive into the Array data type in Noir. Grasp its methods, practical examples, and best practices for efficiently using Arrays in your Noir code. -keywords: - [ - noir, - array type, - methods, - examples, - indexing, - ] -sidebar_position: 4 ---- - -An array is one way of grouping together values into one compound type. Array types can be inferred -or explicitly specified via the syntax `[; ]`: - -```rust -fn main(x : Field, y : Field) { - let my_arr = [x, y]; - let your_arr: [Field; 2] = [x, y]; -} -``` - -Here, both `my_arr` and `your_arr` are instantiated as an array containing two `Field` elements. - -Array elements can be accessed using indexing: - -```rust -fn main() { - let a = [1, 2, 3, 4, 5]; - - let first = a[0]; - let second = a[1]; -} -``` - -All elements in an array must be of the same type (i.e. homogeneous). That is, an array cannot group -a `Field` value and a `u8` value together for example. - -You can write mutable arrays, like: - -```rust -fn main() { - let mut arr = [1, 2, 3, 4, 5]; - assert(arr[0] == 1); - - arr[0] = 42; - assert(arr[0] == 42); -} -``` - -You can instantiate a new array of a fixed size with the same value repeated for each element. The following example instantiates an array of length 32 where each element is of type Field and has the value 0. - -```rust -let array: [Field; 32] = [0; 32]; -``` - -Like in Rust, arrays in Noir are a fixed size. However, if you wish to convert an array to a [slice](./slices.mdx), you can just call `as_slice` on your array: - -```rust -let array: [Field; 32] = [0; 32]; -let sl = array.as_slice() -``` - -You can define multidimensional arrays: - -```rust -let array : [[Field; 2]; 2]; -let element = array[0][0]; -``` - -However, multidimensional slices are not supported. For example, the following code will error at compile time: - -```rust -let slice : [[Field]] = &[]; -``` - -## Types - -You can create arrays of primitive types or structs. There is not yet support for nested arrays -(arrays of arrays) or arrays of structs that contain arrays. - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for arrays. -Each of these functions are located within the generic impl `impl [T; N] {`. -So anywhere `self` appears, it refers to the variable `self: [T; N]`. - -### len - -Returns the length of an array - -```rust -fn len(self) -> Field -``` - -example - -```rust -fn main() { - let array = [42, 42]; - assert(array.len() == 2); -} -``` - -### sort - -Returns a new sorted array. The original array remains untouched. Notice that this function will -only work for arrays of fields or integers, not for any arbitrary type. This is because the sorting -logic it uses internally is optimized specifically for these values. If you need a sort function to -sort any type, you should use the function `sort_via` described below. - -```rust -fn sort(self) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32]; - let sorted = arr.sort(); - assert(sorted == [32, 42]); -} -``` - -### sort_via - -Sorts the array with a custom comparison function - -```rust -fn sort_via(self, ordering: fn(T, T) -> bool) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32] - let sorted_ascending = arr.sort_via(|a, b| a <= b); - assert(sorted_ascending == [32, 42]); // verifies - - let sorted_descending = arr.sort_via(|a, b| a >= b); - assert(sorted_descending == [32, 42]); // does not verify -} -``` - -### map - -Applies a function to each element of the array, returning a new array containing the mapped elements. - -```rust -fn map(self, f: fn(T) -> U) -> [U; N] -``` - -example - -```rust -let a = [1, 2, 3]; -let b = a.map(|a| a * 2); // b is now [2, 4, 6] -``` - -### fold - -Applies a function to each element of the array, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn(U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the array, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = [1]; -let a2 = [1, 2]; -let a3 = [1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let arr = [2, 2, 2, 2, 2]; - let folded = arr.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn(T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let reduced = arr.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let all = arr.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 5]; - let any = arr.any(|a| a == 5); - assert(any); -} - -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/fields.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/fields.md deleted file mode 100644 index a10a4810788..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/fields.md +++ /dev/null @@ -1,192 +0,0 @@ ---- -title: Fields -description: - Dive deep into the Field data type in Noir. Understand its methods, practical examples, and best practices to effectively use Fields in your Noir programs. -keywords: - [ - noir, - field type, - methods, - examples, - best practices, - ] -sidebar_position: 0 ---- - -The field type corresponds to the native field type of the proving backend. - -The size of a Noir field depends on the elliptic curve's finite field for the proving backend -adopted. For example, a field would be a 254-bit integer when paired with the default backend that -spans the Grumpkin curve. - -Fields support integer arithmetic and are often used as the default numeric type in Noir: - -```rust -fn main(x : Field, y : Field) { - let z = x + y; -} -``` - -`x`, `y` and `z` are all private fields in this example. Using the `let` keyword we defined a new -private value `z` constrained to be equal to `x + y`. - -If proving efficiency is of priority, fields should be used as a default for solving problems. -Smaller integer types (e.g. `u64`) incur extra range constraints. - -## Methods - -After declaring a Field, you can use these common methods on it: - -### to_le_bits - -Transforms the field into an array of bits, Little Endian. - -```rust -fn to_le_bits(_x : Field, _bit_size: u32) -> [u1] -``` - -example: - -```rust -fn main() { - let field = 2; - let bits = field.to_le_bits(32); -} -``` - -### to_be_bits - -Transforms the field into an array of bits, Big Endian. - -```rust -fn to_be_bits(_x : Field, _bit_size: u32) -> [u1] -``` - -example: - -```rust -fn main() { - let field = 2; - let bits = field.to_be_bits(32); -} -``` - -### to_le_bytes - -Transforms into an array of bytes, Little Endian - -```rust -fn to_le_bytes(_x : Field, byte_size: u32) -> [u8] -``` - -example: - -```rust -fn main() { - let field = 2; - let bytes = field.to_le_bytes(4); -} -``` - -### to_be_bytes - -Transforms into an array of bytes, Big Endian - -```rust -fn to_be_bytes(_x : Field, byte_size: u32) -> [u8] -``` - -example: - -```rust -fn main() { - let field = 2; - let bytes = field.to_be_bytes(4); -} -``` - -### to_le_radix - -Decomposes into a vector over the specified base, Little Endian - -```rust -fn to_le_radix(_x : Field, _radix: u32, _result_len: u32) -> [u8] -``` - -example: - -```rust -fn main() { - let field = 2; - let radix = field.to_le_radix(256, 4); -} -``` - -### to_be_radix - -Decomposes into a vector over the specified base, Big Endian - -```rust -fn to_be_radix(_x : Field, _radix: u32, _result_len: u32) -> [u8] -``` - -example: - -```rust -fn main() { - let field = 2; - let radix = field.to_be_radix(256, 4); -} -``` - -### pow_32 - -Returns the value to the power of the specified exponent - -```rust -fn pow_32(self, exponent: Field) -> Field -``` - -example: - -```rust -fn main() { - let field = 2 - let pow = field.pow_32(4); - assert(pow == 16); -} -``` - -### assert_max_bit_size - -Adds a constraint to specify that the field can be represented with `bit_size` number of bits - -```rust -fn assert_max_bit_size(self, bit_size: u32) -``` - -example: - -```rust -fn main() { - let field = 2 - field.assert_max_bit_size(32); -} -``` - -### sgn0 - -Parity of (prime) Field element, i.e. sgn0(x mod p) = 0 if x ∈ \{0, ..., p-1\} is even, otherwise sgn0(x mod p) = 1. - -```rust -fn sgn0(self) -> u1 -``` - - -### lt - -Returns true if the field is less than the other field - -```rust -pub fn lt(self, another: Field) -> bool -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/index.md deleted file mode 100644 index 3eadb2dc8a4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/index.md +++ /dev/null @@ -1,118 +0,0 @@ ---- -title: Data Types -description: - Get a clear understanding of the two categories of Noir data types - primitive types and compound - types. Learn about their characteristics, differences, and how to use them in your Noir - programming. -keywords: - [ - noir, - data types, - primitive types, - compound types, - private types, - public types, - ] ---- - -Every value in Noir has a type, which determines which operations are valid for it. - -All values in Noir are fundamentally composed of `Field` elements. For a more approachable -developing experience, abstractions are added on top to introduce different data types in Noir. - -Noir has two category of data types: primitive types (e.g. `Field`, integers, `bool`) and compound -types that group primitive types (e.g. arrays, tuples, structs). Each value can either be private or -public. - -## Private & Public Types - -A **private value** is known only to the Prover, while a **public value** is known by both the -Prover and Verifier. Mark values as `private` when the value should only be known to the prover. All -primitive types (including individual fields of compound types) in Noir are private by default, and -can be marked public when certain values are intended to be revealed to the Verifier. - -> **Note:** For public values defined in Noir programs paired with smart contract verifiers, once -> the proofs are verified on-chain the values can be considered known to everyone that has access to -> that blockchain. - -Public data types are treated no differently to private types apart from the fact that their values -will be revealed in proofs generated. Simply changing the value of a public type will not change the -circuit (where the same goes for changing values of private types as well). - -_Private values_ are also referred to as _witnesses_ sometimes. - -> **Note:** The terms private and public when applied to a type (e.g. `pub Field`) have a different -> meaning than when applied to a function (e.g. `pub fn foo() {}`). -> -> The former is a visibility modifier for the Prover to interpret if a value should be made known to -> the Verifier, while the latter is a visibility modifier for the compiler to interpret if a -> function should be made accessible to external Noir programs like in other languages. - -### pub Modifier - -All data types in Noir are private by default. Types are explicitly declared as public using the -`pub` modifier: - -```rust -fn main(x : Field, y : pub Field) -> pub Field { - x + y -} -``` - -In this example, `x` is **private** while `y` and `x + y` (the return value) are **public**. Note -that visibility is handled **per variable**, so it is perfectly valid to have one input that is -private and another that is public. - -> **Note:** Public types can only be declared through parameters on `main`. - -## Type Aliases - -A type alias is a new name for an existing type. Type aliases are declared with the keyword `type`: - -```rust -type Id = u8; - -fn main() { - let id: Id = 1; - let zero: u8 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can also be used with [generics](../generics.md): - -```rust -type Id = Size; - -fn main() { - let id: Id = 1; - let zero: u32 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can even refer to other aliases. An error will be issued if they form a cycle: - -```rust -// Ok! -type A = B; -type B = Field; - -type Bad1 = Bad2; - -// error: Dependency cycle found -type Bad2 = Bad1; -// ^^^^^^^^^^^ 'Bad2' recursively depends on itself: Bad2 -> Bad1 -> Bad2 -``` - -## Wildcard Type -Noir can usually infer the type of the variable from the context, so specifying the type of a variable is only required when it cannot be inferred. However, specifying a complex type can be tedious, especially when it has multiple generic arguments. Often some of the generic types can be inferred from the context, and Noir only needs a hint to properly infer the other types. We can partially specify a variable's type by using `_` as a marker, indicating where we still want the compiler to infer the type. - -```rust -let a: [_; 4] = foo(b); -``` - - -### BigInt - -You can achieve BigInt functionality using the [Noir BigInt](https://github.com/shuklaayush/noir-bigint) library. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/integers.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/integers.md deleted file mode 100644 index a1d59bf3166..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/integers.md +++ /dev/null @@ -1,156 +0,0 @@ ---- -title: Integers -description: Explore the Integer data type in Noir. Learn about its methods, see real-world examples, and grasp how to efficiently use Integers in your Noir code. -keywords: [noir, integer types, methods, examples, arithmetic] -sidebar_position: 1 ---- - -An integer type is a range constrained field type. -The Noir frontend supports both unsigned and signed integer types. -The allowed sizes are 1, 8, 16, 32 and 64 bits. - -:::info - -When an integer is defined in Noir without a specific type, it will default to `Field`. - -The one exception is for loop indices which default to `u64` since comparisons on `Field`s are not possible. - -::: - -## Unsigned Integers - -An unsigned integer type is specified first with the letter `u` (indicating its unsigned nature) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: u8 = 1; - let y: u8 = 1; - let z = x + y; - assert (z == 2); -} -``` - -The bit size determines the maximum value the integer type can store. For example, a `u8` variable can store a value in the range of 0 to 255 (i.e. $\\2^{8}-1\\$). - -## Signed Integers - -A signed integer type is specified first with the letter `i` (which stands for integer) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: i8 = -1; - let y: i8 = -1; - let z = x + y; - assert (z == -2); -} -``` - -The bit size determines the maximum and minimum range of value the integer type can store. For example, an `i8` variable can store a value in the range of -128 to 127 (i.e. $\\-2^{7}\\$ to $\\2^{7}-1\\$). - -## 128 bits Unsigned Integers - -The built-in structure `U128` allows you to use 128-bit unsigned integers almost like a native integer type. However, there are some differences to keep in mind: -- You cannot cast between a native integer and `U128` -- There is a higher performance cost when using `U128`, compared to a native type. - -Conversion between unsigned integer types and U128 are done through the use of `from_integer` and `to_integer` functions. `from_integer` also accepts the `Field` type as input. - -```rust -fn main() { - let x = U128::from_integer(23); - let y = U128::from_hex("0x7"); - let z = x + y; - assert(z.to_integer() == 30); -} -``` - -`U128` is implemented with two 64 bits limbs, representing the low and high bits, which explains the performance cost. You should expect `U128` to be twice more costly for addition and four times more costly for multiplication. -You can construct a U128 from its limbs: -```rust -fn main(x: u64, y: u64) { - let x = U128::from_u64s_be(x,y); - assert(z.hi == x as Field); - assert(z.lo == y as Field); -} -``` - -Note that the limbs are stored as Field elements in order to avoid unnecessary conversions. -Apart from this, most operations will work as usual: - -```rust -fn main(x: U128, y: U128) { - // multiplication - let c = x * y; - // addition and subtraction - let c = c - x + y; - // division - let c = x / y; - // bit operation; - let c = x & y | y; - // bit shift - let c = x << y; - // comparisons; - let c = x < y; - let c = x == y; -} -``` - -## Overflows - -Computations that exceed the type boundaries will result in overflow errors. This happens with both signed and unsigned integers. For example, attempting to prove: - -```rust -fn main(x: u8, y: u8) { - let z = x + y; -} -``` - -With: - -```toml -x = "255" -y = "1" -``` - -Would result in: - -``` -$ nargo execute -error: Assertion failed: 'attempt to add with overflow' -┌─ ~/src/main.nr:9:13 -│ -│ let z = x + y; -│ ----- -│ -= Call stack: - ... -``` - -A similar error would happen with signed integers: - -```rust -fn main() { - let x: i8 = -118; - let y: i8 = -11; - let z = x + y; -} -``` - -### Wrapping methods - -Although integer overflow is expected to error, some use-cases rely on wrapping. For these use-cases, the standard library provides `wrapping` variants of certain common operations: - -```rust -fn wrapping_add(x: T, y: T) -> T; -fn wrapping_sub(x: T, y: T) -> T; -fn wrapping_mul(x: T, y: T) -> T; -``` - -Example of how it is used: - -```rust - -fn main(x: u8, y: u8) -> pub u8 { - std::wrapping_add(x, y) -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/slices.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/slices.mdx deleted file mode 100644 index 95da2030843..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/slices.mdx +++ /dev/null @@ -1,358 +0,0 @@ ---- -title: Slices -description: Explore the Slice data type in Noir. Understand its methods, see real-world examples, and learn how to effectively use Slices in your Noir programs. -keywords: [noir, slice type, methods, examples, subarrays] -sidebar_position: 5 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A slice is a dynamically-sized view into a sequence of elements. They can be resized at runtime, but because they don't own the data, they cannot be returned from a circuit. You can treat slices as arrays without a constrained size. - -```rust -fn main() -> pub u32 { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -To write a slice literal, use a preceeding ampersand as in: `&[0; 2]` or -`&[1, 2, 3]`. - -It is important to note that slices are not references to arrays. In Noir, -`&[..]` is more similar to an immutable, growable vector. - -View the corresponding test file [here][test-file]. - -[test-file]: https://github.com/noir-lang/noir/blob/f387ec1475129732f72ba294877efdf6857135ac/crates/nargo_cli/tests/test_data_ssa_refactor/slices/src/main.nr - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for slices: - -### push_back - -Pushes a new element to the end of the slice, returning a new slice with a length one greater than the original unmodified slice. - -```rust -fn push_back(_self: [T], _elem: T) -> [T] -``` - -example: - -```rust -fn main() -> pub Field { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -View the corresponding test file [here][test-file]. - -### push_front - -Returns a new array with the specified element inserted at index 0. The existing elements indexes are incremented by 1. - -```rust -fn push_front(_self: Self, _elem: T) -> Self -``` - -Example: - -```rust -let mut new_slice: [Field] = &[]; -new_slice = new_slice.push_front(20); -assert(new_slice[0] == 20); // returns true -``` - -View the corresponding test file [here][test-file]. - -### pop_front - -Returns a tuple of two items, the first element of the array and the rest of the array. - -```rust -fn pop_front(_self: Self) -> (T, Self) -``` - -Example: - -```rust -let (first_elem, rest_of_slice) = slice.pop_front(); -``` - -View the corresponding test file [here][test-file]. - -### pop_back - -Returns a tuple of two items, the beginning of the array with the last element omitted and the last element. - -```rust -fn pop_back(_self: Self) -> (Self, T) -``` - -Example: - -```rust -let (popped_slice, last_elem) = slice.pop_back(); -``` - -View the corresponding test file [here][test-file]. - -### append - -Loops over a slice and adds it to the end of another. - -```rust -fn append(mut self, other: Self) -> Self -``` - -Example: - -```rust -let append = &[1, 2].append(&[3, 4, 5]); -``` - -### insert - -Inserts an element at a specified index and shifts all following elements by 1. - -```rust -fn insert(_self: Self, _index: Field, _elem: T) -> Self -``` - -Example: - -```rust -new_slice = rest_of_slice.insert(2, 100); -assert(new_slice[2] == 100); -``` - -View the corresponding test file [here][test-file]. - -### remove - -Remove an element at a specified index, shifting all elements after it to the left, returning the altered slice and the removed element. - -```rust -fn remove(_self: Self, _index: Field) -> (Self, T) -``` - -Example: - -```rust -let (remove_slice, removed_elem) = slice.remove(3); -``` - -### len - -Returns the length of a slice - -```rust -fn len(self) -> Field -``` - -Example: - -```rust -fn main() { - let slice = &[42, 42]; - assert(slice.len() == 2); -} -``` - -### as_array - -Converts this slice into an array. - -Make sure to specify the size of the resulting array. -Panics if the resulting array length is different than the slice's length. - -```rust -fn as_array(self) -> [T; N] -``` - -Example: - -```rust -fn main() { - let slice = &[5, 6]; - - // Always specify the length of the resulting array! - let array: [Field; 2] = slice.as_array(); - - assert(array[0] == slice[0]); - assert(array[1] == slice[1]); -} -``` - -### map - -Applies a function to each element of the slice, returning a new slice containing the mapped elements. - -```rust -fn map(self, f: fn[Env](T) -> U) -> [U] -``` - -example - -```rust -let a = &[1, 2, 3]; -let b = a.map(|a| a * 2); // b is now &[2, 4, 6] -``` - -### fold - -Applies a function to each element of the slice, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn[Env](U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the slice, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = &[1]; -let a2 = &[1, 2]; -let a3 = &[1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let folded = slice.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn[Env](T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let reduced = slice.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### filter - -Returns a new slice containing only elements for which the given predicate returns true. - -```rust -fn filter(self, f: fn[Env](T) -> bool) -> Self -``` - -example: - -```rust -fn main() { - let slice = &[1, 2, 3, 4, 5]; - let odds = slice.filter(|x| x % 2 == 1); - assert_eq(odds, &[1, 3, 5]); -} -``` - -### join - -Flatten each element in the slice into one value, separated by `separator`. - -Note that although slices implement `Append`, `join` cannot be used on slice -elements since nested slices are prohibited. - -```rust -fn join(self, separator: T) -> T where T: Append -``` - -example: - -```rust -struct Accumulator { - total: Field, -} - -// "Append" two accumulators by adding them -impl Append for Accumulator { - fn empty() -> Self { - Self { total: 0 } - } - - fn append(self, other: Self) -> Self { - Self { total: self.total + other.total } - } -} - -fn main() { - let slice = &[1, 2, 3, 4, 5].map(|total| Accumulator { total }); - - let result = slice.join(Accumulator::empty()); - assert_eq(result, Accumulator { total: 15 }); - - // We can use a non-empty separator to insert additional elements to sum: - let separator = Accumulator { total: 10 }; - let result = slice.join(separator); - assert_eq(result, Accumulator { total: 55 }); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let all = slice.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 5]; - let any = slice.any(|a| a == 5); - assert(any); -} - -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/strings.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/strings.md deleted file mode 100644 index 1fdee42425e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/strings.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Strings -description: - Discover the String data type in Noir. Learn about its methods, see real-world examples, and understand how to effectively manipulate and use Strings in Noir. -keywords: - [ - noir, - string type, - methods, - examples, - concatenation, - ] -sidebar_position: 3 ---- - - -The string type is a fixed length value defined with `str`. - -You can use strings in `assert()` functions or print them with -`println()`. See more about [Logging](../../standard_library/logging.md). - -```rust - -fn main(message : pub str<11>, hex_as_string : str<4>) { - println(message); - assert(message == "hello world"); - assert(hex_as_string == "0x41"); -} -``` - -You can convert a `str` to a byte array by calling `as_bytes()` -or a vector by calling `as_bytes_vec()`. - -```rust -fn main() { - let message = "hello world"; - let message_bytes = message.as_bytes(); - let mut message_vec = message.as_bytes_vec(); - assert(message_bytes.len() == 11); - assert(message_bytes[0] == 104); - assert(message_bytes[0] == message_vec.get(0)); -} -``` - -## Escape characters - -You can use escape characters for your strings: - -| Escape Sequence | Description | -|-----------------|-----------------| -| `\r` | Carriage Return | -| `\n` | Newline | -| `\t` | Tab | -| `\0` | Null Character | -| `\"` | Double Quote | -| `\\` | Backslash | - -Example: - -```rust -let s = "Hello \"world" // prints "Hello "world" -let s = "hey \tyou"; // prints "hey you" -``` - -## Raw strings - -A raw string begins with the letter `r` and is optionally delimited by a number of hashes `#`. - -Escape characters are *not* processed within raw strings. All contents are interpreted literally. - -Example: - -```rust -let s = r"Hello world"; -let s = r#"Simon says "hello world""#; - -// Any number of hashes may be used (>= 1) as long as the string also terminates with the same number of hashes -let s = r#####"One "#, Two "##, Three "###, Four "####, Five will end the string."#####; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/structs.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/structs.md deleted file mode 100644 index dbf68c99813..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/structs.md +++ /dev/null @@ -1,70 +0,0 @@ ---- -title: Structs -description: - Explore the Struct data type in Noir. Learn about its methods, see real-world examples, and grasp how to effectively define and use Structs in your Noir programs. -keywords: - [ - noir, - struct type, - methods, - examples, - data structures, - ] -sidebar_position: 8 ---- - -A struct also allows for grouping multiple values of different types. Unlike tuples, we can also -name each field. - -> **Note:** The usage of _field_ here refers to each element of the struct and is unrelated to the -> field type of Noir. - -Defining a struct requires giving it a name and listing each field within as `: ` pairs: - -```rust -struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` - -An instance of a struct can then be created with actual values in `: ` pairs in any -order. Struct fields are accessible using their given names: - -```rust -fn main() { - let legs = 4; - - let dog = Animal { - eyes: 2, - hands: 0, - legs, - }; - - let zero = dog.hands; -} -``` - -Structs can also be destructured in a pattern, binding each field to a new variable: - -```rust -fn main() { - let Animal { hands, legs: feet, eyes } = get_octopus(); - - let ten = hands + feet + eyes as u8; -} - -fn get_octopus() -> Animal { - let octopus = Animal { - hands: 0, - legs: 8, - eyes: 2, - }; - - octopus -} -``` - -The new variables can be bound with names different from the original struct field names, as -showcased in the `legs --> feet` binding in the example above. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/generics.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/generics.md deleted file mode 100644 index 0c1c27a2221..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/generics.md +++ /dev/null @@ -1,106 +0,0 @@ ---- -title: Generics -description: Learn how to use Generics in Noir -keywords: [Noir, Rust, generics, functions, structs] -sidebar_position: 7 ---- - -Generics allow you to use the same functions with multiple different concrete data types. You can -read more about the concept of generics in the Rust documentation -[here](https://doc.rust-lang.org/book/ch10-01-syntax.html). - -Here is a trivial example showing the identity function that supports any type. In Rust, it is -common to refer to the most general type as `T`. We follow the same convention in Noir. - -```rust -fn id(x: T) -> T { - x -} -``` - -## In Structs - -Generics are useful for specifying types in structs. For example, we can specify that a field in a -struct will be of a certain generic type. In this case `value` is of type `T`. - -```rust -struct RepeatedValue { - value: T, - count: Field, -} - -impl RepeatedValue { - fn print(self) { - for _i in 0 .. self.count { - println(self.value); - } - } -} - -fn main() { - let repeated = RepeatedValue { value: "Hello!", count: 2 }; - repeated.print(); -} -``` - -The `print` function will print `Hello!` an arbitrary number of times, twice in this case. - -If we want to be generic over array lengths (which are type-level integers), we can use numeric -generics. Using these looks just like using regular generics, but these generics can resolve to -integers at compile-time, rather than resolving to types. Here's an example of a struct that is -generic over the size of the array it contains internally: - -```rust -struct BigInt { - limbs: [u32; N], -} - -impl BigInt { - // `N` is in scope of all methods in the impl - fn first(first: BigInt, second: BigInt) -> Self { - assert(first.limbs != second.limbs); - first - - fn second(first: BigInt, second: Self) -> Self { - assert(first.limbs != second.limbs); - second - } -} -``` - -## Calling functions on generic parameters - -Since a generic type `T` can represent any type, how can we call functions on the underlying type? -In other words, how can we go from "any type `T`" to "any type `T` that has certain methods available?" - -This is what [traits](../concepts/traits.md) are for in Noir. Here's an example of a function generic over -any type `T` that implements the `Eq` trait for equality: - -```rust -fn first_element_is_equal(array1: [T; N], array2: [T; N]) -> bool - where T: Eq -{ - if (array1.len() == 0) | (array2.len() == 0) { - true - } else { - array1[0] == array2[0] - } -} - -fn main() { - assert(first_element_is_equal([1, 2, 3], [1, 5, 6])); - - // We can use first_element_is_equal for arrays of any type - // as long as we have an Eq impl for the types we pass in - let array = [MyStruct::new(), MyStruct::new()]; - assert(array_eq(array, array, MyStruct::eq)); -} - -impl Eq for MyStruct { - fn eq(self, other: MyStruct) -> bool { - self.foo == other.foo - } -} -``` - -You can find more details on traits and trait implementations on the [traits page](../concepts/traits.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/globals.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/globals.md deleted file mode 100644 index 063a3d89248..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/globals.md +++ /dev/null @@ -1,72 +0,0 @@ ---- -title: Global Variables -description: - Learn about global variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, globals, global variables, constants] -sidebar_position: 8 ---- - -## Globals - - -Noir supports global variables. The global's type can be inferred by the compiler entirely: - -```rust -global N = 5; // Same as `global N: Field = 5` - -global TUPLE = (3, 2); - -fn main() { - assert(N == 5); - assert(N == TUPLE.0 + TUPLE.1); -} -``` - -:::info - -Globals can be defined as any expression, so long as they don't depend on themselves - otherwise there would be a dependency cycle! For example: - -```rust -global T = foo(T); // dependency error -``` - -::: - - -If they are initialized to a literal integer, globals can be used to specify an array's length: - -```rust -global N: Field = 2; - -fn main(y : [Field; N]) { - assert(y[0] == y[1]) -} -``` - -A global from another module can be imported or referenced externally like any other name: - -```rust -global N = 20; - -fn main() { - assert(my_submodule::N != N); -} - -mod my_submodule { - global N: Field = 10; -} -``` - -When a global is used, Noir replaces the name with its definition on each occurrence. -This means globals defined using function calls will repeat the call each time they're used: - -```rust -global RESULT = foo(); - -fn foo() -> [Field; 100] { ... } -``` - -This is usually fine since Noir will generally optimize any function call that does not -refer to a program input into a constant. It should be kept in mind however, if the called -function performs side-effects like `println`, as these will still occur on each use. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/traits.md deleted file mode 100644 index 51305b38c16..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/traits.md +++ /dev/null @@ -1,405 +0,0 @@ ---- -title: Traits -description: - Traits in Noir can be used to abstract out a common interface for functions across - several data types. -keywords: [noir programming language, traits, interfaces, generic, protocol] -sidebar_position: 14 ---- - -## Overview - -Traits in Noir are a useful abstraction similar to interfaces or protocols in other languages. Each trait defines -the interface of several methods contained within the trait. Types can then implement this trait by providing -implementations for these methods. For example in the program: - -```rust -struct Rectangle { - width: Field, - height: Field, -} - -impl Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -fn log_area(r: Rectangle) { - println(r.area()); -} -``` - -We have a function `log_area` to log the area of a `Rectangle`. Now how should we change the program if we want this -function to work on `Triangle`s as well?: - -```rust -struct Triangle { - width: Field, - height: Field, -} - -impl Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Making `log_area` generic over all types `T` would be invalid since not all types have an `area` method. Instead, we can -introduce a new `Area` trait and make `log_area` generic over all types `T` that implement `Area`: - -```rust -trait Area { - fn area(self) -> Field; -} - -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -We also need to explicitly implement `Area` for `Rectangle` and `Triangle`. We can do that by changing their existing -impls slightly. Note that the parameter types and return type of each of our `area` methods must match those defined -by the `Area` trait. - -```rust -impl Area for Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -impl Area for Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Now we have a working program that is generic over any type of Shape that is used! Others can even use this program -as a library with their own types - such as `Circle` - as long as they also implement `Area` for these types. - -## Where Clauses - -As seen in `log_area` above, when we want to create a function or method that is generic over any type that implements -a trait, we can add a where clause to the generic function. - -```rust -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -It is also possible to apply multiple trait constraints on the same variable at once by combining traits with the `+` -operator. Similarly, we can have multiple trait constraints by separating each with a comma: - -```rust -fn foo(elements: [T], thing: U) where - T: Default + Add + Eq, - U: Bar, -{ - let mut sum = T::default(); - - for element in elements { - sum += element; - } - - if sum == T::default() { - thing.bar(); - } -} -``` - -## Generic Implementations - -You can add generics to a trait implementation by adding the generic list after the `impl` keyword: - -```rust -trait Second { - fn second(self) -> Field; -} - -impl Second for (T, Field) { - fn second(self) -> Field { - self.1 - } -} -``` - -You can also implement a trait for every type this way: - -```rust -trait Debug { - fn debug(self); -} - -impl Debug for T { - fn debug(self) { - println(self); - } -} - -fn main() { - 1.debug(); -} -``` - -### Generic Trait Implementations With Where Clauses - -Where clauses can be placed on trait implementations themselves to restrict generics in a similar way. -For example, while `impl Foo for T` implements the trait `Foo` for every type, `impl Foo for T where T: Bar` -will implement `Foo` only for types that also implement `Bar`. This is often used for implementing generic types. -For example, here is the implementation for array equality: - -```rust -impl Eq for [T; N] where T: Eq { - // Test if two arrays have the same elements. - // Because both arrays must have length N, we know their lengths already match. - fn eq(self, other: Self) -> bool { - let mut result = true; - - for i in 0 .. self.len() { - // The T: Eq constraint is needed to call == on the array elements here - result &= self[i] == other[i]; - } - - result - } -} -``` - -Where clauses can also be placed on struct implementations. -For example, here is a method utilizing a generic type that implements the equality trait. - -```rust -struct Foo { - a: u32, - b: T, -} - -impl Foo where T: Eq { - fn eq(self, other: Self) -> bool { - (self.a == other.a) & self.b.eq(other.b) - } -} -``` - -## Generic Traits - -Traits themselves can also be generic by placing the generic arguments after the trait name. These generics are in -scope of every item within the trait. - -```rust -trait Into { - // Convert `self` to type `T` - fn into(self) -> T; -} -``` - -When implementing generic traits the generic arguments of the trait must be specified. This is also true anytime -when referencing a generic trait (e.g. in a `where` clause). - -```rust -struct MyStruct { - array: [Field; 2], -} - -impl Into<[Field; 2]> for MyStruct { - fn into(self) -> [Field; 2] { - self.array - } -} - -fn as_array(x: T) -> [Field; 2] - where T: Into<[Field; 2]> -{ - x.into() -} - -fn main() { - let array = [1, 2]; - let my_struct = MyStruct { array }; - - assert_eq(as_array(my_struct), array); -} -``` - -## Trait Methods With No `self` - -A trait can contain any number of methods, each of which have access to the `Self` type which represents each type -that eventually implements the trait. Similarly, the `self` variable is available as well but is not required to be used. -For example, we can define a trait to create a default value for a type. This trait will need to return the `Self` type -but doesn't need to take any parameters: - -```rust -trait Default { - fn default() -> Self; -} -``` - -Implementing this trait can be done similarly to any other trait: - -```rust -impl Default for Field { - fn default() -> Field { - 0 - } -} - -struct MyType {} - -impl Default for MyType { - fn default() -> Field { - MyType {} - } -} -``` - -However, since there is no `self` parameter, we cannot call it via the method call syntax `object.method()`. -Instead, we'll need to refer to the function directly. This can be done either by referring to the -specific impl `MyType::default()` or referring to the trait itself `Default::default()`. In the later -case, type inference determines the impl that is selected. - -```rust -let my_struct = MyStruct::default(); - -let x: Field = Default::default(); -let result = x + Default::default(); -``` - -:::warning - -```rust -let _ = Default::default(); -``` - -If type inference cannot select which impl to use because of an ambiguous `Self` type, an impl will be -arbitrarily selected. This occurs most often when the result of a trait function call with no parameters -is unused. To avoid this, when calling a trait function with no `self` or `Self` parameters or return type, -always refer to it via the implementation type's namespace - e.g. `MyType::default()`. -This is set to change to an error in future Noir versions. - -::: - -## Default Method Implementations - -A trait can also have default implementations of its methods by giving a body to the desired functions. -Note that this body must be valid for all types that may implement the trait. As a result, the only -valid operations on `self` will be operations valid for any type or other operations on the trait itself. - -```rust -trait Numeric { - fn add(self, other: Self) -> Self; - - // Default implementation of double is (self + self) - fn double(self) -> Self { - self.add(self) - } -} -``` - -When implementing a trait with default functions, a type may choose to implement only the required functions: - -```rust -impl Numeric for Field { - fn add(self, other: Field) -> Field { - self + other - } -} -``` - -Or it may implement the optional methods as well: - -```rust -impl Numeric for u32 { - fn add(self, other: u32) -> u32 { - self + other - } - - fn double(self) -> u32 { - self * 2 - } -} -``` - -## Impl Specialization - -When implementing traits for a generic type it is possible to implement the trait for only a certain combination -of generics. This can be either as an optimization or because those specific generics are required to implement the trait. - -```rust -trait Sub { - fn sub(self, other: Self) -> Self; -} - -struct NonZero { - value: T, -} - -impl Sub for NonZero { - fn sub(self, other: Self) -> Self { - let value = self.value - other.value; - assert(value != 0); - NonZero { value } - } -} -``` - -## Overlapping Implementations - -Overlapping implementations are disallowed by Noir to ensure Noir's decision on which impl to select is never ambiguous. -This means if a trait `Foo` is already implemented -by a type `Bar` for all `T`, then we cannot also have a separate impl for `Bar` (or any other -type argument). Similarly, if there is an impl for all `T` such as `impl Debug for T`, we cannot create -any more impls to `Debug` for other types since it would be ambiguous which impl to choose for any given -method call. - -```rust -trait Trait {} - -// Previous impl defined here -impl Trait for (A, B) {} - -// error: Impl for type `(Field, Field)` overlaps with existing impl -impl Trait for (Field, Field) {} -``` - -## Trait Coherence - -Another restriction on trait implementations is coherence. This restriction ensures other crates cannot create -impls that may overlap with other impls, even if several unrelated crates are used as dependencies in the same -program. - -The coherence restriction is: to implement a trait, either the trait itself or the object type must be declared -in the crate the impl is in. - -In practice this often comes up when using types provided by libraries. If a library provides a type `Foo` that does -not implement a trait in the standard library such as `Default`, you may not `impl Default for Foo` in your own crate. -While restrictive, this prevents later issues or silent changes in the program if the `Foo` library later added its -own impl for `Default`. If you are a user of the `Foo` library in this scenario and need a trait not implemented by the -library your choices are to either submit a patch to the library or use the newtype pattern. - -### The Newtype Pattern - -The newtype pattern gets around the coherence restriction by creating a new wrapper type around the library type -that we cannot create `impl`s for. Since the new wrapper type is defined in our current crate, we can create -impls for any trait we need on it. - -```rust -struct Wrapper { - foo: some_library::Foo, -} - -impl Default for Wrapper { - fn default() -> Wrapper { - Wrapper { - foo: some_library::Foo::new(), - } - } -} -``` - -Since we have an impl for our own type, the behavior of this code will not change even if `some_library` is updated -to provide its own `impl Default for Foo`. The downside of this pattern is that it requires extra wrapping and -unwrapping of values when converting to and from the `Wrapper` and `Foo` types. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/unconstrained.md deleted file mode 100644 index 96f824c5e42..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/unconstrained.md +++ /dev/null @@ -1,99 +0,0 @@ ---- -title: Unconstrained Functions -description: "Learn about what unconstrained functions in Noir are, how to use them and when you'd want to." - -keywords: [Noir programming language, unconstrained, open] -sidebar_position: 5 ---- - -Unconstrained functions are functions which do not constrain any of the included computation and allow for non-deterministic computation. - -## Why? - -Zero-knowledge (ZK) domain-specific languages (DSL) enable developers to generate ZK proofs from their programs by compiling code down to the constraints of an NP complete language (such as R1CS or PLONKish languages). However, the hard bounds of a constraint system can be very limiting to the functionality of a ZK DSL. - -Enabling a circuit language to perform unconstrained execution is a powerful tool. Said another way, unconstrained execution lets developers generate witnesses from code that does not generate any constraints. Being able to execute logic outside of a circuit is critical for both circuit performance and constructing proofs on information that is external to a circuit. - -Fetching information from somewhere external to a circuit can also be used to enable developers to improve circuit efficiency. - -A ZK DSL does not just prove computation, but proves that some computation was handled correctly. Thus, it is necessary that when we switch from performing some operation directly inside of a circuit to inside of an unconstrained environment that the appropriate constraints are still laid down elsewhere in the circuit. - -## Example - -An in depth example might help drive the point home. This example comes from the excellent [post](https://discord.com/channels/1113924620781883405/1124022445054111926/1128747641853972590) by Tom in the Noir Discord. - -Let's look at how we can optimize a function to turn a `u72` into an array of `u8`s. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u72 & 0xff) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 91 -Backend circuit size: 3619 -``` - -A lot of the operations in this function are optimized away by the compiler (all the bit-shifts turn into divisions by constants). However we can save a bunch of gates by casting to u8 a bit earlier. This automatically truncates the bit-shifted value to fit in a u8 which allows us to remove the AND against 0xff. This saves us ~480 gates in total. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 75 -Backend circuit size: 3143 -``` - -Those are some nice savings already but we can do better. This code is all constrained so we're proving every step of calculating out using num, but we don't actually care about how we calculate this, just that it's correct. This is where brillig comes in. - -It turns out that truncating a u72 into a u8 is hard to do inside a snark, each time we do as u8 we lay down 4 ACIR opcodes which get converted into multiple gates. It's actually much easier to calculate num from out than the other way around. All we need to do is multiply each element of out by a constant and add them all together, both relatively easy operations inside a snark. - -We can then run u72_to_u8 as unconstrained brillig code in order to calculate out, then use that result in our constrained function and assert that if we were to do the reverse calculation we'd get back num. This looks a little like the below: - -```rust -fn main(num: u72) -> pub [u8; 8] { - let out = u72_to_u8(num); - - let mut reconstructed_num: u72 = 0; - for i in 0..8 { - reconstructed_num += (out[i] as u72 << (56 - (8 * i))); - } - assert(num == reconstructed_num); - out -} - -unconstrained fn u72_to_u8(num: u72) -> [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8))) as u8; - } - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 78 -Backend circuit size: 2902 -``` - -This ends up taking off another ~250 gates from our circuit! We've ended up with more ACIR opcodes than before but they're easier for the backend to prove (resulting in fewer gates). - -Generally we want to use brillig whenever there's something that's easy to verify but hard to compute within the circuit. For example, if you wanted to calculate a square root of a number it'll be a much better idea to calculate this in brillig and then assert that if you square the result you get back your number. - -## Break and Continue - -In addition to loops over runtime bounds, `break` and `continue` are also available in unconstrained code. See [break and continue](../concepts/control_flow.md#break-and-continue) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/dependencies.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/dependencies.md deleted file mode 100644 index 24e02de08fe..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/dependencies.md +++ /dev/null @@ -1,124 +0,0 @@ ---- -title: Dependencies -description: - Learn how to specify and manage dependencies in Nargo, allowing you to upload packages to GitHub - and use them easily in your project. -keywords: [Nargo, dependencies, GitHub, package management, versioning] -sidebar_position: 1 ---- - -Nargo allows you to upload packages to GitHub and use them as dependencies. - -## Specifying a dependency - -Specifying a dependency requires a tag to a specific commit and the git url to the url containing -the package. - -Currently, there are no requirements on the tag contents. If requirements are added, it would follow -semver 2.0 guidelines. - -> Note: Without a `tag` , there would be no versioning and dependencies would change each time you -> compile your project. - -For example, to add the [ecrecover-noir library](https://github.com/colinnielsen/ecrecover-noir) to your project, add it to `Nargo.toml`: - -```toml -# Nargo.toml - -[dependencies] -ecrecover = {tag = "v0.8.0", git = "https://github.com/colinnielsen/ecrecover-noir"} -``` - -If the module is in a subdirectory, you can define a subdirectory in your git repository, for example: - -```toml -# Nargo.toml - -[dependencies] -easy_private_token_contract = {tag ="v0.1.0-alpha62", git = "https://github.com/AztecProtocol/aztec-packages", directory = "noir-contracts/contracts/easy_private_token_contract"} -``` - -## Specifying a local dependency - -You can also specify dependencies that are local to your machine. - -For example, this file structure has a library and binary crate - -```tree -├── binary_crate -│   ├── Nargo.toml -│   └── src -│   └── main.nr -└── lib_a - ├── Nargo.toml - └── src - └── lib.nr -``` - -Inside of the binary crate, you can specify: - -```toml -# Nargo.toml - -[dependencies] -lib_a = { path = "../lib_a" } -``` - -## Importing dependencies - -You can import a dependency to a Noir file using the following syntax. For example, to import the -ecrecover-noir library and local lib_a referenced above: - -```rust -use ecrecover; -use lib_a; -``` - -You can also import only the specific parts of dependency that you want to use, like so: - -```rust -use std::hash::sha256; -use std::scalar_mul::fixed_base_embedded_curve; -``` - -Lastly, as demonstrated in the -[elliptic curve example](../standard_library/cryptographic_primitives/ec_primitives.md#examples), you -can import multiple items in the same line by enclosing them in curly braces: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; -``` - -We don't have a way to consume libraries from inside a [workspace](./workspaces.md) as external dependencies right now. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -## Dependencies of Dependencies - -Note that when you import a dependency, you also get access to all of the dependencies of that package. - -For example, the [phy_vector](https://github.com/resurgencelabs/phy_vector) library imports an [fraction](https://github.com/resurgencelabs/fraction) library. If you're importing the phy_vector library, then you can access the functions in fractions library like so: - -```rust -use phy_vector; - -fn main(x : Field, y : pub Field) { - //... - let f = phy_vector::fraction::toFraction(true, 2, 1); - //... -} -``` - -## Available Libraries - -Noir does not currently have an official package manager. You can find a list of available Noir libraries in the [awesome-noir repo here](https://github.com/noir-lang/awesome-noir#libraries). - -Some libraries that are available today include: - -- [Standard Library](https://github.com/noir-lang/noir/tree/master/noir_stdlib) - the Noir Standard Library -- [Ethereum Storage Proof Verification](https://github.com/aragonzkresearch/noir-trie-proofs) - a library that contains the primitives necessary for RLP decoding (in the form of look-up table construction) and Ethereum state and storage proof verification (or verification of any trie proof involving 32-byte long keys) -- [BigInt](https://github.com/shuklaayush/noir-bigint) - a library that provides a custom BigUint56 data type, allowing for computations on large unsigned integers -- [ECrecover](https://github.com/colinnielsen/ecrecover-noir/tree/main) - a library to verify an ECDSA signature and return the source Ethereum address -- [Sparse Merkle Tree Verifier](https://github.com/vocdoni/smtverifier-noir/tree/main) - a library for verification of sparse Merkle trees -- [Signed Int](https://github.com/resurgencelabs/signed_int) - a library for accessing a custom Signed Integer data type, allowing access to negative numbers on Noir -- [Fraction](https://github.com/resurgencelabs/fraction) - a library for accessing fractional number data type in Noir, allowing results that aren't whole numbers diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/modules.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/modules.md deleted file mode 100644 index 16b6307d2fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/modules.md +++ /dev/null @@ -1,185 +0,0 @@ ---- -title: Modules -description: - Learn how to organize your files using modules in Noir, following the same convention as Rust's - module system. Examples included. -keywords: [Noir, Rust, modules, organizing files, sub-modules] -sidebar_position: 2 ---- - -Noir's module system follows the same convention as the _newer_ version of Rust's module system. - -## Purpose of Modules - -Modules are used to organize files. Without modules all of your code would need to live in a single -file. In Noir, the compiler does not automatically scan all of your files to detect modules. This -must be done explicitly by the developer. - -## Examples - -### Importing a module in the crate root - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo.nr` - -```rust -fn from_foo() {} -``` - -In the above snippet, the crate root is the `src/main.nr` file. The compiler sees the module -declaration `mod foo` which prompts it to look for a foo.nr file. - -Visually this module hierarchy looks like the following : - -``` -crate - ├── main - │ - └── foo - └── from_foo - -``` - -The module filename may also be the name of the module as a directory with the contents in a -file named `mod.nr` within that directory. The above example can alternatively be expressed like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -fn from_foo() {} -``` - -Note that it's an error to have both files `src/foo.nr` and `src/foo/mod.nr` in the filesystem. - -### Importing a module throughout the tree - -All modules are accessible from the `crate::` namespace. - -``` -crate - ├── bar - ├── foo - └── main - -``` - -In the above snippet, if `bar` would like to use functions in `foo`, it can do so by `use crate::foo::function_name`. - -### Sub-modules - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -fn from_bar() {} -``` - -In the above snippet, we have added an extra module to the module tree; `bar`. `bar` is a submodule -of `foo` hence we declare bar in `foo.nr` with `mod bar`. Since `foo` is not the crate root, the -compiler looks for the file associated with the `bar` module in `src/foo/bar.nr` - -Visually the module hierarchy looks as follows: - -``` -crate - ├── main - │ - └── foo - ├── from_foo - └── bar - └── from_bar -``` - -Similar to importing a module in the crate root, modules can be placed in a `mod.nr` file, like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar/mod.nr` - -```rust -fn from_bar() {} -``` - -### Referencing a parent module - -Given a submodule, you can refer to its parent module using the `super` keyword. - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; - -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -// Same as bar::from_foo -use super::from_foo; - -fn from_bar() { - from_foo(); // invokes super::from_foo(), which is bar::from_foo() - super::from_foo(); // also invokes bar::from_foo() -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/bigint.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/bigint.md deleted file mode 100644 index 2bfdeec6631..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/bigint.md +++ /dev/null @@ -1,122 +0,0 @@ ---- -title: Big Integers -description: How to use big integers from Noir standard library -keywords: - [ - Big Integer, - Noir programming language, - Noir libraries, - ] ---- - -The BigInt module in the standard library exposes some class of integers which do not fit (well) into a Noir native field. It implements modulo arithmetic, modulo a 'big' prime number. - -:::note - -The module can currently be considered as `Field`s with fixed modulo sizes used by a set of elliptic curves, in addition to just the native curve. [More work](https://github.com/noir-lang/noir/issues/510) is needed to achieve arbitrarily sized big integers. - -::: - -Currently 6 classes of integers (i.e 'big' prime numbers) are available in the module, namely: - -- BN254 Fq: Bn254Fq -- BN254 Fr: Bn254Fr -- Secp256k1 Fq: Secpk1Fq -- Secp256k1 Fr: Secpk1Fr -- Secp256r1 Fr: Secpr1Fr -- Secp256r1 Fq: Secpr1Fq - -Where XXX Fq and XXX Fr denote respectively the order of the base and scalar field of the (usual) elliptic curve XXX. -For instance the big integer 'Secpk1Fq' in the standard library refers to integers modulo $2^{256}-2^{32}-977$. - -Feel free to explore the source code for the other primes: - -```rust title="big_int_definition" showLineNumbers -struct BigInt { - pointer: u32, - modulus: u32, -} -``` -> Source code: noir_stdlib/src/bigint.nr#L14-L19 - - -## Example usage - -A common use-case is when constructing a big integer from its bytes representation, and performing arithmetic operations on it: - -```rust title="big_int_example" showLineNumbers -fn big_int_example(x: u8, y: u8) { - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - let b = Secpk1Fq::from_le_bytes(&[y, x, 9]); - let c = (a + b) * b / a; - let d = c.to_le_bytes(); - println(d[0]); -} -``` -> Source code: test_programs/execution_success/bigint/src/main.nr#L70-L78 - - -## Methods - -The available operations for each big integer are: - -### from_le_bytes - -Construct a big integer from its little-endian bytes representation. Example: - -```rust - // Construct a big integer from a slice of bytes - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - // Construct a big integer from an array of 32 bytes - let a = Secpk1Fq::from_le_bytes_32([1;32]); - ``` - -Sure, here's the formatted version of the remaining methods: - -### to_le_bytes - -Return the little-endian bytes representation of a big integer. Example: - -```rust -let bytes = a.to_le_bytes(); -``` - -### add - -Add two big integers. Example: - -```rust -let sum = a + b; -``` - -### sub - -Subtract two big integers. Example: - -```rust -let difference = a - b; -``` - -### mul - -Multiply two big integers. Example: - -```rust -let product = a * b; -``` - -### div - -Divide two big integers. Note that division is field division and not euclidean division. Example: - -```rust -let quotient = a / b; -``` - -### eq - -Compare two big integers. Example: - -```rust -let are_equal = a == b; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/boundedvec.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/boundedvec.md deleted file mode 100644 index 604d84d5ba4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/boundedvec.md +++ /dev/null @@ -1,419 +0,0 @@ ---- -title: Bounded Vectors -keywords: [noir, vector, bounded vector, slice] -sidebar_position: 1 ---- - -A `BoundedVec` is a growable storage similar to a `Vec` except that it -is bounded with a maximum possible length. Unlike `Vec`, `BoundedVec` is not implemented -via slices and thus is not subject to the same restrictions slices are (notably, nested -slices - and thus nested vectors as well - are disallowed). - -Since a BoundedVec is backed by a normal array under the hood, growing the BoundedVec by -pushing an additional element is also more efficient - the length only needs to be increased -by one. - -For these reasons `BoundedVec` should generally be preferred over `Vec` when there -is a reasonable maximum bound that can be placed on the vector. - -Example: - -```rust -let mut vector: BoundedVec = BoundedVec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -assert(vector.max_len() == 10); -``` - -## Methods - -### new - -```rust -pub fn new() -> Self -``` - -Creates a new, empty vector of length zero. - -Since this container is backed by an array internally, it still needs an initial value -to give each element. To resolve this, each element is zeroed internally. This value -is guaranteed to be inaccessible unless `get_unchecked` is used. - -Example: - -```rust -let empty_vector: BoundedVec = BoundedVec::new(); -assert(empty_vector.len() == 0); -``` - -Note that whenever calling `new` the maximum length of the vector should always be specified -via a type signature: - -```rust title="new_example" showLineNumbers -fn foo() -> BoundedVec { - // Ok! MaxLen is specified with a type annotation - let v1: BoundedVec = BoundedVec::new(); - let v2 = BoundedVec::new(); - - // Ok! MaxLen is known from the type of foo's return value - v2 -} - -fn bad() { - let mut v3 = BoundedVec::new(); - - // Not Ok! We don't know if v3's MaxLen is at least 1, and the compiler often infers 0 by default. - v3.push(5); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L11-L27 - - -This defaulting of `MaxLen` (and numeric generics in general) to zero may change in future noir versions -but for now make sure to use type annotations when using bounded vectors. Otherwise, you will receive a constraint failure at runtime when the vec is pushed to. - -### get - -```rust -pub fn get(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this -will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - let last = v.get(v.len() - 1); - assert(first != last); -} -``` - -### get_unchecked - -```rust -pub fn get_unchecked(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero, without -performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, -it is unsafe! Use at your own risk! - -Example: - -```rust title="get_unchecked_example" showLineNumbers -fn sum_of_first_three(v: BoundedVec) -> u32 { - // Always ensure the length is larger than the largest - // index passed to get_unchecked - assert(v.len() > 2); - let first = v.get_unchecked(0); - let second = v.get_unchecked(1); - let third = v.get_unchecked(2); - first + second + third -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L54-L64 - - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - assert(first != 42); - v.set(0, 42); - let new_first = v.get(0); - assert(new_first == 42); -} -``` - -### set_unchecked - -```rust -pub fn set_unchecked(&mut self: Self, index: u64, value: T) -> T { -``` - -Writes an element to the vector at the given index, starting from zero, without performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, it is unsafe! Use at your own risk! - -Example: - -```rust title="set_unchecked_example" showLineNumbers -fn set_unchecked_example() { - let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([1, 2]); - - // Here we're safely writing within the valid range of `vec` - // `vec` now has the value [42, 2] - vec.set_unchecked(0, 42); - - // We can then safely read this value back out of `vec`. - // Notice that we use the checked version of `get` which would prevent reading unsafe values. - assert_eq(vec.get(0), 42); - - // We've now written past the end of `vec`. - // As this index is still within the maximum potential length of `v`, - // it won't cause a constraint failure. - vec.set_unchecked(2, 42); - println(vec); - - // This will write past the end of the maximum potential length of `vec`, - // it will then trigger a constraint failure. - vec.set_unchecked(5, 42); - println(vec); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L67-L91 - - - -### push - -```rust -pub fn push(&mut self, elem: T) { -``` - -Pushes an element to the end of the vector. This increases the length -of the vector by one. - -Panics if the new length of the vector will be greater than the max length. - -Example: - -```rust title="bounded-vec-push-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - v.push(1); - v.push(2); - - // Panics with failed assertion "push out of bounds" - v.push(3); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L95-L103 - - -### pop - -```rust -pub fn pop(&mut self) -> T -``` - -Pops the element at the end of the vector. This will decrease the length -of the vector by one. - -Panics if the vector is empty. - -Example: - -```rust title="bounded-vec-pop-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.push(1); - v.push(2); - - let two = v.pop(); - let one = v.pop(); - - assert(two == 2); - assert(one == 1); - // error: cannot pop from an empty vector - // let _ = v.pop(); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L108-L120 - - -### len - -```rust -pub fn len(self) -> u64 { -``` - -Returns the current length of this vector - -Example: - -```rust title="bounded-vec-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - assert(v.len() == 0); - - v.push(100); - assert(v.len() == 1); - - v.push(200); - v.push(300); - v.push(400); - assert(v.len() == 4); - - let _ = v.pop(); - let _ = v.pop(); - assert(v.len() == 2); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L125-L140 - - -### max_len - -```rust -pub fn max_len(_self: BoundedVec) -> u64 { -``` - -Returns the maximum length of this vector. This is always -equal to the `MaxLen` parameter this vector was initialized with. - -Example: - -```rust title="bounded-vec-max-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.max_len() == 5); - v.push(10); - assert(v.max_len() == 5); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L145-L151 - - -### storage - -```rust -pub fn storage(self) -> [T; MaxLen] { -``` - -Returns the internal array within this vector. -Since arrays in Noir are immutable, mutating the returned storage array will not mutate -the storage held internally by this vector. - -Note that uninitialized elements may be zeroed out! - -Example: - -```rust title="bounded-vec-storage-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.storage() == [0, 0, 0, 0, 0]); - - v.push(57); - assert(v.storage() == [57, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L156-L163 - - -### extend_from_array - -```rust -pub fn extend_from_array(&mut self, array: [T; Len]) -``` - -Pushes each element from the given array to this vector. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-array-example" showLineNumbers -let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([2, 4]); - - assert(vec.len == 2); - assert(vec.get(0) == 2); - assert(vec.get(1) == 4); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L168-L175 - - -### extend_from_bounded_vec - -```rust -pub fn extend_from_bounded_vec(&mut self, vec: BoundedVec) -``` - -Pushes each element from the other vector to this vector. The length of -the other vector is left unchanged. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-bounded-vec-example" showLineNumbers -let mut v1: BoundedVec = BoundedVec::new(); - let mut v2: BoundedVec = BoundedVec::new(); - - v2.extend_from_array([1, 2, 3]); - v1.extend_from_bounded_vec(v2); - - assert(v1.storage() == [1, 2, 3, 0, 0]); - assert(v2.storage() == [1, 2, 3, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L180-L189 - - -### from_array - -```rust -pub fn from_array(array: [T; Len]) -> Self -``` - -Creates a new vector, populating it with values derived from an array input. -The maximum length of the vector is determined based on the type signature. - -Example: -```rust -let bounded_vec: BoundedVec = BoundedVec::from_array([1, 2, 3]) -``` - -### map - -```rust -pub fn map(self, f: fn[Env](T) -> U) -> BoundedVec -``` - -Creates a new vector of equal size by calling a closure on each element in this vector. - -Example: - -```rust title="bounded-vec-map-example" showLineNumbers -let vec: BoundedVec = BoundedVec::from_array([1, 2, 3, 4]); - let result = vec.map(|value| value * 2); -``` -> Source code: noir_stdlib/src/collections/bounded_vec.nr#L205-L208 - - -### any - -```rust -pub fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -Returns true if the given predicate returns true for any element -in this vector. - -Example: - -```rust title="bounded-vec-any-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.extend_from_array([2, 4, 6]); - - let all_even = !v.any(|elem: u32| elem % 2 != 0); - assert(all_even); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L256-L262 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/hashmap.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/hashmap.md deleted file mode 100644 index 8c50c7e774c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/hashmap.md +++ /dev/null @@ -1,570 +0,0 @@ ---- -title: HashMap -keywords: [noir, map, hash, hashmap] -sidebar_position: 1 ---- - -`HashMap` is used to efficiently store and look up key-value pairs. - -`HashMap` is a bounded type which can store anywhere from zero to `MaxLen` total elements. -Note that due to hash collisions, the actual maximum number of elements stored by any particular -hashmap is likely lower than `MaxLen`. This is true even with cryptographic hash functions since -every hash value will be performed modulo `MaxLen`. - -When creating `HashMap`s, the `MaxLen` generic should always be specified if it is not already -known. Otherwise, the compiler may infer a different value for `MaxLen` (such as zero), which -will likely change the result of the program. This behavior is set to become an error in future -versions instead. - -Example: - -```rust -// Create a mapping from Fields to u32s with a maximum length of 12 -// using a poseidon2 hasher -use std::hash::poseidon2::Poseidon2Hasher; -let mut map: HashMap> = HashMap::default(); - -map.insert(1, 2); -map.insert(3, 4); - -let two = map.get(1).unwrap(); -``` - -## Methods - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default -{ - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L462-L469 - - -Creates a fresh, empty HashMap. - -When using this function, always make sure to specify the maximum size of the hash map. - -This is the same `default` from the `Default` implementation given further below. It is -repeated here for convenience since it is the recommended way to create a hashmap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L204 - - -Because `HashMap` has so many generic arguments that are likely to be the same throughout -your program, it may be helpful to create a type alias: - -```rust title="type_alias" showLineNumbers -type MyMap = HashMap>; -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L195-L197 - - -### with_hasher - -```rust title="with_hasher" showLineNumbers -pub fn with_hasher(_build_hasher: B) -> Self - where - B: BuildHasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L82-L86 - - -Creates a hashmap with an existing `BuildHasher`. This can be used to ensure multiple -hashmaps are created with the same hasher instance. - -Example: - -```rust title="with_hasher_example" showLineNumbers -let my_hasher: BuildHasherDefault = Default::default(); - let hashmap: HashMap> = HashMap::with_hasher(my_hasher); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L206-L210 - - -### get - -```rust title="get" showLineNumbers -pub fn get( - self, - key: K - ) -> Option - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L278-L287 - - -Retrieves a value from the hashmap, returning `Option::none()` if it was not found. - -Example: - -```rust title="get_example" showLineNumbers -fn get_example(map: HashMap>) { - let x = map.get(12); - - if x.is_some() { - assert(x.unwrap() == 42); - } -} -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L298-L306 - - -### insert - -```rust title="insert" showLineNumbers -pub fn insert( - &mut self, - key: K, - value: V - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L313-L323 - - -Inserts a new key-value pair into the map. If the key was already in the map, its -previous value will be overridden with the newly provided one. - -Example: - -```rust title="insert_example" showLineNumbers -let mut map: HashMap> = HashMap::default(); - map.insert(12, 42); - assert(map.len() == 1); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L212-L216 - - -### remove - -```rust title="remove" showLineNumbers -pub fn remove( - &mut self, - key: K - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L356-L365 - - -Removes the given key-value pair from the map. If the key was not already present -in the map, this does nothing. - -Example: - -```rust title="remove_example" showLineNumbers -map.remove(12); - assert(map.is_empty()); - - // If a key was not present in the map, remove does nothing - map.remove(12); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L220-L227 - - -### is_empty - -```rust title="is_empty" showLineNumbers -pub fn is_empty(self) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L115-L117 - - -True if the length of the hash map is empty. - -Example: - -```rust title="is_empty_example" showLineNumbers -assert(map.is_empty()); - - map.insert(1, 2); - assert(!map.is_empty()); - - map.remove(1); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L229-L237 - - -### len - -```rust title="len" showLineNumbers -pub fn len(self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L264-L266 - - -Returns the current length of this hash map. - -Example: - -```rust title="len_example" showLineNumbers -// This is equivalent to checking map.is_empty() - assert(map.len() == 0); - - map.insert(1, 2); - map.insert(3, 4); - map.insert(5, 6); - assert(map.len() == 3); - - // 3 was already present as a key in the hash map, so the length is unchanged - map.insert(3, 7); - assert(map.len() == 3); - - map.remove(1); - assert(map.len() == 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L239-L254 - - -### capacity - -```rust title="capacity" showLineNumbers -pub fn capacity(_self: Self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L271-L273 - - -Returns the maximum capacity of this hashmap. This is always equal to the capacity -specified in the hashmap's type. - -Unlike hashmaps in general purpose programming languages, hashmaps in Noir have a -static capacity that does not increase as the map grows larger. Thus, this capacity -is also the maximum possible element count that can be inserted into the hashmap. -Due to hash collisions (modulo the hashmap length), it is likely the actual maximum -element count will be lower than the full capacity. - -Example: - -```rust title="capacity_example" showLineNumbers -let empty_map: HashMap> = HashMap::default(); - assert(empty_map.len() == 0); - assert(empty_map.capacity() == 42); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L256-L260 - - -### clear - -```rust title="clear" showLineNumbers -pub fn clear(&mut self) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L93-L95 - - -Clears the hashmap, removing all key-value pairs from it. - -Example: - -```rust title="clear_example" showLineNumbers -assert(!map.is_empty()); - map.clear(); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L262-L266 - - -### contains_key - -```rust title="contains_key" showLineNumbers -pub fn contains_key( - self, - key: K - ) -> bool - where - K: Hash + Eq, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L101-L110 - - -True if the hashmap contains the given key. Unlike `get`, this will not also return -the value associated with the key. - -Example: - -```rust title="contains_key_example" showLineNumbers -if map.contains_key(7) { - let value = map.get(7); - assert(value.is_some()); - } else { - println("No value for key 7!"); - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L268-L275 - - -### entries - -```rust title="entries" showLineNumbers -pub fn entries(self) -> BoundedVec<(K, V), N> { -``` -> Source code: noir_stdlib/src/collections/map.nr#L123-L125 - - -Returns a vector of each key-value pair present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="entries_example" showLineNumbers -let entries = map.entries(); - - // The length of a hashmap may not be compile-time known, so we - // need to loop over its capacity instead - for i in 0..map.capacity() { - if i < entries.len() { - let (key, value) = entries.get(i); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L309-L320 - - -### keys - -```rust title="keys" showLineNumbers -pub fn keys(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L144-L146 - - -Returns a vector of each key present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="keys_example" showLineNumbers -let keys = map.keys(); - - for i in 0..keys.max_len() { - if i < keys.len() { - let key = keys.get_unchecked(i); - let value = map.get(key).unwrap_unchecked(); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L322-L332 - - -### values - -```rust title="values" showLineNumbers -pub fn values(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L164-L166 - - -Returns a vector of each value present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="values_example" showLineNumbers -let values = map.values(); - - for i in 0..values.max_len() { - if i < values.len() { - let value = values.get_unchecked(i); - println(f"Found value {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L334-L343 - - -### iter_mut - -```rust title="iter_mut" showLineNumbers -pub fn iter_mut( - &mut self, - f: fn(K, V) -> (K, V) - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L183-L192 - - -Iterates through each key-value pair of the HashMap, setting each key-value pair to the -result returned from the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If this is not desired, use `iter_values_mut` if only values need to be mutated, -or `entries` if neither keys nor values need to be mutated. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_mut_example" showLineNumbers -// Add 1 to each key in the map, and double the value associated with that key. - map.iter_mut(|k, v| (k + 1, v * 2)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L347-L350 - - -### iter_keys_mut - -```rust title="iter_keys_mut" showLineNumbers -pub fn iter_keys_mut( - &mut self, - f: fn(K) -> K - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L208-L217 - - -Iterates through the HashMap, mutating each key to the result returned from -the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If only iteration is desired and the keys are not intended to be mutated, -prefer using `entries` instead. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_keys_mut_example" showLineNumbers -// Double each key, leaving the value associated with that key untouched - map.iter_keys_mut(|k| k * 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L352-L355 - - -### iter_values_mut - -```rust title="iter_values_mut" showLineNumbers -pub fn iter_values_mut(&mut self, f: fn(V) -> V) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L233-L235 - - -Iterates through the HashMap, applying the given function to each value and mutating the -value to equal the result. This function is more efficient than `iter_mut` and `iter_keys_mut` -because the keys are untouched and the underlying hashmap thus does not need to be reordered. - -Example: - -```rust title="iter_values_mut_example" showLineNumbers -// Halve each value - map.iter_values_mut(|v| v / 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L357-L360 - - -### retain - -```rust title="retain" showLineNumbers -pub fn retain(&mut self, f: fn(K, V) -> bool) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L247-L249 - - -Retains only the key-value pairs for which the given function returns true. -Any key-value pairs for which the function returns false will be removed from the map. - -Example: - -```rust title="retain_example" showLineNumbers -map.retain(|k, v| (k != 0) & (v != 0)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L280-L282 - - -## Trait Implementations - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default -{ - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L462-L469 - - -Constructs an empty HashMap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L204 - - -### eq - -```rust title="eq" showLineNumbers -impl Eq for HashMap -where - K: Eq + Hash, - V: Eq, - B: BuildHasher, - H: Hasher -{ - fn eq(self, other: HashMap) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L426-L435 - - -Checks if two HashMaps are equal. - -Example: - -```rust title="eq_example" showLineNumbers -let mut map1: HashMap> = HashMap::default(); - let mut map2: HashMap> = HashMap::default(); - - map1.insert(1, 2); - map1.insert(3, 4); - - map2.insert(3, 4); - map2.insert(1, 2); - - assert(map1 == map2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L284-L295 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/ec_primitives.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/ec_primitives.md deleted file mode 100644 index f262d8160d6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/ec_primitives.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Elliptic Curve Primitives -keywords: [cryptographic primitives, Noir project] -sidebar_position: 4 ---- - -Data structures and methods on them that allow you to carry out computations involving elliptic -curves over the (mathematical) field corresponding to `Field`. For the field currently at our -disposal, applications would involve a curve embedded in BN254, e.g. the -[Baby Jubjub curve](https://eips.ethereum.org/EIPS/eip-2494). - -## Data structures - -### Elliptic curve configurations - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Curve`), i.e. the specific elliptic -curve you want to use, which would be specified using any one of the methods -`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::new` which take the coefficients in the -defining equation together with a generator point as parameters. You can find more detail in the -comments in -[`noir_stdlib/src/ec/mod.nr`](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr), but -the gist of it is that the elliptic curves of interest are usually expressed in one of the standard -forms implemented here (Twisted Edwards, Montgomery and Short Weierstraß), and in addition to that, -you could choose to use `affine` coordinates (Cartesian coordinates - the usual (x,y) - possibly -together with a point at infinity) or `curvegroup` coordinates (some form of projective coordinates -requiring more coordinates but allowing for more efficient implementations of elliptic curve -operations). Conversions between all of these forms are provided, and under the hood these -conversions are done whenever an operation is more efficient in a different representation (or a -mixed coordinate representation is employed). - -### Points - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Point`), i.e. points lying on the -elliptic curve. For a curve configuration `c` and a point `p`, it may be checked that `p` -does indeed lie on `c` by calling `c.contains(p1)`. - -## Methods - -(given a choice of curve representation, e.g. use `std::ec::tecurve::affine::Curve` and use -`std::ec::tecurve::affine::Point`) - -- The **zero element** is given by `Point::zero()`, and we can verify whether a point `p: Point` is - zero by calling `p.is_zero()`. -- **Equality**: Points `p1: Point` and `p2: Point` may be checked for equality by calling - `p1.eq(p2)`. -- **Addition**: For `c: Curve` and points `p1: Point` and `p2: Point` on the curve, adding these two - points is accomplished by calling `c.add(p1,p2)`. -- **Negation**: For a point `p: Point`, `p.negate()` is its negation. -- **Subtraction**: For `c` and `p1`, `p2` as above, subtracting `p2` from `p1` is accomplished by - calling `c.subtract(p1,p2)`. -- **Scalar multiplication**: For `c` as above, `p: Point` a point on the curve and `n: Field`, - scalar multiplication is given by `c.mul(n,p)`. If instead `n :: [u1; N]`, i.e. `n` is a bit - array, the `bit_mul` method may be used instead: `c.bit_mul(n,p)` -- **Multi-scalar multiplication**: For `c` as above and arrays `n: [Field; N]` and `p: [Point; N]`, - multi-scalar multiplication is given by `c.msm(n,p)`. -- **Coordinate representation conversions**: The `into_group` method converts a point or curve - configuration in the affine representation to one in the CurveGroup representation, and - `into_affine` goes in the other direction. -- **Curve representation conversions**: `tecurve` and `montcurve` curves and points are equivalent - and may be converted between one another by calling `into_montcurve` or `into_tecurve` on their - configurations or points. `swcurve` is more general and a curve c of one of the other two types - may be converted to this representation by calling `c.into_swcurve()`, whereas a point `p` lying - on the curve given by `c` may be mapped to its corresponding `swcurve` point by calling - `c.map_into_swcurve(p)`. -- **Map-to-curve methods**: The Elligator 2 method of mapping a field element `n: Field` into a - `tecurve` or `montcurve` with configuration `c` may be called as `c.elligator2_map(n)`. For all of - the curve configurations, the SWU map-to-curve method may be called as `c.swu_map(z,n)`, where - `z: Field` depends on `Field` and `c` and must be chosen by the user (the conditions it needs to - satisfy are specified in the comments - [here](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr)). - -## Examples - -The -[ec_baby_jubjub test](https://github.com/noir-lang/noir/blob/master/test_programs/compile_success_empty/ec_baby_jubjub/src/main.nr) -illustrates all of the above primitives on various forms of the Baby Jubjub curve. A couple of more -interesting examples in Noir would be: - -Public-key cryptography: Given an elliptic curve and a 'base point' on it, determine the public key -from the private key. This is a matter of using scalar multiplication. In the case of Baby Jubjub, -for example, this code would do: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; - -fn bjj_pub_key(priv_key: Field) -> Point -{ - - let bjj = Curve::new(168700, 168696, G::new(995203441582195749578291179787384436505546430278305826713579947235728471134,5472060717959818805561601436314318772137091100104008585924551046643952123905)); - - let base_pt = Point::new(5299619240641551281634865583518297030282874472190772894086521144482721001553, 16950150798460657717958625567821834550301663161624707787222815936182638968203); - - bjj.mul(priv_key,base_pt) -} -``` - -This would come in handy in a Merkle proof. - -- EdDSA signature verification: This is a matter of combining these primitives with a suitable hash - function. See - [feat(stdlib): EdDSA sig verification noir#1136](https://github.com/noir-lang/noir/pull/1136) for - the case of Baby Jubjub and the Poseidon hash function. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx deleted file mode 100644 index 4c22e70e8de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: ECDSA Signature Verification -description: Learn about the cryptographic primitives regarding ECDSA over the secp256k1 and secp256r1 curves -keywords: [cryptographic primitives, Noir project, ecdsa, secp256k1, secp256r1, signatures] -sidebar_position: 3 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports ECDSA signatures verification over the secp256k1 and secp256r1 curves. - -## ecdsa_secp256k1::verify_signature - -Verifier for ECDSA Secp256k1 signatures. -See ecdsa_secp256k1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256k1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256k1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256k1::verify_signature_slice - -Verifier for ECDSA Secp256k1 signatures where the message is a slice. - -```rust title="ecdsa_secp256k1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L13-L20 - - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures. -See ecdsa_secp256r1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256r1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256r1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures where the message is a slice. - -```rust title="ecdsa_secp256r1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/eddsa.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/eddsa.mdx deleted file mode 100644 index ef4386052eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/eddsa.mdx +++ /dev/null @@ -1,37 +0,0 @@ ---- -title: EdDSA Verification -description: Learn about the cryptographic primitives regarding EdDSA -keywords: [cryptographic primitives, Noir project, eddsa, signatures] -sidebar_position: 5 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## eddsa::eddsa_poseidon_verify - -Verifier for EdDSA signatures - -```rust -fn eddsa_poseidon_verify(public_key_x : Field, public_key_y : Field, signature_s: Field, signature_r8_x: Field, signature_r8_y: Field, message: Field) -> bool -``` - -It is also possible to specify the hash algorithm used for the signature by using the `eddsa_verify` function by passing a type implementing the Hasher trait with the turbofish operator. -For instance, if you want to use Poseidon2 instead, you can do the following: -```rust -use std::hash::poseidon2::Poseidon2Hasher; - -eddsa_verify::(pub_key_a.x, pub_key_a.y, s_a, r8_a.x, r8_a.y, msg); -``` - - - -## eddsa::eddsa_to_pub - -Private to public key conversion. - -Returns `(pub_key_x, pub_key_y)` - -```rust -fn eddsa_to_pub(secret : Field) -> (Field, Field) -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx deleted file mode 100644 index 68d033e9d60..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: Scalar multiplication -description: See how you can perform scalar multiplication in Noir -keywords: [cryptographic primitives, Noir project, scalar multiplication] -sidebar_position: 1 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -The following functions perform operations over the embedded curve whose coordinates are defined by the configured noir field. -For the BN254 scalar field, this is BabyJubJub or Grumpkin. - -:::note -Suffixes `_low` and `_high` denote low and high limbs of a scalar. -::: - -## embedded_curve_ops::multi_scalar_mul - -Performs multi scalar multiplication over the embedded curve. -The function accepts arbitrary amount of point-scalar pairs on the input, it multiplies the individual pairs over -the curve and returns a sum of the resulting points. - -Points represented as x and y coordinates [x1, y1, x2, y2, ...], scalars as low and high limbs [low1, high1, low2, high2, ...]. - -```rust title="multi_scalar_mul" showLineNumbers -pub fn multi_scalar_mul( - points: [EmbeddedCurvePoint; N], - scalars: [EmbeddedCurveScalar; N] -) -> [Field; 3] -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L92-L97 - - -example - -```rust -fn main(point_x: Field, point_y: Field, scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::multi_scalar_mul([point_x, point_y], [scalar_low, scalar_high]); - println(point); -} -``` - -## embedded_curve_ops::fixed_base_scalar_mul - -Performs fixed base scalar multiplication over the embedded curve (multiplies input scalar with a generator point). -The function accepts a single scalar on the input represented as 2 fields. - -```rust title="fixed_base_scalar_mul" showLineNumbers -pub fn fixed_base_scalar_mul( - scalar_low: Field, - scalar_high: Field -) -> [Field; 3] -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L103-L108 - - -example - -```rust -fn main(scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::fixed_base_scalar_mul(scalar_low, scalar_high); - println(point); -} -``` - -## embedded_curve_ops::embedded_curve_add - -Adds two points on the embedded curve. -This function takes two `EmbeddedCurvePoint` structures as parameters, representing points on the curve, and returns a new `EmbeddedCurvePoint` structure that represents their sum. - -### Parameters: -- `point1` (`EmbeddedCurvePoint`): The first point to add. -- `point2` (`EmbeddedCurvePoint`): The second point to add. - -### Returns: -- `EmbeddedCurvePoint`: The resulting point after the addition of `point1` and `point2`. - -```rust title="embedded_curve_add" showLineNumbers -fn embedded_curve_add( - point1: EmbeddedCurvePoint, - point2: EmbeddedCurvePoint -) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L117-L122 - - -example - -```rust -fn main() { - let point1 = EmbeddedCurvePoint { x: 1, y: 2 }; - let point2 = EmbeddedCurvePoint { x: 3, y: 4 }; - let result = std::embedded_curve_ops::embedded_curve_add(point1, point2); - println!("Resulting Point: ({}, {})", result.x, result.y); -} -``` - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/hashes.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/hashes.mdx deleted file mode 100644 index ddcfbb2175f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/hashes.mdx +++ /dev/null @@ -1,253 +0,0 @@ ---- -title: Hash methods -description: - Learn about the cryptographic primitives ready to use for any Noir project, including sha256, - blake2s, pedersen, mimc_bn254 and mimc -keywords: - [cryptographic primitives, Noir project, sha256, blake2s, pedersen, mimc_bn254, mimc, hash] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## sha256 - -Given an array of bytes, returns the resulting sha256 hash. -Specify a message_size to hash only the first `message_size` bytes of the input. - -```rust title="sha256" showLineNumbers -pub fn sha256(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L13-L15 - - -example: -```rust title="sha256_var" showLineNumbers -let digest = std::hash::sha256_var([x as u8], 1); -``` -> Source code: test_programs/execution_success/sha256/src/main.nr#L16-L18 - - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::sha256::sha256_var(x, 4); -} -``` - - - - -## blake2s - -Given an array of bytes, returns an array with the Blake2 hash - -```rust title="blake2s" showLineNumbers -pub fn blake2s(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L19-L21 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake2s(x); -} -``` - - - -## blake3 - -Given an array of bytes, returns an array with the Blake3 hash - -```rust title="blake3" showLineNumbers -pub fn blake3(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L25-L27 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake3(x); -} -``` - - - -## pedersen_hash - -Given an array of Fields, returns the Pedersen hash. - -```rust title="pedersen_hash" showLineNumbers -pub fn pedersen_hash(input: [Field; N]) -> Field -``` -> Source code: noir_stdlib/src/hash/mod.nr#L61-L63 - - -example: - -```rust title="pedersen-hash" showLineNumbers -fn main(x: Field, y: Field, expected_hash: Field) { - let hash = std::hash::pedersen_hash([x, y]); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/pedersen_hash/src/main.nr#L1-L7 - - - - -## pedersen_commitment - -Given an array of Fields, returns the Pedersen commitment. - -```rust title="pedersen_commitment" showLineNumbers -pub fn pedersen_commitment(input: [Field; N]) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/hash/mod.nr#L30-L32 - - -example: - -```rust title="pedersen-commitment" showLineNumbers -fn main(x: Field, y: Field, expected_commitment: std::embedded_curve_ops::EmbeddedCurvePoint) { - let commitment = std::hash::pedersen_commitment([x, y]); - assert_eq(commitment.x, expected_commitment.x); - assert_eq(commitment.y, expected_commitment.y); -} -``` -> Source code: test_programs/execution_success/pedersen_commitment/src/main.nr#L1-L8 - - - - -## keccak256 - -Given an array of bytes (`u8`), returns the resulting keccak hash as an array of -32 bytes (`[u8; 32]`). Specify a message_size to hash only the first -`message_size` bytes of the input. - -```rust title="keccak256" showLineNumbers -pub fn keccak256(input: [u8; N], message_size: u32) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L104-L106 - - -example: - -```rust title="keccak256" showLineNumbers -fn main(x: Field, result: [u8; 32]) { - // We use the `as` keyword here to denote the fact that we want to take just the first byte from the x Field - // The padding is taken care of by the program - let digest = std::hash::keccak256([x as u8], 1); - assert(digest == result); - - //#1399: variable message size - let message_size = 4; - let hash_a = std::hash::keccak256([1, 2, 3, 4], message_size); - let hash_b = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size); - - assert(hash_a == hash_b); - - let message_size_big = 8; - let hash_c = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size_big); - - assert(hash_a != hash_c); -} -``` -> Source code: test_programs/execution_success/keccak256/src/main.nr#L1-L21 - - - - -## poseidon - -Given an array of Fields, returns a new Field with the Poseidon Hash. Mind that you need to specify -how many inputs are there to your Poseidon function. - -```rust -// example for hash_1, hash_2 accepts an array of length 2, etc -fn hash_1(input: [Field; 1]) -> Field -``` - -example: - -```rust title="poseidon" showLineNumbers -use std::hash::poseidon; - -fn main(x1: [Field; 2], y1: pub Field, x2: [Field; 4], y2: pub Field) { - let hash1 = poseidon::bn254::hash_2(x1); - assert(hash1 == y1); - - let hash2 = poseidon::bn254::hash_4(x2); - assert(hash2 == y2); -} -``` -> Source code: test_programs/execution_success/poseidon_bn254_hash/src/main.nr#L1-L11 - - -## poseidon 2 - -Given an array of Fields, returns a new Field with the Poseidon2 Hash. Contrary to the Poseidon -function, there is only one hash and you can specify a message_size to hash only the first -`message_size` bytes of the input, - -```rust -// example for hashing the first three elements of the input -Poseidon2::hash(input, 3); -``` - -example: - -```rust title="poseidon2" showLineNumbers -use std::hash::poseidon2; - -fn main(inputs: [Field; 4], expected_hash: Field) { - let hash = poseidon2::Poseidon2::hash(inputs, inputs.len()); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/poseidon2/src/main.nr#L1-L8 - - -## mimc_bn254 and mimc - -`mimc_bn254` is `mimc`, but with hardcoded parameters for the BN254 curve. You can use it by -providing an array of Fields, and it returns a Field with the hash. You can use the `mimc` method if -you're willing to input your own constants: - -```rust -fn mimc(x: Field, k: Field, constants: [Field; N], exp : Field) -> Field -``` - -otherwise, use the `mimc_bn254` method: - -```rust -fn mimc_bn254(array: [Field; N]) -> Field -``` - -example: - -```rust - -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::mimc::mimc_bn254(x); -} -``` - -## hash_to_field - -```rust -fn hash_to_field(_input : [Field]) -> Field {} -``` - -Calculates the `blake2s` hash of the inputs and returns the hash modulo the field modulus to return -a value which can be represented as a `Field`. - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/schnorr.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/schnorr.mdx deleted file mode 100644 index 00e7f257612..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/schnorr.mdx +++ /dev/null @@ -1,64 +0,0 @@ ---- -title: Schnorr Signatures -description: Learn how you can verify Schnorr signatures using Noir -keywords: [cryptographic primitives, Noir project, schnorr, signatures] -sidebar_position: 2 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## schnorr::verify_signature - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin). -See schnorr::verify_signature_slice for a version that works directly on slices. - -```rust title="schnorr_verify" showLineNumbers -pub fn verify_signature( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L2-L9 - - -where `_signature` can be generated like so using the npm package -[@noir-lang/barretenberg](https://www.npmjs.com/package/@noir-lang/barretenberg) - -```js -const { BarretenbergWasm } = require('@noir-lang/barretenberg/dest/wasm'); -const { Schnorr } = require('@noir-lang/barretenberg/dest/crypto/schnorr'); - -... - -const barretenberg = await BarretenbergWasm.new(); -const schnorr = new Schnorr(barretenberg); -const pubKey = schnorr.computePublicKey(privateKey); -const message = ... -const signature = Array.from( - schnorr.constructSignature(hash, privateKey).toBuffer() -); - -... -``` - - - -## schnorr::verify_signature_slice - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin) -where the message is a slice. - -```rust title="schnorr_verify_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/recursion.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/recursion.mdx deleted file mode 100644 index 8fdb8e8f514..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/recursion.mdx +++ /dev/null @@ -1,85 +0,0 @@ ---- -title: Recursive Proofs -description: Learn about how to write recursive proofs in Noir. -keywords: [recursion, recursive proofs, verification_key, verify_proof] ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports recursively verifying proofs, meaning you verify the proof of a Noir program in another Noir program. This enables creating proofs of arbitrary size by doing step-wise verification of smaller components of a large proof. - -Read [the explainer on recursion](../../explainers/explainer-recursion.md) to know more about this function and the [guide on how to use it.](../../how_to/how-to-recursion.md) - -## The `#[recursive]` Attribute - -In Noir, the `#[recursive]` attribute is used to indicate that a circuit is designed for recursive proof generation. When applied, it informs the compiler and the tooling that the circuit should be compiled in a way that makes its proofs suitable for recursive verification. This attribute eliminates the need for manual flagging of recursion at the tooling level, streamlining the proof generation process for recursive circuits. - -### Example usage with `#[recursive]` - -```rust -#[recursive] -fn main(x: Field, y: pub Field) { - assert(x == y, "x and y are not equal"); -} - -// This marks the circuit as recursion-friendly and indicates that proofs generated from this circuit -// are intended for recursive verification. -``` - -By incorporating this attribute directly in the circuit's definition, tooling like Nargo and NoirJS can automatically execute recursive-specific duties for Noir programs (e.g. recursive-friendly proof artifact generation) without additional flags or configurations. - -## Verifying Recursive Proofs - -```rust -#[foreign(recursive_aggregation)] -pub fn verify_proof(verification_key: [Field], proof: [Field], public_inputs: [Field], key_hash: Field) {} -``` - - - -## Example usage - -```rust - -fn main( - verification_key : [Field; 114], - proof : [Field; 93], - public_inputs : [Field; 1], - key_hash : Field, - proof_b : [Field; 93], -) { - std::verify_proof( - verification_key.as_slice(), - proof.as_slice(), - public_inputs.as_slice(), - key_hash - ); - - std::verify_proof( - verification_key.as_slice(), - proof_b.as_slice(), - public_inputs.as_slice(), - key_hash - ); -} -``` - -You can see a full example of recursive proofs in [this example recursion demo repo](https://github.com/noir-lang/noir-examples/tree/master/recursion). - -## Parameters - -### `verification_key` - -The verification key for the zk program that is being verified. - -### `proof` - -The proof for the zk program that is being verified. - -### `public_inputs` - -These represent the public inputs of the proof we are verifying. - -### `key_hash` - -A key hash is used to check the validity of the verification key. The circuit implementing this opcode can use this hash to ensure that the key provided to the circuit matches the key produced by the circuit creator. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/traits.md deleted file mode 100644 index 2ce8360bc83..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/traits.md +++ /dev/null @@ -1,501 +0,0 @@ ---- -title: Traits -description: Noir's stdlib provides a few commonly used traits. -keywords: [traits, trait, interface, protocol, default, add, eq] ---- - -## `std::default` - -### `std::default::Default` - -```rust title="default-trait" showLineNumbers -trait Default { - fn default() -> Self; -} -``` -> Source code: noir_stdlib/src/default.nr#L1-L5 - - -Constructs a default value of a type. - -Implementations: -```rust -impl Default for Field { .. } - -impl Default for i8 { .. } -impl Default for i16 { .. } -impl Default for i32 { .. } -impl Default for i64 { .. } - -impl Default for u8 { .. } -impl Default for u16 { .. } -impl Default for u32 { .. } -impl Default for u64 { .. } - -impl Default for () { .. } -impl Default for bool { .. } - -impl Default for [T; N] - where T: Default { .. } - -impl Default for [T] { .. } - -impl Default for (A, B) - where A: Default, B: Default { .. } - -impl Default for (A, B, C) - where A: Default, B: Default, C: Default { .. } - -impl Default for (A, B, C, D) - where A: Default, B: Default, C: Default, D: Default { .. } - -impl Default for (A, B, C, D, E) - where A: Default, B: Default, C: Default, D: Default, E: Default { .. } -``` - -For primitive integer types, the return value of `default` is `0`. Container -types such as arrays are filled with default values of their element type, -except slices whose length is unknown and thus defaulted to zero. - ---- - -## `std::convert` - -### `std::convert::From` - -```rust title="from-trait" showLineNumbers -trait From { - fn from(input: T) -> Self; -} -``` -> Source code: noir_stdlib/src/convert.nr#L1-L5 - - -The `From` trait defines how to convert from a given type `T` to the type on which the trait is implemented. - -The Noir standard library provides a number of implementations of `From` between primitive types. -```rust title="from-impls" showLineNumbers -// Unsigned integers - -impl From for u32 { fn from(value: u8) -> u32 { value as u32 } } - -impl From for u64 { fn from(value: u8) -> u64 { value as u64 } } -impl From for u64 { fn from(value: u32) -> u64 { value as u64 } } - -impl From for Field { fn from(value: u8) -> Field { value as Field } } -impl From for Field { fn from(value: u32) -> Field { value as Field } } -impl From for Field { fn from(value: u64) -> Field { value as Field } } - -// Signed integers - -impl From for i32 { fn from(value: i8) -> i32 { value as i32 } } - -impl From for i64 { fn from(value: i8) -> i64 { value as i64 } } -impl From for i64 { fn from(value: i32) -> i64 { value as i64 } } - -// Booleans -impl From for u8 { fn from(value: bool) -> u8 { value as u8 } } -impl From for u32 { fn from(value: bool) -> u32 { value as u32 } } -impl From for u64 { fn from(value: bool) -> u64 { value as u64 } } -impl From for i8 { fn from(value: bool) -> i8 { value as i8 } } -impl From for i32 { fn from(value: bool) -> i32 { value as i32 } } -impl From for i64 { fn from(value: bool) -> i64 { value as i64 } } -impl From for Field { fn from(value: bool) -> Field { value as Field } } -``` -> Source code: noir_stdlib/src/convert.nr#L25-L52 - - -#### When to implement `From` - -As a general rule of thumb, `From` may be implemented in the [situations where it would be suitable in Rust](https://doc.rust-lang.org/std/convert/trait.From.html#when-to-implement-from): - -- The conversion is *infallible*: Noir does not provide an equivalent to Rust's `TryFrom`, if the conversion can fail then provide a named method instead. -- The conversion is *lossless*: semantically, it should not lose or discard information. For example, `u32: From` can losslessly convert any `u16` into a valid `u32` such that the original `u16` can be recovered. On the other hand, `u16: From` should not be implemented as `2**16` is a `u32` which cannot be losslessly converted into a `u16`. -- The conversion is *value-preserving*: the conceptual kind and meaning of the resulting value is the same, even though the Noir type and technical representation might be different. While it's possible to infallibly and losslessly convert a `u8` into a `str<2>` hex representation, `4u8` and `"04"` are too different for `str<2>: From` to be implemented. -- The conversion is *obvious*: it's the only reasonable conversion between the two types. If there's ambiguity on how to convert between them such that the same input could potentially map to two different values then a named method should be used. For instance rather than implementing `U128: From<[u8; 16]>`, the methods `U128::from_le_bytes` and `U128::from_be_bytes` are used as otherwise the endianness of the array would be ambiguous, resulting in two potential values of `U128` from the same byte array. - -One additional recommendation specific to Noir is: -- The conversion is *efficient*: it's relatively cheap to convert between the two types. Due to being a ZK DSL, it's more important to avoid unnecessary computation compared to Rust. If the implementation of `From` would encourage users to perform unnecessary conversion, resulting in additional proving time, then it may be preferable to expose functionality such that this conversion may be avoided. - -### `std::convert::Into` - -The `Into` trait is defined as the reciprocal of `From`. It should be easy to convince yourself that if we can convert to type `A` from type `B`, then it's possible to convert type `B` into type `A`. - -For this reason, implementing `From` on a type will automatically generate a matching `Into` implementation. One should always prefer implementing `From` over `Into` as implementing `Into` will not generate a matching `From` implementation. - -```rust title="into-trait" showLineNumbers -trait Into { - fn into(self) -> T; -} - -impl Into for U where T: From { - fn into(self) -> T { - T::from(self) - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L13-L23 - - -`Into` is most useful when passing function arguments where the types don't quite match up with what the function expects. In this case, the compiler has enough type information to perform the necessary conversion by just appending `.into()` onto the arguments in question. - ---- - -## `std::cmp` - -### `std::cmp::Eq` - -```rust title="eq-trait" showLineNumbers -trait Eq { - fn eq(self, other: Self) -> bool; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L1-L5 - - -Returns `true` if `self` is equal to `other`. Implementing this trait on a type -allows the type to be used with `==` and `!=`. - -Implementations: -```rust -impl Eq for Field { .. } - -impl Eq for i8 { .. } -impl Eq for i16 { .. } -impl Eq for i32 { .. } -impl Eq for i64 { .. } - -impl Eq for u8 { .. } -impl Eq for u16 { .. } -impl Eq for u32 { .. } -impl Eq for u64 { .. } - -impl Eq for () { .. } -impl Eq for bool { .. } - -impl Eq for [T; N] - where T: Eq { .. } - -impl Eq for [T] - where T: Eq { .. } - -impl Eq for (A, B) - where A: Eq, B: Eq { .. } - -impl Eq for (A, B, C) - where A: Eq, B: Eq, C: Eq { .. } - -impl Eq for (A, B, C, D) - where A: Eq, B: Eq, C: Eq, D: Eq { .. } - -impl Eq for (A, B, C, D, E) - where A: Eq, B: Eq, C: Eq, D: Eq, E: Eq { .. } -``` - -### `std::cmp::Ord` - -```rust title="ord-trait" showLineNumbers -trait Ord { - fn cmp(self, other: Self) -> Ordering; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L102-L106 - - -`a.cmp(b)` compares two values returning `Ordering::less()` if `a < b`, -`Ordering::equal()` if `a == b`, or `Ordering::greater()` if `a > b`. -Implementing this trait on a type allows `<`, `<=`, `>`, and `>=` to be -used on values of the type. - -`std::cmp` also provides `max` and `min` functions for any type which implements the `Ord` trait. - -Implementations: - -```rust -impl Ord for u8 { .. } -impl Ord for u16 { .. } -impl Ord for u32 { .. } -impl Ord for u64 { .. } - -impl Ord for i8 { .. } -impl Ord for i16 { .. } -impl Ord for i32 { .. } - -impl Ord for i64 { .. } - -impl Ord for () { .. } -impl Ord for bool { .. } - -impl Ord for [T; N] - where T: Ord { .. } - -impl Ord for [T] - where T: Ord { .. } - -impl Ord for (A, B) - where A: Ord, B: Ord { .. } - -impl Ord for (A, B, C) - where A: Ord, B: Ord, C: Ord { .. } - -impl Ord for (A, B, C, D) - where A: Ord, B: Ord, C: Ord, D: Ord { .. } - -impl Ord for (A, B, C, D, E) - where A: Ord, B: Ord, C: Ord, D: Ord, E: Ord { .. } -``` - ---- - -## `std::ops` - -### `std::ops::Add`, `std::ops::Sub`, `std::ops::Mul`, and `std::ops::Div` - -These traits abstract over addition, subtraction, multiplication, and division respectively. -Implementing these traits for a given type will also allow that type to be used with the corresponding operator -for that trait (`+` for Add, etc) in addition to the normal method names. - -```rust title="add-trait" showLineNumbers -trait Add { - fn add(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L1-L5 - -```rust title="sub-trait" showLineNumbers -trait Sub { - fn sub(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L19-L23 - -```rust title="mul-trait" showLineNumbers -trait Mul { - fn mul(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L37-L41 - -```rust title="div-trait" showLineNumbers -trait Div { - fn div(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L55-L59 - - -The implementations block below is given for the `Add` trait, but the same types that implement -`Add` also implement `Sub`, `Mul`, and `Div`. - -Implementations: -```rust -impl Add for Field { .. } - -impl Add for i8 { .. } -impl Add for i16 { .. } -impl Add for i32 { .. } -impl Add for i64 { .. } - -impl Add for u8 { .. } -impl Add for u16 { .. } -impl Add for u32 { .. } -impl Add for u64 { .. } -``` - -### `std::ops::Rem` - -```rust title="rem-trait" showLineNumbers -trait Rem{ - fn rem(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L73-L77 - - -`Rem::rem(a, b)` is the remainder function returning the result of what is -left after dividing `a` and `b`. Implementing `Rem` allows the `%` operator -to be used with the implementation type. - -Unlike other numeric traits, `Rem` is not implemented for `Field`. - -Implementations: -```rust -impl Rem for u8 { fn rem(self, other: u8) -> u8 { self % other } } -impl Rem for u16 { fn rem(self, other: u16) -> u16 { self % other } } -impl Rem for u32 { fn rem(self, other: u32) -> u32 { self % other } } -impl Rem for u64 { fn rem(self, other: u64) -> u64 { self % other } } - -impl Rem for i8 { fn rem(self, other: i8) -> i8 { self % other } } -impl Rem for i16 { fn rem(self, other: i16) -> i16 { self % other } } -impl Rem for i32 { fn rem(self, other: i32) -> i32 { self % other } } -impl Rem for i64 { fn rem(self, other: i64) -> i64 { self % other } } -``` - -### `std::ops::Neg` - -```rust title="neg-trait" showLineNumbers -trait Neg { - fn neg(self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L89-L93 - - -`Neg::neg` is equivalent to the unary negation operator `-`. - -Implementations: -```rust title="neg-trait-impls" showLineNumbers -impl Neg for Field { fn neg(self) -> Field { -self } } - -impl Neg for i8 { fn neg(self) -> i8 { -self } } -impl Neg for i16 { fn neg(self) -> i16 { -self } } -impl Neg for i32 { fn neg(self) -> i32 { -self } } -impl Neg for i64 { fn neg(self) -> i64 { -self } } -``` -> Source code: noir_stdlib/src/ops/arith.nr#L95-L102 - - -### `std::ops::Not` - -```rust title="not-trait" showLineNumbers -trait Not { - fn not(self: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L1-L5 - - -`Not::not` is equivalent to the unary bitwise NOT operator `!`. - -Implementations: -```rust title="not-trait-impls" showLineNumbers -impl Not for bool { fn not(self) -> bool { !self } } - -impl Not for u64 { fn not(self) -> u64 { !self } } -impl Not for u32 { fn not(self) -> u32 { !self } } -impl Not for u16 { fn not(self) -> u16 { !self } } -impl Not for u8 { fn not(self) -> u8 { !self } } -impl Not for u1 { fn not(self) -> u1 { !self } } - -impl Not for i8 { fn not(self) -> i8 { !self } } -impl Not for i16 { fn not(self) -> i16 { !self } } -impl Not for i32 { fn not(self) -> i32 { !self } } -impl Not for i64 { fn not(self) -> i64 { !self } } -``` -> Source code: noir_stdlib/src/ops/bit.nr#L7-L20 - - -### `std::ops::{ BitOr, BitAnd, BitXor }` - -```rust title="bitor-trait" showLineNumbers -trait BitOr { - fn bitor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L22-L26 - -```rust title="bitand-trait" showLineNumbers -trait BitAnd { - fn bitand(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L40-L44 - -```rust title="bitxor-trait" showLineNumbers -trait BitXor { - fn bitxor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L58-L62 - - -Traits for the bitwise operations `|`, `&`, and `^`. - -Implementing `BitOr`, `BitAnd` or `BitXor` for a type allows the `|`, `&`, or `^` operator respectively -to be used with the type. - -The implementations block below is given for the `BitOr` trait, but the same types that implement -`BitOr` also implement `BitAnd` and `BitXor`. - -Implementations: -```rust -impl BitOr for bool { fn bitor(self, other: bool) -> bool { self | other } } - -impl BitOr for u8 { fn bitor(self, other: u8) -> u8 { self | other } } -impl BitOr for u16 { fn bitor(self, other: u16) -> u16 { self | other } } -impl BitOr for u32 { fn bitor(self, other: u32) -> u32 { self | other } } -impl BitOr for u64 { fn bitor(self, other: u64) -> u64 { self | other } } - -impl BitOr for i8 { fn bitor(self, other: i8) -> i8 { self | other } } -impl BitOr for i16 { fn bitor(self, other: i16) -> i16 { self | other } } -impl BitOr for i32 { fn bitor(self, other: i32) -> i32 { self | other } } -impl BitOr for i64 { fn bitor(self, other: i64) -> i64 { self | other } } -``` - -### `std::ops::{ Shl, Shr }` - -```rust title="shl-trait" showLineNumbers -trait Shl { - fn shl(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L76-L80 - -```rust title="shr-trait" showLineNumbers -trait Shr { - fn shr(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L93-L97 - - -Traits for a bit shift left and bit shift right. - -Implementing `Shl` for a type allows the left shift operator (`<<`) to be used with the implementation type. -Similarly, implementing `Shr` allows the right shift operator (`>>`) to be used with the type. - -Note that bit shifting is not currently implemented for signed types. - -The implementations block below is given for the `Shl` trait, but the same types that implement -`Shl` also implement `Shr`. - -Implementations: -```rust -impl Shl for u8 { fn shl(self, other: u8) -> u8 { self << other } } -impl Shl for u16 { fn shl(self, other: u16) -> u16 { self << other } } -impl Shl for u32 { fn shl(self, other: u32) -> u32 { self << other } } -impl Shl for u64 { fn shl(self, other: u64) -> u64 { self << other } } -``` - ---- - -## `std::append` - -### `std::append::Append` - -`Append` can abstract over types that can be appended to - usually container types: - -```rust title="append-trait" showLineNumbers -trait Append { - fn empty() -> Self; - fn append(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/append.nr#L9-L14 - - -`Append` requires two methods: - -- `empty`: Constructs an empty value of `Self`. -- `append`: Append two values together, returning the result. - -Additionally, it is expected that for any implementation: - -- `T::empty().append(x) == x` -- `x.append(T::empty()) == x` - -Implementations: -```rust -impl Append for [T] -impl Append for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/zeroed.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/zeroed.md deleted file mode 100644 index f450fecdd36..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/zeroed.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Zeroed Function -description: - The zeroed function returns a zeroed value of any type. -keywords: - [ - zeroed - ] ---- - -Implements `fn zeroed() -> T` to return a zeroed value of any type. This function is generally unsafe to use as the zeroed bit pattern is not guaranteed to be valid for all types. It can however, be useful in cases when the value is guaranteed not to be used such as in a BoundedVec library implementing a growable vector, up to a certain length, backed by an array. The array can be initialized with zeroed values which are guaranteed to be inaccessible until the vector is pushed to. Similarly, enumerations in noir can be implemented using this method by providing zeroed values for the unused variants. - -You can access the function at `std::unsafe::zeroed`. - -This function currently supports the following types: - -- Field -- Bool -- Uint -- Array -- Slice -- String -- Tuple -- Function - -Using it on other types could result in unexpected behavior. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md deleted file mode 100644 index 42f065f4a4e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md +++ /dev/null @@ -1,141 +0,0 @@ -# BarretenbergBackend - -## Implements - -- [`Backend`](../index.md#backend) -- [`Backend`](../index.md#backend) - -## Constructors - -### new BarretenbergBackend(acirCircuit, options) - -```ts -new BarretenbergBackend(acirCircuit, options): BarretenbergBackend -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `acirCircuit` | `CompiledCircuit` | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`BarretenbergBackend`](BarretenbergBackend.md) - -## Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `acirComposer` | `any` | - | -| `acirUncompressedBytecode` | `Uint8Array` | - | -| `api` | `Barretenberg` | - | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - | - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### generateProof() - -```ts -generateProof(compressedWitness): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `compressedWitness` | `Uint8Array` | - -#### Returns - -`Promise`\<`ProofData`\> - -#### Description - -Generates a proof - -*** - -### generateRecursiveProofArtifacts() - -```ts -generateRecursiveProofArtifacts(proofData, numOfPublicInputs): Promise -``` - -Generates artifacts that will be passed to a circuit that will verify this proof. - -Instead of passing the proof and verification key as a byte array, we pass them -as fields which makes it cheaper to verify in a circuit. - -The proof that is passed here will have been created using a circuit -that has the #[recursive] attribute on its `main` method. - -The number of public inputs denotes how many public inputs are in the inner proof. - -#### Parameters - -| Parameter | Type | Default value | -| :------ | :------ | :------ | -| `proofData` | `ProofData` | `undefined` | -| `numOfPublicInputs` | `number` | `0` | - -#### Returns - -`Promise`\<`object`\> - -#### Example - -```typescript -const artifacts = await backend.generateRecursiveProofArtifacts(proof, numOfPublicInputs); -``` - -*** - -### getVerificationKey() - -```ts -getVerificationKey(): Promise -``` - -#### Returns - -`Promise`\<`Uint8Array`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md deleted file mode 100644 index 500276ea748..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md +++ /dev/null @@ -1,58 +0,0 @@ -# BarretenbergVerifier - -## Constructors - -### new BarretenbergVerifier(options) - -```ts -new BarretenbergVerifier(options): BarretenbergVerifier -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`BarretenbergVerifier`](BarretenbergVerifier.md) - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData, verificationKey): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | -| `verificationKey` | `Uint8Array` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/index.md deleted file mode 100644 index 14dfac681d4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/index.md +++ /dev/null @@ -1,40 +0,0 @@ -# backend_barretenberg - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [BarretenbergBackend](classes/BarretenbergBackend.md) | - | -| [BarretenbergVerifier](classes/BarretenbergVerifier.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [BackendOptions](type-aliases/BackendOptions.md) | - | - -## References - -### CompiledCircuit - -Renames and re-exports [Backend](index.md#backend) - -*** - -### ProofData - -Renames and re-exports [Backend](index.md#backend) - -## Variables - -### Backend - -```ts -Backend: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md deleted file mode 100644 index b49a479f4f4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md +++ /dev/null @@ -1,21 +0,0 @@ -# BackendOptions - -```ts -type BackendOptions: object; -``` - -## Description - -An options object, currently only used to specify the number of threads to use. - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `memory` | `object` | - | -| `memory.maximum` | `number` | - | -| `threads` | `number` | **Description**

Number of threads | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs deleted file mode 100644 index d7d5128f9e3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend","label":"BarretenbergBackend"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier","label":"BarretenbergVerifier"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions","label":"BackendOptions"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/functions/keccak256.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/functions/keccak256.md deleted file mode 100644 index d10f155ce86..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/functions/keccak256.md +++ /dev/null @@ -1,21 +0,0 @@ -# keccak256() - -```ts -keccak256(inputs): Uint8Array -``` - -Calculates the Keccak256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/functions/sha256.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/functions/sha256.md deleted file mode 100644 index 6ba4ecac022..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/functions/sha256.md +++ /dev/null @@ -1,21 +0,0 @@ -# sha256() - -```ts -sha256(inputs): Uint8Array -``` - -Calculates the SHA256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/index.md deleted file mode 100644 index 166508f7124..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_js - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [Noir](classes/Noir.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [ErrorWithPayload](type-aliases/ErrorWithPayload.md) | - | -| [ForeignCallHandler](type-aliases/ForeignCallHandler.md) | A callback which performs an foreign call and returns the response. | -| [ForeignCallInput](type-aliases/ForeignCallInput.md) | - | -| [ForeignCallOutput](type-aliases/ForeignCallOutput.md) | - | -| [WitnessMap](type-aliases/WitnessMap.md) | - | - -### Functions - -| Function | Description | -| :------ | :------ | -| [and](functions/and.md) | Performs a bitwise AND operation between `lhs` and `rhs` | -| [blake2s256](functions/blake2s256.md) | Calculates the Blake2s256 hash of the input bytes | -| [ecdsa\_secp256k1\_verify](functions/ecdsa_secp256k1_verify.md) | Verifies a ECDSA signature over the secp256k1 curve. | -| [ecdsa\_secp256r1\_verify](functions/ecdsa_secp256r1_verify.md) | Verifies a ECDSA signature over the secp256r1 curve. | -| [keccak256](functions/keccak256.md) | Calculates the Keccak256 hash of the input bytes | -| [sha256](functions/sha256.md) | Calculates the SHA256 hash of the input bytes | -| [xor](functions/xor.md) | Performs a bitwise XOR operation between `lhs` and `rhs` | - -## References - -### CompiledCircuit - -Renames and re-exports [InputMap](index.md#inputmap) - -## Variables - -### InputMap - -```ts -InputMap: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs deleted file mode 100644 index b3156097df6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/noir_js/classes/Noir","label":"Noir"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ErrorWithPayload","label":"ErrorWithPayload"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallHandler","label":"ForeignCallHandler"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallInput","label":"ForeignCallInput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallOutput","label":"ForeignCallOutput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/WitnessMap","label":"WitnessMap"}]},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_js/functions/and","label":"and"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/blake2s256","label":"blake2s256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify","label":"ecdsa_secp256k1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify","label":"ecdsa_secp256r1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/keccak256","label":"keccak256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/sha256","label":"sha256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/xor","label":"xor"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_wasm/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_wasm/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/NoirJS/noir_wasm/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/nargo_commands.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/nargo_commands.md deleted file mode 100644 index 2d6defc5a44..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/reference/nargo_commands.md +++ /dev/null @@ -1,271 +0,0 @@ ---- -title: Nargo -description: - Noir CLI Commands for Noir Prover and Verifier to create, execute, prove and verify programs, - generate Solidity verifier smart contract and compile into JSON file containing ACIR - representation and ABI of circuit. -keywords: - [ - Nargo, - Noir CLI, - Noir Prover, - Noir Verifier, - generate Solidity verifier, - compile JSON file, - ACIR representation, - ABI of circuit, - TypeScript, - ] -sidebar_position: 0 ---- - -# Command-Line Help for `nargo` - -This document contains the help content for the `nargo` command-line program. - -**Command Overview:** - -* [`nargo`↴](#nargo) -* [`nargo check`↴](#nargo-check) -* [`nargo fmt`↴](#nargo-fmt) -* [`nargo compile`↴](#nargo-compile) -* [`nargo new`↴](#nargo-new) -* [`nargo init`↴](#nargo-init) -* [`nargo execute`↴](#nargo-execute) -* [`nargo debug`↴](#nargo-debug) -* [`nargo test`↴](#nargo-test) -* [`nargo info`↴](#nargo-info) -* [`nargo lsp`↴](#nargo-lsp) - -## `nargo` - -Noir's package manager - -**Usage:** `nargo ` - -###### **Subcommands:** - -* `check` — Checks the constraint system for errors -* `fmt` — Format the Noir files in a workspace -* `compile` — Compile the program and its secret execution trace into ACIR format -* `new` — Create a Noir project in a new directory -* `init` — Create a Noir project in the current directory -* `execute` — Executes a circuit to calculate its return value -* `debug` — Executes a circuit in debug mode -* `test` — Run the tests for this program -* `info` — Provides detailed information on each of a program's function (represented by a single circuit) -* `lsp` — Starts the Noir LSP server - -###### **Options:** - - - - -## `nargo check` - -Checks the constraint system for errors - -**Usage:** `nargo check [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to check -* `--workspace` — Check all packages in the workspace -* `--overwrite` — Force overwrite of existing files -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" - - - -## `nargo fmt` - -Format the Noir files in a workspace - -**Usage:** `nargo fmt [OPTIONS]` - -###### **Options:** - -* `--check` — Run noirfmt in check mode - - - -## `nargo compile` - -Compile the program and its secret execution trace into ACIR format - -**Usage:** `nargo compile [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to compile -* `--workspace` — Compile all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" - - - -## `nargo new` - -Create a Noir project in a new directory - -**Usage:** `nargo new [OPTIONS] ` - -###### **Arguments:** - -* `` — The path to save the new project - -###### **Options:** - -* `--name ` — Name of the package [default: package directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo init` - -Create a Noir project in the current directory - -**Usage:** `nargo init [OPTIONS]` - -###### **Options:** - -* `--name ` — Name of the package [default: current directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo execute` - -Executes a circuit to calculate its return value - -**Usage:** `nargo execute [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--workspace` — Execute all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo debug` - -Executes a circuit in debug mode - -**Usage:** `nargo debug [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--acir-mode` — Force ACIR output (disabling instrumentation) -* `--skip-instrumentation ` — Disable vars debug instrumentation (enabled by default) - - Possible values: `true`, `false` - - - - -## `nargo test` - -Run the tests for this program - -**Usage:** `nargo test [OPTIONS] [TEST_NAME]` - -###### **Arguments:** - -* `` — If given, only tests with names containing this string will be run - -###### **Options:** - -* `--show-output` — Display output of `println` statements -* `--exact` — Only run tests that match exactly -* `--package ` — The name of the package to test -* `--workspace` — Test all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo info` - -Provides detailed information on each of a program's function (represented by a single circuit) - -Current information provided per circuit: 1. The number of ACIR opcodes 2. Counts the final number gates in the circuit used by a backend - -**Usage:** `nargo info [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to detail -* `--workspace` — Detail all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" - - - -## `nargo lsp` - -Starts the Noir LSP server - -Starts an LSP server which allows IDEs such as VS Code to display diagnostics in Noir source. - -VS Code Noir Language Support: https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir - -**Usage:** `nargo lsp` - - - -
- - - This document was generated automatically by - clap-markdown. - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/tooling/debugger.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/tooling/debugger.md deleted file mode 100644 index 9b7565ba9ff..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/tooling/debugger.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Debugger -description: Learn about the Noir Debugger, in its REPL or VS Code versions. -keywords: [Nargo, VSCode, Visual Studio Code, REPL, Debugger] -sidebar_position: 2 ---- - -# Noir Debugger - -There are currently two ways of debugging Noir programs: - -1. From VS Code, via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). -2. Via the REPL debugger, which ships with Nargo. - -In order to use either version of the debugger, you will need to install recent enough versions of Noir, [Nargo](../getting_started/installation/index.md) and vscode-noir: - -- Noir & Nargo ≥0.28.0 -- Noir's VS Code extension ≥0.0.11 - -:::info -At the moment, the debugger supports debugging binary projects, but not contracts. -::: - -We cover the VS Code Noir debugger more in depth in [its VS Code debugger how-to guide](../how_to/debugger/debugging_with_vs_code.md) and [the reference](../reference/debugger/debugger_vscode.md). - -The REPL debugger is discussed at length in [the REPL debugger how-to guide](../how_to/debugger/debugging_with_the_repl.md) and [the reference](../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v0.32.0/tutorials/noirjs_app.md deleted file mode 100644 index 8c23b639f12..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/tutorials/noirjs_app.md +++ /dev/null @@ -1,348 +0,0 @@ ---- -title: Building a web app with NoirJS -description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. -keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] -sidebar_position: 0 -pagination_next: noir/concepts/data_types/index ---- - -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! - -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/installation/index.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! - -### Nargo - -Run: - -```bash -nargo new circuit -``` - -And... That's about it. Your program is ready to be compiled and run. - -To compile, let's `cd` into the `circuit` folder to enter our project, and call: - -```bash -nargo compile -``` - -This compiles our circuit into `json` format and add it to a new `target` folder. - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` - -::: - -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/hello_noir/index.md) guide. However, we want our app to run on the browser, so we need Vite. - -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. - -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". - -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: - -```bash -cd vite-project -``` - -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; - -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); -``` - -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: - -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev -``` - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... -``` - -::: - -#### Some cleanup - -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. - -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) - -## HTML - -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: - -```html - - - - - - -

Noir app

-
- - -
-
-

Logs

-

Proof

-
- - -``` - -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). - -Start by pasting in this boilerplate code: - -```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} - -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } -}); -``` - -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html -``` - -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - -::: - -## Some NoirJS - -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: - -```ts -import circuit from '../circuit/target/circuit.json'; -``` - -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: - -```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -import { Noir } from '@noir-lang/noir_js'; -``` - -And instantiate them inside our try-catch block: - -```ts -// try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); -// } -``` - -:::note - -For the remainder of the tutorial, everything will be happening inside the `try` block - -::: - -## Our app - -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: - -```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; -``` - -Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: - -```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); -const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); -``` - -You're probably eager to see stuff happening, so go and run your app now! - -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. - -![Getting Started 0](@site/static/img/noir_getting_started_1.png) - -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! - -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. - -## Verifying - -Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: - -```js -display('logs', 'Verifying proof... ⌛'); -const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); -``` - -You have successfully generated a client-side Noir web app! - -![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) - -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-oracle.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-oracle.md deleted file mode 100644 index 821e1f95c04..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-oracle.md +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: Oracles -description: This guide provides an in-depth understanding of how Oracles work in Noir programming. Learn how to use outside calculations in your programs, constrain oracles, and understand their uses and limitations. -keywords: - - Noir Programming - - Oracles - - JSON-RPC - - Foreign Call Handlers - - Constrained Functions - - Blockchain Programming -sidebar_position: 1 ---- - -If you've seen "The Matrix" you may recall "The Oracle" as Gloria Foster smoking cigarettes and baking cookies. While she appears to "know things", she is actually providing a calculation of a pre-determined future. Noir Oracles are similar, in a way. They don't calculate the future (yet), but they allow you to use outside calculations in your programs. - -![matrix oracle prediction](@site/static/img/memes/matrix_oracle.jpeg) - -A Noir program is usually self-contained. You can pass certain inputs to it, and it will generate a deterministic output for those inputs. But what if you wanted to defer some calculation to an outside process or source? - -Oracles are functions that provide this feature. - -## Use cases - -An example usage for Oracles is proving something on-chain. For example, proving that the ETH-USDC quote was below a certain target at a certain block time. Or even making more complex proofs like proving the ownership of an NFT as an anonymous login method. - -Another interesting use case is to defer expensive calculations to be made outside of the Noir program, and then constraining the result; similar to the use of [unconstrained functions](../noir/concepts//unconstrained.md). - -In short, anything that can be constrained in a Noir program but needs to be fetched from an external source is a great candidate to be used in oracles. - -## Constraining oracles - -Just like in The Matrix, Oracles are powerful. But with great power, comes great responsibility. Just because you're using them in a Noir program doesn't mean they're true. Noir has no superpowers. If you want to prove that Portugal won the Euro Cup 2016, you're still relying on potentially untrusted information. - -To give a concrete example, Alice wants to login to the [NounsDAO](https://nouns.wtf/) forum with her username "noir_nouner" by proving she owns a noun without revealing her ethereum address. Her Noir program could have an oracle call like this: - -```rust -#[oracle(getNoun)] -unconstrained fn get_noun(address: Field) -> Field -``` - -This oracle could naively resolve with the number of Nouns she possesses. However, it is useless as a trusted source, as the oracle could resolve to anything Alice wants. In order to make this oracle call actually useful, Alice would need to constrain the response from the oracle, by proving her address and the noun count belongs to the state tree of the contract. - -In short, **Oracles don't prove anything. Your Noir program does.** - -:::danger - -If you don't constrain the return of your oracle, you could be clearly opening an attack vector on your Noir program. Make double-triple sure that the return of an oracle call is constrained! - -::: - -## How to use Oracles - -On CLI, Nargo resolves oracles by making JSON RPC calls, which means it would require an RPC node to be running. - -In JavaScript, NoirJS accepts and resolves arbitrary call handlers (that is, not limited to JSON) as long as they match the expected types the developer defines. Refer to [Foreign Call Handler](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) to learn more about NoirJS's call handling. - -If you want to build using oracles, follow through to the [oracle guide](../how_to/how-to-oracles.md) for a simple example on how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-recursion.md deleted file mode 100644 index df8529ef4e0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-recursion.md +++ /dev/null @@ -1,176 +0,0 @@ ---- -title: Recursive proofs -description: Explore the concept of recursive proofs in Zero-Knowledge programming. Understand how recursion works in Noir, a language for writing smart contracts on the EVM blockchain. Learn through practical examples like Alice and Bob's guessing game, Charlie's recursive merkle tree, and Daniel's reusable components. Discover how to use recursive proofs to optimize computational resources and improve efficiency. - -keywords: - [ - "Recursive Proofs", - "Zero-Knowledge Programming", - "Noir", - "EVM Blockchain", - "Smart Contracts", - "Recursion in Noir", - "Alice and Bob Guessing Game", - "Recursive Merkle Tree", - "Reusable Components", - "Optimizing Computational Resources", - "Improving Efficiency", - "Verification Key", - "Aggregation", - "Recursive zkSNARK schemes", - "PLONK", - "Proving and Verification Keys" - ] -sidebar_position: 1 -pagination_next: how_to/how-to-recursion ---- - -In programming, we tend to think of recursion as something calling itself. A classic example would be the calculation of the factorial of a number: - -```js -function factorial(n) { - if (n === 0 || n === 1) { - return 1; - } else { - return n * factorial(n - 1); - } -} -``` - -In this case, while `n` is not `1`, this function will keep calling itself until it hits the base case, bubbling up the result on the call stack: - -```md - Is `n` 1? <--------- - /\ / - / \ n = n -1 - / \ / - Yes No -------- -``` - -In Zero-Knowledge, recursion has some similarities. - -It is not a Noir function calling itself, but a proof being used as an input to another circuit. In short, you verify one proof *inside* another proof, returning the proof that both proofs are valid. - -This means that, given enough computational resources, you can prove the correctness of any arbitrary number of proofs in a single proof. This could be useful to design state channels (for which a common example would be [Bitcoin's Lightning Network](https://en.wikipedia.org/wiki/Lightning_Network)), to save on gas costs by settling one proof on-chain, or simply to make business logic less dependent on a consensus mechanism. - -## Examples - -Let us look at some of these examples - -### Alice and Bob - Guessing game - -Alice and Bob are friends, and they like guessing games. They want to play a guessing game online, but for that, they need a trusted third-party that knows both of their secrets and finishes the game once someone wins. - -So, they use zero-knowledge proofs. Alice tries to guess Bob's number, and Bob will generate a ZK proof stating whether she succeeded or failed. - -This ZK proof can go on a smart contract, revealing the winner and even giving prizes. However, this means every turn needs to be verified on-chain. This incurs some cost and waiting time that may simply make the game too expensive or time-consuming to be worth it. - -As a solution, Alice proposes the following: "what if Bob generates his proof, and instead of sending it on-chain, I verify it *within* my own proof before playing my own turn?". - -She can then generate a proof that she verified his proof, and so on. - -```md - Did you fail? <-------------------------- - / \ / - / \ n = n -1 - / \ / - Yes No / - | | / - | | / - | You win / - | / - | / -Generate proof of that / - + / - my own guess ---------------- -``` - -### Charlie - Recursive merkle tree - -Charlie is a concerned citizen, and wants to be sure his vote in an election is accounted for. He votes with a ZK proof, but he has no way of knowing that his ZK proof was included in the total vote count! - -If the vote collector puts all of the votes into a [Merkle tree](https://en.wikipedia.org/wiki/Merkle_tree), everyone can prove the verification of two proofs within one proof, as such: - -```md - abcd - __________|______________ - | | - ab cd - _____|_____ ______|______ - | | | | - alice bob charlie daniel -``` - -Doing this recursively allows us to arrive on a final proof `abcd` which if true, verifies the correctness of all the votes. - -### Daniel - Reusable components - -Daniel has a big circuit and a big headache. A part of his circuit is a setup phase that finishes with some assertions that need to be made. But that section alone takes most of the proving time, and is largely independent of the rest of the circuit. - -He might find it more efficient to generate a proof for that setup phase separately, and verify that proof recursively in the actual business logic section of his circuit. This will allow for parallelization of both proofs, which results in a considerable speedup. - -## What params do I need - -As you can see in the [recursion reference](noir/standard_library/recursion.mdx), a simple recursive proof requires: - -- The proof to verify -- The Verification Key of the circuit that generated the proof -- A hash of this verification key, as it's needed for some backends -- The public inputs for the proof - -:::info - -Recursive zkSNARK schemes do not necessarily "verify a proof" in the sense that you expect a true or false to be spit out by the verifier. Rather an aggregation object is built over the public inputs. - -So, taking the example of Alice and Bob and their guessing game: - -- Alice makes her guess. Her proof is *not* recursive: it doesn't verify any proof within it! It's just a standard `assert(x != y)` circuit -- Bob verifies Alice's proof and makes his own guess. In this circuit, he doesn't exactly *prove* the verification of Alice's proof. Instead, he *aggregates* his proof to Alice's proof. The actual verification is done when the full proof is verified, for example when using `nargo verify` or through the verifier smart contract. - -We can imagine recursive proofs a [relay race](https://en.wikipedia.org/wiki/Relay_race). The first runner doesn't have to receive the baton from anyone else, as he/she already starts with it. But when his/her turn is over, the next runner needs to receive it, run a bit more, and pass it along. Even though every runner could theoretically verify the baton mid-run (why not? 🏃🔍), only at the end of the race does the referee verify that the whole race is valid. - -::: - -## Some architecture - -As with everything in computer science, there's no one-size-fits all. But there are some patterns that could help understanding and implementing them. To give three examples: - -### Adding some logic to a proof verification - -This would be an approach for something like our guessing game, where proofs are sent back and forth and are verified by each opponent. This circuit would be divided in two sections: - -- A `recursive verification` section, which would be just the call to `std::verify_proof`, and that would be skipped on the first move (since there's no proof to verify) -- A `guessing` section, which is basically the logic part where the actual guessing happens - -In such a situation, and assuming Alice is first, she would skip the first part and try to guess Bob's number. Bob would then verify her proof on the first section of his run, and try to guess Alice's number on the second part, and so on. - -### Aggregating proofs - -In some one-way interaction situations, recursion would allow for aggregation of simple proofs that don't need to be immediately verified on-chain or elsewhere. - -To give a practical example, a barman wouldn't need to verify a "proof-of-age" on-chain every time he serves alcohol to a customer. Instead, the architecture would comprise two circuits: - -- A `main`, non-recursive circuit with some logic -- A `recursive` circuit meant to verify two proofs in one proof - -The customer's proofs would be intermediate, and made on their phones, and the barman could just verify them locally. He would then aggregate them into a final proof sent on-chain (or elsewhere) at the end of the day. - -### Recursively verifying different circuits - -Nothing prevents you from verifying different circuits in a recursive proof, for example: - -- A `circuit1` circuit -- A `circuit2` circuit -- A `recursive` circuit - -In this example, a regulator could verify that taxes were paid for a specific purchase by aggregating both a `payer` circuit (proving that a purchase was made and taxes were paid), and a `receipt` circuit (proving that the payment was received) - -## How fast is it - -At the time of writing, verifying recursive proofs is surprisingly fast. This is because most of the time is spent on generating the verification key that will be used to generate the next proof. So you are able to cache the verification key and reuse it later. - -Currently, Noir JS packages don't expose the functionality of loading proving and verification keys, but that feature exists in the underlying `bb.js` package. - -## How can I try it - -Learn more about using recursion in Nargo and NoirJS in the [how-to guide](../how_to/how-to-recursion.md) and see a full example in [noir-examples](https://github.com/noir-lang/noir-examples). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-writing-noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-writing-noir.md deleted file mode 100644 index c8a42c379e6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/explainer-writing-noir.md +++ /dev/null @@ -1,173 +0,0 @@ ---- -title: Writing Performant Noir -description: Understand new considerations when writing Noir -keywords: [Noir, programming, rust] -tags: [Optimization] -sidebar_position: 0 ---- - - -This article intends to set you up with key concepts essential for writing more viable applications that use zero knowledge proofs, namely around efficient circuits. - -## Context - 'Efficient' is subjective - -When writing a web application for a performant computer with high-speed internet connection, writing efficient code sometimes is seen as an afterthought only if needed. Large multiplications running at the innermost of nested loops may not even be on a dev's radar. -When writing firmware for a battery-powered microcontroller, you think of cpu cycles as rations to keep within a product's power budget. - -> Code is written to create applications that perform specific tasks within specific constraints - -And these constraints differ depending on where the compiled code is execute. - -### The Ethereum Virtual Machine (EVM) - -In scenarios where extremely low gas costs are required for an Ethereum application to be viable/competitive, Ethereum smart contract developers get into what is colloquially known as: "*gas golfing*". Finding the lowest execution cost of their compiled code (EVM bytecode) to achieve a specific task. - -The equivalent optimization task when writing zk circuits is affectionately referred to as "*gate golfing*", finding the lowest gate representation of the compiled Noir code. - -### Coding for circuits - a paradigm shift - -In zero knowledge cryptography, code is compiled to "circuits" consisting of arithmetic gates, and gate count is the significant cost. Depending on the proving system this is linearly proportionate to proving time, and so from a product point this should be kept as low as possible. - -Whilst writing efficient code for web apps and Solidity has a few key differences, writing efficient circuits have a different set of considerations. It is a bit of a paradigm shift, like writing code for GPUs for the first time... - -For example, drawing a circle at (0, 0) of radius `r`: -- For a single CPU thread, -``` -for theta in 0..2*pi { - let x = r * cos(theta); - let y = r * sin(theta); - draw(x, y); -} // note: would do 0 - pi/2 and draw +ve/-ve x and y. -``` - -- For GPUs (simultaneous parallel calls with x, y across image), -``` -if (x^2 + y^2 = r^2) { - draw(x, y); -} -``` - -([Related](https://www.youtube.com/watch?v=-P28LKWTzrI)) - -Whilst this CPU -> GPU does not translate to circuits exactly, it is intended to exemplify the difference in intuition when coding for different machine capabilities/constraints. - -### Context Takeaway - -For those coming from a primarily web app background, this article will explain what you need to consider when writing circuits. Furthermore, for those experienced writing efficient machine code, prepare to shift what you think is efficient 😬 - -## Translating from Rust - -For some applications using Noir, existing code might be a convenient starting point to then proceed to optimize the gate count of. - -:::note -Many valuable functions and algorithms have been written in more established languages (C/C++), and converted to modern ones (like Rust). -::: - -Fortunately for Noir developers, when needing a particular function a Rust implementation can be readily compiled into Noir with some key changes. While the compiler does a decent amount of optimizations, it won't be able to change code that has been optimized for clock-cycles into code optimized for arithmetic gates. - -A few things to do when converting Rust code to Noir: -- `println!` is not a macro, use `println` function (same for `assert_eq`) -- No early `return` in function. Use constrain via assertion instead -- No passing by reference. Remove `&` operator to pass by value (copy) -- No boolean operators (`&&`, `||`). Use bitwise operators (`&`, `|`) with boolean values -- No type `usize`. Use types `u8`, `u32`, `u64`, ... -- `main` return must be public, `pub` -- No `const`, use `global` -- Noir's LSP is your friend, so error message should be informative enough to resolve syntax issues. - -## Writing efficient Noir for performant products - -The following points help refine our understanding over time. - -:::note -A Noir program makes a statement that can be verified. -::: - -It compiles to a structure that represents the calculation, and can assert results within the calculation at any stage (via the `constrain` keyword). - -A Noir program compiles to an Abstract Circuit Intermediate Representation which is: - - A tree structure - - Leaves (inputs) are the `Field` type - - Nodes contain arithmetic operations to combine them (gates) - - The root is the final result (return value) - -:::tip -The command `nargo info` shows the programs circuit size, and is useful to compare the value of changes made. -You can dig deeper and use the `--print-acir` param to take a closer look at individual ACIR opcodes, and the proving backend to see its gate count (eg for barretenberg, `bb gates -b ./target/program.json`). -::: - -### Use the `Field` type - -Since the native type of values in circuits are `Field`s, using them for variables in Noir means less gates converting them under the hood. - -:::tip -Where possible, use `Field` type for values. Using smaller value types, and bit-packing strategies, will result in MORE gates -::: - -**Note:** Need to remain mindful of overflow. Types with less bits may be used to limit the range of possible values prior to a calculation. - -### Use Arithmetic over non-arithmetic operations - -Since circuits are made of arithmetic gates, the cost of arithmetic operations tends to be one gate. Whereas for procedural code, they represent several clock cycles. - -Inversely, non-arithmetic operators are achieved with multiple gates, vs 1 clock cycle for procedural code. - -| (cost\op) | arithmetic
(`*`, `+`) | bit-wise ops
(eg `<`, `\|`, `>>`) | -| - | - | - | -| **cycles** | 10+ | 1 | -| **gates** | 1 | 10+ | - -Bit-wise operations (e.g. bit shifts `<<` and `>>`), albeit commonly used in general programming and especially for clock cycle optimizations, are on the contrary expensive in gates when performed within circuits. - -Translate away from bit shifts when writing constrained functions for the best performance. - -On the flip side, feel free to use bit shifts in unconstrained functions and tests if necessary, as they are executed outside of circuits and does not induce performance hits. - -### Use static over dynamic values - -Another general theme that manifests in different ways is that static reads are represented with less gates than dynamic ones. - -Reading from read-only memory (ROM) adds less gates than random-access memory (RAM), 2 vs ~3.25 due to the additional bounds checks. Arrays of fixed length (albeit used at a lower capacity), will generate less gates than dynamic storage. - -Related to this, if an index used to access an array is not known at compile time (ie unknown until run time), then ROM will be converted to RAM, expanding the gate count. - -:::tip -Use arrays and indices that are known at compile time where possible. -Using `assert_constant(i);` before an index, `i`, is used in an array will give a compile error if `i` is NOT known at compile time. -::: - -### Leverage unconstrained execution - -Constrained verification can leverage unconstrained execution, this is especially useful for operations that are represented by many gates. -Use an [unconstrained function](../noir/concepts/unconstrained.md) to perform gate-heavy calculations, then verify and constrain the result. - -Eg division generates more gates than multiplication, so calculating the quotient in an unconstrained function then constraining the product for the quotient and divisor (+ any remainder) equals the dividend will be more efficient. - -Use ` if is_unconstrained() { /`, to conditionally execute code if being called in an unconstrained vs constrained way. - -## Advanced - -Unless you're well into the depth of gate optimization, this advanced section can be ignored. - -### Combine arithmetic operations - -A Noir program can be honed further by combining arithmetic operators in a way that makes the most of each constraint of the backend proving system. This is in scenarios where the backend might not be doing this perfectly. - -Eg Barretenberg backend (current default for Noir) is a width-4 PLONKish constraint system -$ w_1*w_2*q_m + w_1*q_1 + w_2*q_2 + w_3*q_3 + w_4*q_4 + q_c = 0 $ - -Here we see there is one occurrence of witness 1 and 2 ($w_1$, $w_2$) being multiplied together, with addition to witnesses 1-4 ($w_1$ .. $w_4$) multiplied by 4 corresponding circuit constants ($q_1$ .. $q_4$) (plus a final circuit constant, $q_c$). - -Use `nargo info --print-acir`, to inspect the ACIR opcodes (and the proving system for gates), and it may present opportunities to amend the order of operations and reduce the number of constraints. - -#### Variable as witness vs expression - -If you've come this far and really know what you're doing at the equation level, a temporary lever (that will become unnecessary/useless over time) is: `std::as_witness`. This informs the compiler to save a variable as a witness not an expression. - -The compiler will mostly be correct and optimal, but this may help some near term edge cases that are yet to optimize. -Note: When used incorrectly it will create **less** efficient circuits (higher gate count). - -## References -- Guillaume's ["`Cryptdoku`" talk](https://www.youtube.com/watch?v=MrQyzuogxgg) (Jun'23) -- Tips from Tom, Jake and Zac. -- [Idiomatic Noir](https://www.vlayer.xyz/blog/idiomatic-noir-part-1-collections) blog post diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/backend/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/backend/_category_.json deleted file mode 100644 index b82e92beb0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/backend/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "position": 1, - "label": "Install Proving Backend", - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/backend/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/backend/index.md deleted file mode 100644 index 7192d954877..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/backend/index.md +++ /dev/null @@ -1,31 +0,0 @@ ---- -title: Proving Backend Installation -description: Proving backends offer command line tools for proving and verifying Noir programs. This page describes how to install `bb` as an example. -keywords: [ - Proving - Backend - Barretenberg - bb - bbup - Installation - Terminal - Command - CLI - Version -] -pagination_next: getting_started/hello_noir/index ---- - -Proving backends each provide their own tools for working with Noir programs, providing functionality like proof generation, proof verification, and verifier smart contract generation. - -For the latest information on tooling provided by each proving backend, installation instructions, Noir version compatibility... you may refer to the proving backends' own documentation. - -You can find the full list of proving backends compatible with Noir in [Awesome Noir](https://github.com/noir-lang/awesome-noir/?tab=readme-ov-file#proving-backends). - -## Example: Installing `bb` - -`bb` is the CLI tool provided by the [Barretenberg proving backend](https://github.com/AztecProtocol/barretenberg) developed by Aztec Labs. - -You can find the instructions for installation in [`bb`'s documentation](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/cpp/src/barretenberg/bb/readme.md#installation). - -Once installed, we are ready to start working on [our first Noir program](../hello_noir/index.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/_category_.json deleted file mode 100644 index 976a2325de0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/index.md deleted file mode 100644 index 3baae217eb3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/index.md +++ /dev/null @@ -1,152 +0,0 @@ ---- -title: Creating a Project -description: - Learn how to create and verify your first Noir program using Nargo, a programming language for - zero-knowledge proofs. -keywords: - [ - Nargo, - Noir, - zero-knowledge proofs, - programming language, - create Noir program, - verify Noir program, - step-by-step guide, - ] -sidebar_position: 1 - ---- - -Now that we have installed Nargo and a proving backend, it is time to make our first hello world program! - -### 1. Create a new project directory - -Noir code can live anywhere on your computer. Let us create a _projects_ folder in the home -directory to house our first Noir program. - -Create the directory and change directory into it by running: - -```sh -mkdir ~/projects -cd ~/projects -``` - -## Nargo - -Nargo provides the ability to initiate and execute Noir projects. Read the [Nargo installation](../installation/index.md) section to learn more about Nargo and how to install it. - -### 2. Create a new Noir project - -Now that we are in the projects directory, create a new Nargo project by running: - -```sh -nargo new hello_world -``` - -`hello_world` can be any arbitrary project name, we are simply using `hello_world` for demonstration. - -In production, it is common practice to name the project folder, `circuits`, for clarity amongst other folders in the codebase (like: `contracts`, `scripts`, and `test`). - -A `hello_world` folder would be created. Similar to Rust, the folder houses _src/main.nr_ and -_Nargo.toml_ which contain the source code and environmental options of your Noir program -respectively. - -#### Intro to Noir Syntax - -Let us take a closer look at _main.nr_. The default _main.nr_ generated should look like this: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -The first line of the program specifies the program's inputs: - -```rust -x : Field, y : pub Field -``` - -Program inputs in Noir are private by default (e.g. `x`), but can be labeled public using the -keyword `pub` (e.g. `y`). To learn more about private and public values, check the -[Data Types](../../noir/concepts/data_types/index.md) section. - -The next line of the program specifies its body: - -```rust -assert(x != y); -``` - -The Noir syntax `assert` can be interpreted as something similar to constraints in other zk-contract languages. - -For more Noir syntax, check the [Language Concepts](../../noir/concepts/comments.md) chapter. - -### 3. Build in/output files - -Change directory into _hello_world_ and build in/output files for your Noir program by running: - -```sh -cd hello_world -nargo check -``` - -A _Prover.toml_ file will be generated in your project directory, to allow specifying input values to the program. - -### 4. Execute the Noir program - -Now that the project is set up, we can execute our Noir program. - -Fill in input values for execution in the _Prover.toml_ file. For example: - -```toml -x = "1" -y = "2" -``` - -Execute your Noir program: - -```sh -nargo execute witness-name -``` - -The witness corresponding to this execution will then be written to the file `./target/witness-name.gz`. - -The command also automatically compiles your Noir program if it was not already / was edited, which you may notice the compiled artifacts being written to the file `./target/hello_world.json`. - -## Proving Backend - -Proving backends provide the ability to generate and verify proofs of executing Noir programs, following Noir's tooling that compiles and executes the programs. Read the [proving backend installation](../backend/index.md) section to learn more about proving backends and how to install them. - -Barretenberg is used as an example here to demonstrate how proving and verifying could be implemented and used. Read the [`bb` installation](../backend/index.md#example-installing-bb) section for how to install Barretenberg's CLI tool; refer to [`bb`'s documentation](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/cpp/src/barretenberg/bb/readme.md) for full details about the tool. - -### 5. Prove an execution of the Noir program - -Using Barretenberg as an example, prove the valid execution of your Noir program running: - -```sh -bb prove -b ./target/hello_world.json -w ./target/witness-name.gz -o ./target/proof -``` - -The proof generated will then be written to the file `./target/proof`. - -### 6. Verify the execution proof - -Once a proof is generated, we can verify correct execution of our Noir program by verifying the proof file. - -Using Barretenberg as an example, compute the verification key for the Noir program by running: - -```sh -bb write_vk -b ./target/hello_world.json -o ./target/vk -``` - -And verify your proof by running: - -```sh -bb verify -k ./target/vk -p ./target/proof -``` - -If successful, the verification will complete in silence; if unsuccessful, the command will trigger logging of the corresponding error. - -Congratulations, you have now created and verified a proof for your very first Noir program! - -In the [next section](./project_breakdown.md), we will go into more detail on each step performed. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/project_breakdown.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/project_breakdown.md deleted file mode 100644 index 96e653f6c08..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/hello_noir/project_breakdown.md +++ /dev/null @@ -1,159 +0,0 @@ ---- -title: Project Breakdown -description: - Learn about the anatomy of a Nargo project, including the purpose of the Prover TOML - file, and how to prove and verify your program. -keywords: - [Nargo, Nargo project, Prover.toml, proof verification, private asset transfer] -sidebar_position: 2 ---- - -This section breaks down our hello world program from the previous section. - -## Anatomy of a Nargo Project - -Upon creating a new project with `nargo new` and building the in/output files with `nargo check` -commands, you would get a minimal Nargo project of the following structure: - - - src - - Prover.toml - - Nargo.toml - -The source directory _src_ holds the source code for your Noir program. By default only a _main.nr_ -file will be generated within it. - -### Prover.toml - -_Prover.toml_ is used for specifying the input values for executing and proving the program. You can specify `toml` files with different names by using the `--prover-name` or `-p` flags, see the [Prover](#provertoml) section below. Optionally you may specify expected output values for prove-time checking as well. - -### Nargo.toml - -_Nargo.toml_ contains the environmental options of your project. It contains a "package" section and a "dependencies" section. - -Example Nargo.toml: - -```toml -[package] -name = "noir_starter" -type = "bin" -authors = ["Alice"] -compiler_version = "0.9.0" -description = "Getting started with Noir" -entry = "circuit/main.nr" -license = "MIT" - -[dependencies] -ecrecover = {tag = "v0.9.0", git = "https://github.com/colinnielsen/ecrecover-noir.git"} -``` - -Nargo.toml for a [workspace](../../noir/modules_packages_crates/workspaces.md) will look a bit different. For example: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -#### Package section - -The package section defines a number of fields including: - -- `name` (**required**) - the name of the package -- `type` (**required**) - can be "bin", "lib", or "contract" to specify whether its a binary, library or Aztec contract -- `authors` (optional) - authors of the project -- `compiler_version` - specifies the version of the compiler to use. This is enforced by the compiler and follow's [Rust's versioning](https://doc.rust-lang.org/cargo/reference/manifest.html#the-version-field), so a `compiler_version = 0.18.0` will enforce Nargo version 0.18.0, `compiler_version = ^0.18.0` will enforce anything above 0.18.0 but below 0.19.0, etc. For more information, see how [Rust handles these operators](https://docs.rs/semver/latest/semver/enum.Op.html) -- `description` (optional) -- `entry` (optional) - a relative filepath to use as the entry point into your package (overrides the default of `src/lib.nr` or `src/main.nr`) -- `backend` (optional) -- `license` (optional) -- `expression_width` (optional) - Sets the default backend expression width. This field will override the default backend expression width specified by the Noir compiler (currently set to width 4). - -#### Dependencies section - -This is where you will specify any dependencies for your project. See the [Dependencies page](../../noir/modules_packages_crates/dependencies.md) for more info. - -`./proofs/` and `./contract/` directories will not be immediately visible until you create a proof or -verifier contract respectively. - -### main.nr - -The _main.nr_ file contains a `main` method, this method is the entry point into your Noir program. - -In our sample program, _main.nr_ looks like this: - -```rust -fn main(x : Field, y : Field) { - assert(x != y); -} -``` - -The parameters `x` and `y` can be seen as the API for the program and must be supplied by the prover. Since neither `x` nor `y` is marked as public, the verifier does not supply any inputs, when verifying the proof. - -The prover supplies the values for `x` and `y` in the _Prover.toml_ file. - -As for the program body, `assert` ensures that the condition to be satisfied (e.g. `x != y`) is constrained by the proof of the execution of said program (i.e. if the condition was not met, the verifier would reject the proof as an invalid proof). - -### Prover.toml - -The _Prover.toml_ file is a file which the prover uses to supply the inputs to the Noir program (both private and public). - -In our hello world program the _Prover.toml_ file looks like this: - -```toml -x = "1" -y = "2" -``` - -When the command `nargo execute` is executed, nargo will execute the Noir program using the inputs specified in `Prover.toml`, aborting if it finds that these do not satisfy the constraints defined by `main`. In this example, `x` and `y` must satisfy the inequality constraint `assert(x != y)`. - -If an output name is specified such as `nargo execute foo`, the witness generated by this execution will be written to `./target/foo.gz`. This can then be used to generate a proof of the execution. - -#### Arrays of Structs - -The following code shows how to pass an array of structs to a Noir program to generate a proof. - -```rust -// main.nr -struct Foo { - bar: Field, - baz: Field, -} - -fn main(foos: [Foo; 3]) -> pub Field { - foos[2].bar + foos[2].baz -} -``` - -Prover.toml: - -```toml -[[foos]] # foos[0] -bar = 0 -baz = 0 - -[[foos]] # foos[1] -bar = 0 -baz = 0 - -[[foos]] # foos[2] -bar = 1 -baz = 2 -``` - -#### Custom toml files - -You can specify a `toml` file with a different name to use for execution by using the `--prover-name` or `-p` flags. - -This command looks for proof inputs in the default **Prover.toml** and generates the witness and saves it at `./target/foo.gz`: - -```bash -nargo execute foo -``` - -This command looks for proof inputs in the custom **OtherProver.toml** and generates the witness and saves it at `./target/bar.gz`: - -```bash -nargo execute -p OtherProver bar -``` - -Now that you understand the concepts, you'll probably want some editor feedback while you are writing more complex code. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/_category_.json deleted file mode 100644 index 0c02fb5d4d7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "position": 0, - "label": "Install Nargo", - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/index.md deleted file mode 100644 index 53ea9c7891c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/index.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Nargo Installation -description: - nargo is a command line tool for interacting with Noir programs. This page is a quick guide on how to install Nargo through the most common and easy method, noirup -keywords: [ - Nargo - Noir - Rust - Cargo - Noirup - Installation - Terminal Commands - Version Check - Nightlies - Specific Versions - Branches - Noirup Repository -] -pagination_next: getting_started/hello_noir/index ---- - -`nargo` is a tool for working with Noir programs on the CLI, providing you with the ability to start new projects, compile, execute and test Noir programs from the terminal. - -The name is inspired by Rust's package manager `cargo`; and similar to Rust's `rustup`, Noir also has an easy installation script `noirup`. - -## Installing Noirup - -Open a terminal on your machine, and write: - -```bash -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Close the terminal, open another one, and run - -```bash -noirup -``` - -Done. That's it. You should have the latest version working. You can check with `nargo --version`. - -You can also install nightlies, specific versions -or branches. Check out the [noirup repository](https://github.com/noir-lang/noirup) for more -information. - -Now we're ready to start working on [our first Noir program!](../hello_noir/index.md) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/other_install_methods.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/other_install_methods.md deleted file mode 100644 index 3634723562b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/getting_started/installation/other_install_methods.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Alternative Installations -description: There are different ways to install Nargo, the one-stop shop and command-line tool for developing Noir programs. This guide explains how to specify which version to install when using noirup, and using WSL for windows. -keywords: [ - Installation - Nargo - Noirup - Binaries - Compiling from Source - WSL for Windows - macOS - Linux - Nix - Direnv - Uninstalling Nargo - ] -sidebar_position: 1 ---- - -## Encouraged Installation Method: Noirup - -Noirup is the endorsed method for installing Nargo, streamlining the process of fetching binaries or compiling from source. It supports a range of options to cater to your specific needs, from nightly builds and specific versions to compiling from various sources. - -### Installing Noirup - -First, ensure you have `noirup` installed: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -### Fetching Binaries - -With `noirup`, you can easily switch between different Nargo versions, including nightly builds: - -- **Nightly Version**: Install the latest nightly build. - - ```sh - noirup --version nightly - ``` - -- **Specific Version**: Install a specific version of Nargo. - ```sh - noirup --version - ``` - -### Compiling from Source - -`noirup` also enables compiling Nargo from various sources: - -- **From a Specific Branch**: Install from the latest commit on a branch. - - ```sh - noirup --branch - ``` - -- **From a Fork**: Install from the main branch of a fork. - - ```sh - noirup --repo - ``` - -- **From a Specific Branch in a Fork**: Install from a specific branch in a fork. - - ```sh - noirup --repo --branch - ``` - -- **From a Specific Pull Request**: Install from a specific PR. - - ```sh - noirup --pr - ``` - -- **From a Specific Commit**: Install from a specific commit. - - ```sh - noirup -C - ``` - -- **From Local Source**: Compile and install from a local directory. - ```sh - noirup --path ./path/to/local/source - ``` - -## Installation on Windows - -The default backend for Noir (Barretenberg) doesn't provide Windows binaries at this time. For that reason, Noir cannot be installed natively. However, it is available by using Windows Subsystem for Linux (WSL). - -Step 1: Follow the instructions [here](https://learn.microsoft.com/en-us/windows/wsl/install) to install and run WSL. - -step 2: Follow the [Noirup instructions](#encouraged-installation-method-noirup). - -## Uninstalling Nargo - -If you installed Nargo with `noirup`, you can uninstall Nargo by removing the files in `~/.nargo`, `~/nargo`, and `~/noir_cache`. This ensures that all installed binaries, configurations, and cache related to Nargo are fully removed from your system. - -```bash -rm -r ~/.nargo -rm -r ~/nargo -rm -r ~/noir_cache -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/_category_.json deleted file mode 100644 index 23b560f610b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/_category_.json deleted file mode 100644 index cc2cbb1c253..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugging", - "position": 5, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/debugging_with_the_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/debugging_with_the_repl.md deleted file mode 100644 index 1d64dae3f37..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/debugging_with_the_repl.md +++ /dev/null @@ -1,164 +0,0 @@ ---- -title: Using the REPL Debugger -description: - Step-by-step guide on how to debug your Noir circuits with the REPL Debugger. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -#### Pre-requisites - -In order to use the REPL debugger, first you need to install recent enough versions of Nargo and vscode-noir. - -## Debugging a simple circuit - -Let's debug a simple circuit: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -To start the REPL debugger, using a terminal, go to a Noir circuit's home directory. Then: - -`$ nargo debug` - -You should be seeing this in your terminal: - -``` -[main] Starting debugger -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:9 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> -``` - -The debugger displays the current Noir code location, and it is now waiting for us to drive it. - -Let's first take a look at the available commands. For that we'll use the `help` command. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -Some commands operate only for unconstrained functions, such as `memory` and `memset`. If you try to use them while execution is paused at an ACIR opcode, the debugger will simply inform you that you are not executing unconstrained code: - -``` -> memory -Unconstrained VM memory not available -> -``` - -Before continuing, we can take a look at the initial witness map: - -``` -> witness -_0 = 1 -_1 = 2 -> -``` - -Cool, since `x==1`, `y==2`, and we want to check that `x != y`, our circuit should succeed. At this point we could intervene and use the witness setter command to change one of the witnesses. Let's set `y=3`, then back to 2, so we don't affect the expected result: - -``` -> witness -_0 = 1 -_1 = 2 -> witness 1 3 -_1 = 3 -> witness -_0 = 1 -_1 = 3 -> witness 1 2 -_1 = 2 -> witness -_0 = 1 -_1 = 2 -> -``` - -Now we can inspect the current state of local variables. For that we use the `vars` command. - -``` -> vars -> -``` - -We currently have no vars in context, since we are at the entry point of the program. Let's use `next` to execute until the next point in the program. - -``` -> vars -> next -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:20 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> vars -x:Field = 0x01 -``` - -As a result of stepping, the variable `x`, whose initial value comes from the witness map, is now in context and returned by `vars`. - -``` -> next - 1 fn main(x : Field, y : pub Field) { - 2 -> assert(x != y); - 3 } -> vars -y:Field = 0x02 -x:Field = 0x01 -``` - -Stepping again we can finally see both variables and their values. And now we can see that the next assertion should succeed. - -Let's continue to the end: - -``` -> continue -(Continuing execution...) -Finished execution -> q -[main] Circuit witness successfully solved -``` - -Upon quitting the debugger after a solved circuit, the resulting circuit witness gets saved, equivalent to what would happen if we had run the same circuit with `nargo execute`. - -We just went through the basics of debugging using Noir REPL debugger. For a comprehensive reference, check out [the reference page](../../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/debugging_with_vs_code.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/debugging_with_vs_code.md deleted file mode 100644 index a5858c1a5eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/debugger/debugging_with_vs_code.md +++ /dev/null @@ -1,68 +0,0 @@ ---- -title: Using the VS Code Debugger -description: - Step by step guide on how to debug your Noir circuits with the VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -This guide will show you how to use VS Code with the vscode-noir extension to debug a Noir project. - -#### Pre-requisites - -- Nargo -- vscode-noir -- A Noir project with a `Nargo.toml`, `Prover.toml` and at least one Noir (`.nr`) containing an entry point function (typically `main`). - -## Running the debugger - -The easiest way to start debugging is to open the file you want to debug, and press `F5`. This will cause the debugger to launch, using your `Prover.toml` file as input. - -You should see something like this: - -![Debugger launched](@site/static/img/debugger/1-started.png) - -Let's inspect the state of the program. For that, we open VS Code's _Debug pane_. Look for this icon: - -![Debug pane icon](@site/static/img/debugger/2-icon.png) - -You will now see two categories of variables: Locals and Witness Map. - -![Debug pane expanded](@site/static/img/debugger/3-debug-pane.png) - -1. **Locals**: variables of your program. At this point in execution this section is empty, but as we step through the code it will get populated by `x`, `result`, `digest`, etc. - -2. **Witness map**: these are initially populated from your project's `Prover.toml` file. In this example, they will be used to populate `x` and `result` at the beginning of the `main` function. - -Most of the time you will probably be focusing mostly on locals, as they represent the high level state of your program. - -You might be interested in inspecting the witness map in case you are trying to solve a really low level issue in the compiler or runtime itself, so this concerns mostly advanced or niche users. - -Let's step through the program, by using the debugger buttons or their corresponding keyboard shortcuts. - -![Debugger buttons](@site/static/img/debugger/4-debugger-buttons.png) - -Now we can see in the variables pane that there's values for `digest`, `result` and `x`. - -![Inspecting locals](@site/static/img/debugger/5-assert.png) - -We can also inspect the values of variables by directly hovering on them on the code. - -![Hover locals](@site/static/img/debugger/6-hover.png) - -Let's set a break point at the `keccak256` function, so we can continue execution up to the point when it's first invoked without having to go one step at a time. - -We just need to click the to the right of the line number 18. Once the breakpoint appears, we can click the `continue` button or use its corresponding keyboard shortcut (`F5` by default). - -![Breakpoint](@site/static/img/debugger/7-break.png) - -Now we are debugging the `keccak256` function, notice the _Call Stack pane_ at the lower right. This lets us inspect the current call stack of our process. - -That covers most of the current debugger functionalities. Check out [the reference](../../reference/debugger/debugger_vscode.md) for more details on how to configure the debugger. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-oracles.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-oracles.md deleted file mode 100644 index 392dd8b452e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-oracles.md +++ /dev/null @@ -1,275 +0,0 @@ ---- -title: How to use Oracles -description: Learn how to use oracles in your Noir program with examples in both Nargo and NoirJS. This guide also covers writing a JSON RPC server and providing custom foreign call handlers for NoirJS. -keywords: - - Noir Programming - - Oracles - - Nargo - - NoirJS - - JSON RPC Server - - Foreign Call Handlers -sidebar_position: 1 ---- - -This guide shows you how to use oracles in your Noir program. For the sake of clarity, it assumes that: - -- You have read the [explainer on Oracles](../explainers/explainer-oracle.md) and are comfortable with the concept. -- You have a Noir program to add oracles to. You can create one using the [vite-hardhat starter](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) as a boilerplate. -- You understand the concept of a JSON-RPC server. Visit the [JSON-RPC website](https://www.jsonrpc.org/) if you need a refresher. -- You are comfortable with server-side JavaScript (e.g. Node.js, managing packages, etc.). - -## Rundown - -This guide has 3 major steps: - -1. How to modify our Noir program to make use of oracle calls as unconstrained functions -2. How to write a JSON RPC Server to resolve these oracle calls with Nargo -3. How to use them in Nargo and how to provide a custom resolver in NoirJS - -## Step 1 - Modify your Noir program - -An oracle is defined in a Noir program by defining two methods: - -- An unconstrained method - This tells the compiler that it is executing an [unconstrained functions](../noir/concepts//unconstrained.md). -- A decorated oracle method - This tells the compiler that this method is an RPC call. - -An example of an oracle that returns a `Field` would be: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(number: Field) -> Field { } - -unconstrained fn get_sqrt(number: Field) -> Field { - sqrt(number) -} -``` - -In this example, we're wrapping our oracle function in an unconstrained method, and decorating it with `oracle(getSqrt)`. We can then call the unconstrained function as we would call any other function: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); -} -``` - -In the next section, we will make this `getSqrt` (defined on the `sqrt` decorator) be a method of the RPC server Noir will use. - -:::danger - -As explained in the [Oracle Explainer](../explainers/explainer-oracle.md), this `main` function is unsafe unless you constrain its return value. For example: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); - assert(sqrt.pow_32(2) as u64 == input as u64); // <---- constrain the return of an oracle! -} -``` - -::: - -:::info - -Currently, oracles only work with single params or array params. For example: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt([Field; 2]) -> [Field; 2] { } -``` - -::: - -## Step 2 - Write an RPC server - -Brillig will call *one* RPC server. Most likely you will have to write your own, and you can do it in whatever language you prefer. In this guide, we will do it in Javascript. - -Let's use the above example of an oracle that consumes an array with two `Field` and returns their square roots: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(input: [Field; 2]) -> [Field; 2] { } - -unconstrained fn get_sqrt(input: [Field; 2]) -> [Field; 2] { - sqrt(input) -} - -fn main(input: [Field; 2]) { - let sqrt = get_sqrt(input); - assert(sqrt[0].pow_32(2) as u64 == input[0] as u64); - assert(sqrt[1].pow_32(2) as u64 == input[1] as u64); -} - -#[test] -fn test() { - let input = [4, 16]; - main(input); -} -``` - -:::info - -Why square root? - -In general, computing square roots is computationally more expensive than multiplications, which takes a toll when speaking about ZK applications. In this case, instead of calculating the square root in Noir, we are using our oracle to offload that computation to be made in plain. In our circuit we can simply multiply the two values. - -::: - -Now, we should write the correspondent RPC server, starting with the [default JSON-RPC 2.0 boilerplate](https://www.npmjs.com/package/json-rpc-2.0#example): - -```js -import { JSONRPCServer } from "json-rpc-2.0"; -import express from "express"; -import bodyParser from "body-parser"; - -const app = express(); -app.use(bodyParser.json()); - -const server = new JSONRPCServer(); -app.post("/", (req, res) => { - const jsonRPCRequest = req.body; - server.receive(jsonRPCRequest).then((jsonRPCResponse) => { - if (jsonRPCResponse) { - res.json(jsonRPCResponse); - } else { - res.sendStatus(204); - } - }); -}); - -app.listen(5555); -``` - -Now, we will add our `getSqrt` method, as expected by the `#[oracle(getSqrt)]` decorator in our Noir code. It maps through the params array and returns their square roots: - -```js -server.addMethod("resolve_foreign_call", async (params) => { - if (params[0].function !== "getSqrt") { - throw Error("Unexpected foreign call") - }; - const values = params[0].inputs[0].map((field) => { - return `${Math.sqrt(parseInt(field, 16))}`; - }); - return { values: [values] }; -}); -``` - -If you're using Typescript, the following types may be helpful in understanding the expected return value and making sure they're easy to follow: - -```js -export type ForeignCallSingle = string; - -export type ForeignCallArray = string[]; - -export type ForeignCallResult = { - values: (ForeignCallSingle | ForeignCallArray)[]; -}; -``` - -:::info Multidimensional Arrays - -If the Oracle function is returning an array containing other arrays, such as `[['1','2],['3','4']]`, you need to provide the values in JSON as flattened values. In the previous example, it would be `['1', '2', '3', '4']`. In the Noir program, the Oracle signature can use a nested type, the flattened values will be automatically converted to the nested type. - -::: - -## Step 3 - Usage with Nargo - -Using the [`nargo` CLI tool](../getting_started/installation/index.md), you can use oracles in the `nargo test` and `nargo execute` commands by passing a value to `--oracle-resolver`. For example: - -```bash -nargo test --oracle-resolver http://localhost:5555 -``` - -This tells `nargo` to use your RPC Server URL whenever it finds an oracle decorator. - -## Step 4 - Usage with NoirJS - -In a JS environment, an RPC server is not strictly necessary, as you may want to resolve your oracles without needing any JSON call at all. NoirJS simply expects that you pass a callback function when you generate proofs, and that callback function can be anything. - -For example, if your Noir program expects the host machine to provide CPU pseudo-randomness, you could simply pass it as the `foreignCallHandler`. You don't strictly need to create an RPC server to serve pseudo-randomness, as you may as well get it directly in your app: - -```js -const foreignCallHandler = (name, inputs) => crypto.randomBytes(16) // etc - -await noir.execute(inputs, foreignCallHandler) -``` - -As one can see, in NoirJS, the [`foreignCallHandler`](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) function simply means "a callback function that returns a value of type [`ForeignCallOutput`](../reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md). It doesn't have to be an RPC call like in the case for Nargo. - -:::tip - -Does this mean you don't have to write an RPC server like in [Step #2](#step-2---write-an-rpc-server)? - -You don't technically have to, but then how would you run `nargo test`? To use both `Nargo` and `NoirJS` in your development flow, you will have to write a JSON RPC server. - -::: - -In this case, let's make `foreignCallHandler` call the JSON RPC Server we created in [Step #2](#step-2---write-an-rpc-server), by making it a JSON RPC Client. - -For example, using the same `getSqrt` program in [Step #1](#step-1---modify-your-noir-program) (comments in the code): - -```js -import { JSONRPCClient } from "json-rpc-2.0"; - -// declaring the JSONRPCClient -const client = new JSONRPCClient((jsonRPCRequest) => { -// hitting the same JSON RPC Server we coded above - return fetch("http://localhost:5555", { - method: "POST", - headers: { - "content-type": "application/json", - }, - body: JSON.stringify(jsonRPCRequest), - }).then((response) => { - if (response.status === 200) { - return response - .json() - .then((jsonRPCResponse) => client.receive(jsonRPCResponse)); - } else if (jsonRPCRequest.id !== undefined) { - return Promise.reject(new Error(response.statusText)); - } - }); -}); - -// declaring a function that takes the name of the foreign call (getSqrt) and the inputs -const foreignCallHandler = async (name, input) => { - const inputs = input[0].map((i) => i.toString("hex")) - // notice that the "inputs" parameter contains *all* the inputs - // in this case we to make the RPC request with the first parameter "numbers", which would be input[0] - const oracleReturn = await client.request("resolve_foreign_call", [ - { - function: name, - inputs: [inputs] - }, - ]); - return [oracleReturn.values[0]]; -}; - -// the rest of your NoirJS code -const input = { input: [4, 16] }; -const { witness } = await noir.execute(input, foreignCallHandler); -``` - -:::tip - -If you're in a NoirJS environment running your RPC server together with a frontend app, you'll probably hit a familiar problem in full-stack development: requests being blocked by [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) policy. For development only, you can simply install and use the [`cors` npm package](https://www.npmjs.com/package/cors) to get around the problem: - -```bash -yarn add cors -``` - -and use it as a middleware: - -```js -import cors from "cors"; - -const app = express(); -app.use(cors()) -``` - -::: - -## Conclusion - -Hopefully by the end of this guide, you should be able to: - -- Write your own logic around Oracles and how to write a JSON RPC server to make them work with your Nargo commands. -- Provide custom foreign call handlers for NoirJS. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-recursion.md deleted file mode 100644 index c8c4dc9f5b4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-recursion.md +++ /dev/null @@ -1,180 +0,0 @@ ---- -title: How to use recursion on NoirJS -description: Learn how to implement recursion with NoirJS, a powerful tool for creating smart contracts on the EVM blockchain. This guide assumes familiarity with NoirJS, solidity verifiers, and the Barretenberg proving backend. Discover how to generate both final and intermediate proofs using `noir_js` and `backend_barretenberg`. -keywords: - [ - "NoirJS", - "EVM blockchain", - "smart contracts", - "recursion", - "solidity verifiers", - "Barretenberg backend", - "noir_js", - "backend_barretenberg", - "intermediate proofs", - "final proofs", - "nargo compile", - "json import", - "recursive circuit", - "recursive app" - ] -sidebar_position: 1 ---- - -This guide shows you how to use recursive proofs in your NoirJS app. For the sake of clarity, it is assumed that: - -- You already have a NoirJS app. If you don't, please visit the [NoirJS tutorial](../tutorials/noirjs_app.md) and the [reference](../reference/NoirJS/noir_js/index.md). -- You are familiar with what are recursive proofs and you have read the [recursion explainer](../explainers/explainer-recursion.md) -- You already built a recursive circuit following [the reference](../noir/standard_library/recursion.mdx), and understand how it works. - -It is also assumed that you're not using `noir_wasm` for compilation, and instead you've used [`nargo compile`](../reference/nargo_commands.md) to generate the `json` you're now importing into your project. However, the guide should work just the same if you're using `noir_wasm`. - -:::info - -As you've read in the [explainer](../explainers/explainer-recursion.md), a recursive proof is an intermediate proof. This means that it doesn't necessarily generate the final step that makes it verifiable in a smart contract. However, it is easy to verify within another circuit. - -While "standard" usage of NoirJS packages abstracts final proofs, it currently lacks the necessary interface to abstract away intermediate proofs. This means that these proofs need to be created by using the backend directly. - -In short: - -- `noir_js` generates *only* final proofs -- `backend_barretenberg` generates both types of proofs - -::: - -In a standard recursive app, you're also dealing with at least two circuits. For the purpose of this guide, we will assume the following: - -- `main`: a circuit of type `assert(x != y)`, where `main` is marked with a `#[recursive]` attribute. This attribute states that the backend should generate proofs that are friendly for verification within another circuit. -- `recursive`: a circuit that verifies `main` - -For a full example of how recursive proofs work, please refer to the [noir-examples](https://github.com/noir-lang/noir-examples) repository. We will *not* be using it as a reference for this guide. - -## Step 1: Setup - -In a common NoirJS app, you need to instantiate a backend with something like `const backend = new Backend(circuit)`. Then you feed it to the `noir_js` interface. - -For recursion, this doesn't happen, and the only need for `noir_js` is only to `execute` a circuit and get its witness and return value. Everything else is not interfaced, so it needs to happen on the `backend` object. - -It is also recommended that you instantiate the backend with as many threads as possible, to allow for maximum concurrency: - -```js -const backend = new Backend(circuit, { threads: 8 }) -``` - -:::tip -You can use the [`os.cpus()`](https://nodejs.org/api/os.html#oscpus) object in `nodejs` or [`navigator.hardwareConcurrency`](https://developer.mozilla.org/en-US/docs/Web/API/Navigator/hardwareConcurrency) on the browser to make the most out of those glorious cpu cores -::: - -## Step 2: Generating the witness and the proof for `main` - -After instantiating the backend, you should also instantiate `noir_js`. We will use it to execute the circuit and get the witness. - -```js -const noir = new Noir(circuit) -const { witness } = noir.execute(input) -``` - -With this witness, you are now able to generate the intermediate proof for the main circuit: - -```js -const { proof, publicInputs } = await backend.generateProof(witness) -``` - -:::warning - -Always keep in mind what is actually happening on your development process, otherwise you'll quickly become confused about what circuit we are actually running and why! - -In this case, you can imagine that Alice (running the `main` circuit) is proving something to Bob (running the `recursive` circuit), and Bob is verifying her proof within his proof. - -With this in mind, it becomes clear that our intermediate proof is the one *meant to be verified within another circuit*, so it must be Alice's. Actually, the only final proof in this theoretical scenario would be the last one, sent on-chain. - -::: - -## Step 3 - Verification and proof artifacts - -Optionally, you are able to verify the intermediate proof: - -```js -const verified = await backend.verifyProof({ proof, publicInputs }) -``` - -This can be useful to make sure our intermediate proof was correctly generated. But the real goal is to do it within another circuit. For that, we need to generate recursive proof artifacts that will be passed to the circuit that is verifying the proof we just generated. Instead of passing the proof and verification key as a byte array, we pass them as fields which makes it cheaper to verify in a circuit: - -```js -const { proofAsFields, vkAsFields, vkHash } = await backend.generateRecursiveProofArtifacts( { publicInputs, proof }, publicInputsCount) -``` - -This call takes the public inputs and the proof, but also the public inputs count. While this is easily retrievable by simply counting the `publicInputs` length, the backend interface doesn't currently abstract it away. - -:::info - -The `proofAsFields` has a constant size `[Field; 93]` and verification keys in Barretenberg are always `[Field; 114]`. - -::: - -:::warning - -One common mistake is to forget *who* makes this call. - -In a situation where Alice is generating the `main` proof, if she generates the proof artifacts and sends them to Bob, which gladly takes them as true, this would mean Alice could prove anything! - -Instead, Bob needs to make sure *he* extracts the proof artifacts, using his own instance of the `main` circuit backend. This way, Alice has to provide a valid proof for the correct `main` circuit. - -::: - -## Step 4 - Recursive proof generation - -With the artifacts, generating a recursive proof is no different from a normal proof. You simply use the `backend` (with the recursive circuit) to generate it: - -```js -const recursiveInputs = { - verification_key: vkAsFields, // array of length 114 - proof: proofAsFields, // array of length 93 + size of public inputs - publicInputs: [mainInput.y], // using the example above, where `y` is the only public input - key_hash: vkHash, -} - -const { witness, returnValue } = noir.execute(recursiveInputs) // we're executing the recursive circuit now! -const { proof, publicInputs } = backend.generateProof(witness) -const verified = backend.verifyProof({ proof, publicInputs }) -``` - -You can obviously chain this proof into another proof. In fact, if you're using recursive proofs, you're probably interested of using them this way! - -:::tip - -Managing circuits and "who does what" can be confusing. To make sure your naming is consistent, you can keep them in an object. For example: - -```js -const circuits = { - main: mainJSON, - recursive: recursiveJSON -} -const backends = { - main: new BarretenbergBackend(circuits.main), - recursive: new BarretenbergBackend(circuits.recursive) -} -const noir_programs = { - main: new Noir(circuits.main), - recursive: new Noir(circuits.recursive) -} -``` - -This allows you to neatly call exactly the method you want without conflicting names: - -```js -// Alice runs this 👇 -const { witness: mainWitness } = await noir_programs.main.execute(input) -const proof = await backends.main.generateProof(mainWitness) - -// Bob runs this 👇 -const verified = await backends.main.verifyProof(proof) -const { proofAsFields, vkAsFields, vkHash } = await backends.main.generateRecursiveProofArtifacts( - proof, - numPublicInputs, -); -const { witness: recursiveWitness } = await noir_programs.recursive.execute(recursiveInputs) -const recursiveProof = await backends.recursive.generateProof(recursiveWitness); -``` - -::: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-solidity-verifier.md deleted file mode 100644 index 063538f1fc2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/how-to-solidity-verifier.md +++ /dev/null @@ -1,253 +0,0 @@ ---- -title: Generate a Solidity Verifier -description: - Learn how to run the verifier as a smart contract on the blockchain. Compile a Solidity verifier - contract for your Noir program and deploy it on any EVM blockchain acting as a verifier smart - contract. Read more to find out -keywords: - [ - solidity verifier, - smart contract, - blockchain, - compiler, - plonk_vk.sol, - EVM blockchain, - verifying Noir programs, - proving backend, - Barretenberg, - ] -sidebar_position: 0 -pagination_next: tutorials/noirjs_app ---- - -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. - -This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. - -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: - -- You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network -- You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/installation/index.md) with Nargo and the example Hello Noir circuit -- You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. - -## Rundown - -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: - -1. How to generate a solidity smart contract -2. How to compile the smart contract in the RemixIDE -3. How to deploy it to a testnet - -## Step 1 - Generate a contract - -This is by far the most straightforward step. Just run: - -```sh -nargo compile -``` - -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: - -```sh -# Here we pass the path to the newly generated Noir artifact. -bb write_vk -b ./target/.json -bb contract -``` - -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. - -You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/hello_noir/index.md#proving-backend). - -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. - -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: - -## Step 2 - Compiling - -We will mostly skip the details of RemixIDE, as the UI can change from version to version. For now, we can just open -Remix and create a blank workspace. - -![Create Workspace](@site/static/img/how-tos/solidity_verifier_1.png) - -We will create a new file to contain the contract Nargo generated, and copy-paste its content. - -:::warning - -You'll likely see a warning advising you to not trust pasted code. While it is an important warning, it is irrelevant in the context of this guide and can be ignored. We will not be deploying anywhere near a mainnet. - -::: - -To compile our the verifier, we can navigate to the compilation tab: - -![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) - -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: - -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) - -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: - -![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) - -## Step 3 - Deploying - -At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. - -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": - -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) - -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. - -:::note - -Why "UltraVerifier"? - -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. - -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: - -## Step 4 - Verifying - -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: - -```solidity -function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) -``` - -When using the default example in the [Hello Noir](../getting_started/hello_noir/index.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. - -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/hello_noir/index.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the - -```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') - -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS - -echo "Proof:" -echo "0x$HEX_PROOF" -``` - -Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. - -A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): - -```solidity -function castVote(bytes calldata proof, uint proposalId, uint vote, bytes32 nullifierHash) public returns (bool) { - // ... - bytes32[] memory publicInputs = new bytes32[](4); - publicInputs[0] = merkleRoot; - publicInputs[1] = bytes32(proposalId); - publicInputs[2] = bytes32(vote); - publicInputs[3] = nullifierHash; - require(verifier.verify(proof, publicInputs), "Invalid proof"); -``` - -:::info[Return Values] - -A circuit doesn't have the concept of a return value. Return values are just syntactic sugar in Noir. - -Under the hood, the return value is passed as an input to the circuit and is checked at the end of the circuit program. - -For example, if you have Noir program like this: - -```rust -fn main( - // Public inputs - pubkey_x: pub Field, - pubkey_y: pub Field, - // Private inputs - priv_key: Field, -) -> pub Field -``` - -the `verify` function will expect the public inputs array (second function parameter) to be of length 3, the two inputs and the return value. - -Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. - -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. - -::: - -:::tip[Structs] - -You can pass structs to the verifier contract. They will be flattened so that the array of inputs is 1-dimensional array. - -For example, consider the following program: - -```rust -struct Type1 { - val1: Field, - val2: Field, -} - -struct Nested { - t1: Type1, - is_true: bool, -} - -fn main(x: pub Field, nested: pub Nested, y: pub Field) { - //... -} -``` - -The order of these inputs would be flattened to: `[x, nested.t1.val1, nested.t1.val2, nested.is_true, y]` - -::: - -The other function you can call is our entrypoint `verify` function, as defined above. - -:::tip - -It's worth noticing that the `verify` function is actually a `view` function. A `view` function does not alter the blockchain state, so it doesn't need to be distributed (i.e. it will run only on the executing node), and therefore doesn't cost any gas. - -This can be particularly useful in some situations. If Alice generated a proof and wants Bob to verify its correctness, Bob doesn't need to run Nargo, NoirJS, or any Noir specific infrastructure. He can simply make a call to the blockchain with the proof and verify it is correct without paying any gas. - -It would be incorrect to say that a Noir proof verification costs any gas at all. However, most of the time the result of `verify` is used to modify state (for example, to update a balance, a game state, etc). In that case the whole network needs to execute it, which does incur gas costs (calldata and execution, but not storage). - -::: - -## A Note on EVM chains - -Noir proof verification requires the ecMul, ecAdd and ecPairing precompiles. Not all EVM chains support EC Pairings, notably some of the ZK-EVMs. This means that you won't be able to use the verifier contract in all of them. You can find an incomplete list of which EVM chains support these precompiles [here](https://www.evmdiff.com/features?feature=precompiles). - -For example, chains like `zkSync ERA` and `Polygon zkEVM` do not currently support these precompiles, so proof verification via Solidity verifier contracts won't work. Here's a quick list of EVM chains that have been tested and are known to work: - -- Optimism -- Arbitrum -- Polygon PoS -- Scroll -- Celo - -If you test any other chains, please open a PR on this page to update the list. See [this doc](https://github.com/noir-lang/noir-starter/tree/main/with-foundry#testing-on-chain) for more info about testing verifier contracts on different EVM chains. - -## What's next - -Now that you know how to call a Noir Solidity Verifier on a smart contract using Remix, you should be comfortable with using it with some programmatic frameworks, such as [hardhat](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) and [foundry](https://github.com/noir-lang/noir-starter/tree/main/with-foundry). - -You can find other tools, examples, boilerplates and libraries in the [awesome-noir](https://github.com/noir-lang/awesome-noir) repository. - -You should also be ready to write and deploy your first NoirJS app and start generating proofs on websites, phones, and NodeJS environments! Head on to the [NoirJS tutorial](../tutorials/noirjs_app.md) to learn how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/merkle-proof.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/merkle-proof.mdx deleted file mode 100644 index 0a128adb2de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/merkle-proof.mdx +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Prove Merkle Tree Membership -description: - Learn how to use merkle membership proof in Noir to prove that a given leaf is a member of a - merkle tree with a specified root, at a given index. -keywords: - [merkle proof, merkle membership proof, Noir, rust, hash function, Pedersen, sha256, merkle tree] -sidebar_position: 4 ---- - -Let's walk through an example of a merkle membership proof in Noir that proves that a given leaf is -in a merkle tree. - -```rust - -fn main(message : [Field; 62], index : Field, hashpath : [Field; 40], root : Field) { - let leaf = std::hash::hash_to_field(message.as_slice()); - let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); - assert(merkle_root == root); -} - -``` - -The message is hashed using `hash_to_field`. The specific hash function that is being used is chosen -by the backend. The only requirement is that this hash function can heuristically be used as a -random oracle. If only collision resistance is needed, then one can call `std::hash::pedersen_hash` -instead. - -```rust -let leaf = std::hash::hash_to_field(message.as_slice()); -``` - -The leaf is then passed to a compute_merkle_root function with the root, index and hashpath. The returned root can then be asserted to be the same as the provided root. - -```rust -let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); -assert (merkle_root == root); -``` - -> **Note:** It is possible to re-implement the merkle tree implementation without standard library. -> However, for most usecases, it is enough. In general, the standard library will always opt to be -> as conservative as possible, while striking a balance with efficiency. - -An example, the merkle membership proof, only requires a hash function that has collision -resistance, hence a hash function like Pedersen is allowed, which in most cases is more efficient -than the even more conservative sha256. - -[View an example on the starter repo](https://github.com/noir-lang/noir-examples/blob/3ea09545cabfa464124ec2f3ea8e60c608abe6df/stealthdrop/circuits/src/main.nr#L20) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/using-devcontainers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/using-devcontainers.mdx deleted file mode 100644 index 727ec6ca667..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/how_to/using-devcontainers.mdx +++ /dev/null @@ -1,110 +0,0 @@ ---- -title: Developer Containers and Codespaces -description: "Learn how to set up a devcontainer in your GitHub repository for a seamless coding experience with Codespaces. Follow our easy 8-step guide to create your own Noir environment without installing Nargo locally." -keywords: ["Devcontainer", "Codespaces", "GitHub", "Noir Environment", "Docker Image", "Development Environment", "Remote Coding", "GitHub Codespaces", "Noir Programming", "Nargo", "VSCode Extensions", "Noirup"] -sidebar_position: 1 ---- - -Adding a developer container configuration file to your Noir project is one of the easiest way to unlock coding in browser. - -## What's a devcontainer after all? - -A [Developer Container](https://containers.dev/) (devcontainer for short) is a Docker image that comes preloaded with tools, extensions, and other tools you need to quickly get started or continue a project, without having to install Nargo locally. Think of it as a development environment in a box. - -There are many advantages to this: - -- It's platform and architecture agnostic -- You don't need to have an IDE installed, or Nargo, or use a terminal at all -- It's safer for using on a public machine or public network - -One of the best ways of using devcontainers is... not using your machine at all, for maximum control, performance, and ease of use. -Enter Codespaces. - -## Codespaces - -If a devcontainer is just a Docker image, then what stops you from provisioning a `p3dn.24xlarge` AWS EC2 instance with 92 vCPUs and 768 GiB RAM and using it to prove your 10-gate SNARK proof? - -Nothing! Except perhaps the 30-40$ per hour it will cost you. - -The problem is that provisioning takes time, and I bet you don't want to see the AWS console every time you want to code something real quick. - -Fortunately, there's an easy and free way to get a decent remote machine ready and loaded in less than 2 minutes: Codespaces. [Codespaces is a Github feature](https://github.com/features/codespaces) that allows you to code in a remote machine by using devcontainers, and it's pretty cool: - -- You can start coding Noir in less than a minute -- It uses the resources of a remote machine, so you can code on your grandma's phone if needed be -- It makes it easy to share work with your frens -- It's fully reusable, you can stop and restart whenever you need to - -:::info - -Don't take out your wallet just yet. Free GitHub accounts get about [15-60 hours of coding](https://github.com/features/codespaces) for free per month, depending on the size of your provisioned machine. - -::: - -## Tell me it's _actually_ easy - -It is! - -Github comes with a default codespace and you can use it to code your own devcontainer. That's exactly what we will be doing in this guide. - - - -8 simple steps: - -#### 1. Create a new repository on GitHub. - -#### 2. Click "Start coding with Codespaces". This will use the default image. - -#### 3. Create a folder called `.devcontainer` in the root of your repository. - -#### 4. Create a Dockerfile in that folder, and paste the following code: - -```docker -FROM --platform=linux/amd64 node:lts-bookworm-slim -SHELL ["/bin/bash", "-c"] -RUN apt update && apt install -y curl bash git tar gzip libc++-dev -RUN curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -ENV PATH="/root/.nargo/bin:$PATH" -RUN noirup -ENTRYPOINT ["nargo"] -``` -#### 5. Create a file called `devcontainer.json` in the same folder, and paste the following code: - -```json -{ - "name": "Noir on Codespaces", - "build": { - "context": ".", - "dockerfile": "Dockerfile" - }, - "customizations": { - "vscode": { - "extensions": ["noir-lang.vscode-noir"] - } - } -} -``` -#### 6. Commit and push your changes - -This will pull the new image and build it, so it could take a minute or so - -#### 8. Done! -Just wait for the build to finish, and there's your easy Noir environment. - - -Refer to [noir-starter](https://github.com/noir-lang/noir-starter/) as an example of how devcontainers can be used together with codespaces. - - - -## How do I use it? - -Using the codespace is obviously much easier than setting it up. -Just navigate to your repository and click "Code" -> "Open with Codespaces". It should take a few seconds to load, and you're ready to go. - -:::info - -If you really like the experience, you can add a badge to your readme, links to existing codespaces, and more. -Check out the [official docs](https://docs.github.com/en/codespaces/setting-up-your-project-for-codespaces/setting-up-your-repository/facilitating-quick-creation-and-resumption-of-codespaces) for more info. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/index.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/index.mdx deleted file mode 100644 index 9ed9662b0b9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/index.mdx +++ /dev/null @@ -1,67 +0,0 @@ ---- -title: Noir Lang -hide_title: true -description: - Learn about the public alpha release of Noir, a domain specific language heavily influenced by Rust that compiles to - an intermediate language which can be compiled to an arithmetic circuit or a rank-1 constraint system. -keywords: - [Noir, - Domain Specific Language, - Rust, - Intermediate Language, - Arithmetic Circuit, - Rank-1 Constraint System, - Ethereum Developers, - Protocol Developers, - Blockchain Developers, - Proving System, - Smart Contract Language] -sidebar_position: 0 ---- - -import Tabs from '@theme/Tabs'; -import TabItem from '@theme/TabItem'; - -Noir Logo - -Noir is an open-source Domain-Specific Language for safe and seamless construction of privacy-preserving Zero-Knowledge programs, requiring no previous knowledge on the underlying mathematics or cryptography. - -ZK programs are programs that can generate short proofs of statements without revealing all inputs to the statements. You can read more about Zero-Knowledge Proofs [here](https://dev.to/spalladino/a-beginners-intro-to-coding-zero-knowledge-proofs-c56). - -## What's new about Noir? - -Noir works differently from most ZK languages by taking a two-pronged path. First, it compiles the program to an adaptable intermediate language known as ACIR. From there, depending on a given project's needs, ACIR can be further compiled into an arithmetic circuit for integration with the proving backend. - -:::info - -Noir is backend agnostic, which means it makes no assumptions on which proving backend powers the ZK proof. Being the language that powers [Aztec Contracts](https://docs.aztec.network/developers/contracts/main), it defaults to Aztec's Barretenberg proving backend. - -However, the ACIR output can be transformed to be compatible with other PLONK-based backends, or into a [rank-1 constraint system](https://www.rareskills.io/post/rank-1-constraint-system) suitable for backends such as Arkwork's Marlin. - -::: - -## Who is Noir for? - -Noir can be used both in complex cloud-based backends and in user's smartphones, requiring no knowledge on the underlying math or cryptography. From authorization systems that keep a password in the user's device, to complex on-chain verification of recursive proofs, Noir is designed to abstract away complexity without any significant overhead. Here are some examples of situations where Noir can be used: - - - - Noir Logo - - Aztec Contracts leverage Noir to allow for the storage and execution of private information. Writing an Aztec Contract is as easy as writing Noir, and Aztec developers can easily interact with the network storage and execution through the [Aztec.nr](https://docs.aztec.network/developers/contracts/main) library. - - - Solidity Verifier Example - Noir can auto-generate Solidity verifier contracts that verify Noir proofs. This allows for non-interactive verification of proofs containing private information in an immutable system. This feature powers a multitude of use-case scenarios, from P2P chess tournaments, to [Aztec Layer-2 Blockchain](https://docs.aztec.network/) - - - Aztec Labs developed NoirJS, an easy interface to generate and verify Noir proofs in a Javascript environment. This allows for Noir to be used in webpages, mobile apps, games, and any other environment supporting JS execution in a standalone manner. - - - - -## Libraries - -Noir is meant to be easy to extend by simply importing Noir libraries just like in Rust. -The [awesome-noir repo](https://github.com/noir-lang/awesome-noir#libraries) is a collection of libraries developed by the Noir community. -Writing a new library is easy and makes code be composable and easy to reuse. See the section on [dependencies](noir/modules_packages_crates/dependencies.md) for more information. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/migration_notes.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/migration_notes.md deleted file mode 100644 index 6bd740024e5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/migration_notes.md +++ /dev/null @@ -1,105 +0,0 @@ ---- -title: Migration notes -description: Read about migration notes from previous versions, which could solve problems while updating -keywords: [Noir, notes, migration, updating, upgrading] ---- - -Noir is in full-speed development. Things break fast, wild, and often. This page attempts to leave some notes on errors you might encounter when upgrading and how to resolve them until proper patches are built. - -### `backend encountered an error: libc++.so.1` - -Depending on your OS, you may encounter the following error when running `nargo prove` for the first time: - -```text -The backend encountered an error: "/home/codespace/.nargo/backends/acvm-backend-barretenberg/backend_binary: error while loading shared libraries: libc++.so.1: cannot open shared object file: No such file or directory\n" -``` - -Install the `libc++-dev` library with: - -```bash -sudo apt install libc++-dev -``` - -## ≥0.19 - -### Enforcing `compiler_version` - -From this version on, the compiler will check for the `compiler_version` field in `Nargo.toml`, and will error if it doesn't match the current Nargo version in use. - -To update, please make sure this field in `Nargo.toml` matches the output of `nargo --version`. - -## ≥0.14 - -The index of the [for loops](noir/concepts/control_flow.md#loops) is now of type `u64` instead of `Field`. An example refactor would be: - -```rust -for i in 0..10 { - let i = i as Field; -} -``` - -## ≥v0.11.0 and Nargo backend - -From this version onwards, Nargo starts managing backends through the `nargo backend` command. Upgrading to the versions per usual steps might lead to: - -### `backend encountered an error` - -This is likely due to the existing locally installed version of proving backend (e.g. barretenberg) is incompatible with the version of Nargo in use. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo prove -``` - -with your Noir program. - -This will trigger the download and installation of the latest version of barretenberg compatible with your Nargo in use. - -### `backend encountered an error: illegal instruction` - -On certain Intel-based systems, an `illegal instruction` error may arise due to incompatibility of barretenberg with certain CPU instructions. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo backend install acvm-backend-barretenberg https://github.com/noir-lang/barretenberg-js-binary/raw/master/run-bb.tar.gz -``` - -This downloads and installs a specific bb.js based version of barretenberg binary from GitHub. - -The gzipped file is running [this bash script](https://github.com/noir-lang/barretenberg-js-binary/blob/master/run-bb-js.sh), where we need to gzip it as the Nargo currently expect the backend to be zipped up. - -Then run: - -``` -DESIRED_BINARY_VERSION=0.8.1 nargo info -``` - -This overrides the bb native binary with a bb.js node application instead, which should be compatible with most if not all hardware. This does come with the drawback of being generally slower than native binary. - -0.8.1 indicates bb.js version 0.8.1, so if you change that it will update to a different version or the default version in the script if none was supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/_category_.json deleted file mode 100644 index 7da08f8a8c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Concepts", - "position": 0, - "collapsible": true, - "collapsed": true -} \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/assert.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/assert.md deleted file mode 100644 index 2132de42072..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/assert.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Assert Function -description: - Learn about the `assert` and `static_assert` functions in Noir, which can be used to explicitly - constrain the predicate or comparison expression that follows to be true, and what happens if - the expression is false at runtime or compile-time, respectively. -keywords: [Noir programming language, assert statement, predicate expression, comparison expression] -sidebar_position: 4 ---- - -Noir includes a special `assert` function which will explicitly constrain the predicate/comparison -expression that follows to be true. If this expression is false at runtime, the program will fail to -be proven. Example: - -```rust -fn main(x : Field, y : Field) { - assert(x == y); -} -``` - -> Assertions only work for predicate operations, such as `==`. If there's any ambiguity on the operation, the program will fail to compile. For example, it is unclear if `assert(x + y)` would check for `x + y == 0` or simply would return `true`. - -You can optionally provide a message to be logged when the assertion fails: - -```rust -assert(x == y, "x and y are not equal"); -``` - -Aside string literals, the optional message can be a format string or any other type supported as input for Noir's [print](../standard_library/logging.md) functions. This feature lets you incorporate runtime variables into your failed assertion logs: - -```rust -assert(x == y, f"Expected x == y, but got {x} == {y}"); -``` - -Using a variable as an assertion message directly: - -```rust -struct myStruct { - myField: Field -} - -let s = myStruct { myField: y }; -assert(s.myField == x, s); -``` - -There is also a special `static_assert` function that behaves like `assert`, -but that runs at compile-time. - -```rust -fn main(xs: [Field; 3]) { - let x = 2 + 2; - let y = 4; - static_assert(x == y, "expected 2 + 2 to equal 4"); - - // This passes since the length of `xs` is known at compile-time - static_assert(xs.len() == 3, "expected the input to have 3 elements"); -} -``` - -This function fails when passed a dynamic (run-time) argument: - -```rust -fn main(x : Field, y : Field) { - // this fails because `x` is not known at compile-time - static_assert(x == 2, "expected x to be known at compile-time and equal to 2"); - - let mut example_slice = &[]; - if y == 4 { - example_slice = example_slice.push_back(0); - } - - // This fails because the length of `example_slice` is not known at - // compile-time - let error_message = "expected an empty slice, known at compile-time"; - static_assert(example_slice.len() == 0, error_message); -} -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/comments.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/comments.md deleted file mode 100644 index b51a85f5c94..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/comments.md +++ /dev/null @@ -1,33 +0,0 @@ ---- -title: Comments -description: - Learn how to write comments in Noir programming language. A comment is a line of code that is - ignored by the compiler, but it can be read by programmers. Single-line and multi-line comments - are supported in Noir. -keywords: [Noir programming language, comments, single-line comments, multi-line comments] -sidebar_position: 10 ---- - -A comment is a line in your codebase which the compiler ignores, however it can be read by -programmers. - -Here is a single line comment: - -```rust -// This is a comment and is ignored -``` - -`//` is used to tell the compiler to ignore the rest of the line. - -Noir also supports multi-line block comments. Start a block comment with `/*` and end the block with `*/`. - -Noir does not natively support doc comments. You may be able to use [Rust doc comments](https://doc.rust-lang.org/reference/comments.html) in your code to leverage some Rust documentation build tools with Noir code. - -```rust -/* - This is a block comment describing a complex function. -*/ -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/control_flow.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/control_flow.md deleted file mode 100644 index 045d3c3a5f5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/control_flow.md +++ /dev/null @@ -1,77 +0,0 @@ ---- -title: Control Flow -description: - Learn how to use loops and if expressions in the Noir programming language. Discover the syntax - and examples for for loops and if-else statements. -keywords: [Noir programming language, loops, for loop, if-else statements, Rust syntax] -sidebar_position: 2 ---- - -## If Expressions - -Noir supports `if-else` statements. The syntax is most similar to Rust's where it is not required -for the statement's conditional to be surrounded by parentheses. - -```rust -let a = 0; -let mut x: u32 = 0; - -if a == 0 { - if a != 0 { - x = 6; - } else { - x = 2; - } -} else { - x = 5; - assert(x == 5); -} -assert(x == 2); -``` - -## Loops - -Noir has one kind of loop: the `for` loop. `for` loops allow you to repeat a block of code multiple -times. - -The following block of code between the braces is run 10 times. - -```rust -for i in 0..10 { - // do something -} -``` - -The index for loops is of type `u64`. - -### Break and Continue - -In unconstrained code, `break` and `continue` are also allowed in `for` loops. These are only allowed -in unconstrained code since normal constrained code requires that Noir knows exactly how many iterations -a loop may have. `break` and `continue` can be used like so: - -```rust -for i in 0 .. 10 { - println("Iteration start") - - if i == 2 { - continue; - } - - if i == 5 { - break; - } - - println(i); -} -println("Loop end") -``` - -When used, `break` will end the current loop early and jump to the statement after the for loop. In the example -above, the `break` will stop the loop and jump to the `println("Loop end")`. - -`continue` will stop the current iteration of the loop, and jump to the start of the next iteration. In the example -above, `continue` will jump to `println("Iteration start")` when used. Note that the loop continues as normal after this. -The iteration variable `i` is still increased by one as normal when `continue` is used. - -`break` and `continue` cannot currently be used to jump out of more than a single loop at a time. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_bus.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_bus.mdx deleted file mode 100644 index e55e58622ce..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_bus.mdx +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: Data Bus -sidebar_position: 13 ---- -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -The data bus is an optimization that the backend can use to make recursion more efficient. -In order to use it, you must define some inputs of the program entry points (usually the `main()` -function) with the `call_data` modifier, and the return values with the `return_data` modifier. -These modifiers are incompatible with `pub` and `mut` modifiers. - -## Example - -```rust -fn main(mut x: u32, y: call_data u32, z: call_data [u32;4] ) -> return_data u32 { - let a = z[x]; - a+y -} -``` - -As a result, both call_data and return_data will be treated as private inputs and encapsulated into a read-only array each, for the backend to process. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/arrays.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/arrays.md deleted file mode 100644 index d26f6dff070..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/arrays.md +++ /dev/null @@ -1,253 +0,0 @@ ---- -title: Arrays -description: - Dive into the Array data type in Noir. Grasp its methods, practical examples, and best practices for efficiently using Arrays in your Noir code. -keywords: - [ - noir, - array type, - methods, - examples, - indexing, - ] -sidebar_position: 4 ---- - -An array is one way of grouping together values into one compound type. Array types can be inferred -or explicitly specified via the syntax `[; ]`: - -```rust -fn main(x : Field, y : Field) { - let my_arr = [x, y]; - let your_arr: [Field; 2] = [x, y]; -} -``` - -Here, both `my_arr` and `your_arr` are instantiated as an array containing two `Field` elements. - -Array elements can be accessed using indexing: - -```rust -fn main() { - let a = [1, 2, 3, 4, 5]; - - let first = a[0]; - let second = a[1]; -} -``` - -All elements in an array must be of the same type (i.e. homogeneous). That is, an array cannot group -a `Field` value and a `u8` value together for example. - -You can write mutable arrays, like: - -```rust -fn main() { - let mut arr = [1, 2, 3, 4, 5]; - assert(arr[0] == 1); - - arr[0] = 42; - assert(arr[0] == 42); -} -``` - -You can instantiate a new array of a fixed size with the same value repeated for each element. The following example instantiates an array of length 32 where each element is of type Field and has the value 0. - -```rust -let array: [Field; 32] = [0; 32]; -``` - -Like in Rust, arrays in Noir are a fixed size. However, if you wish to convert an array to a [slice](./slices.mdx), you can just call `as_slice` on your array: - -```rust -let array: [Field; 32] = [0; 32]; -let sl = array.as_slice() -``` - -You can define multidimensional arrays: - -```rust -let array : [[Field; 2]; 2]; -let element = array[0][0]; -``` - -However, multidimensional slices are not supported. For example, the following code will error at compile time: - -```rust -let slice : [[Field]] = &[]; -``` - -## Types - -You can create arrays of primitive types or structs. There is not yet support for nested arrays -(arrays of arrays) or arrays of structs that contain arrays. - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for arrays. -Each of these functions are located within the generic impl `impl [T; N] {`. -So anywhere `self` appears, it refers to the variable `self: [T; N]`. - -### len - -Returns the length of an array - -```rust -fn len(self) -> Field -``` - -example - -```rust -fn main() { - let array = [42, 42]; - assert(array.len() == 2); -} -``` - -### sort - -Returns a new sorted array. The original array remains untouched. Notice that this function will -only work for arrays of fields or integers, not for any arbitrary type. This is because the sorting -logic it uses internally is optimized specifically for these values. If you need a sort function to -sort any type, you should use the function `sort_via` described below. - -```rust -fn sort(self) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32]; - let sorted = arr.sort(); - assert(sorted == [32, 42]); -} -``` - -### sort_via - -Sorts the array with a custom comparison function - -```rust -fn sort_via(self, ordering: fn(T, T) -> bool) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32] - let sorted_ascending = arr.sort_via(|a, b| a <= b); - assert(sorted_ascending == [32, 42]); // verifies - - let sorted_descending = arr.sort_via(|a, b| a >= b); - assert(sorted_descending == [32, 42]); // does not verify -} -``` - -### map - -Applies a function to each element of the array, returning a new array containing the mapped elements. - -```rust -fn map(self, f: fn(T) -> U) -> [U; N] -``` - -example - -```rust -let a = [1, 2, 3]; -let b = a.map(|a| a * 2); // b is now [2, 4, 6] -``` - -### fold - -Applies a function to each element of the array, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn(U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the array, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = [1]; -let a2 = [1, 2]; -let a3 = [1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let arr = [2, 2, 2, 2, 2]; - let folded = arr.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn(T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let reduced = arr.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let all = arr.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 5]; - let any = arr.any(|a| a == 5); - assert(any); -} - -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/booleans.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/booleans.md deleted file mode 100644 index 2507af710e7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/booleans.md +++ /dev/null @@ -1,28 +0,0 @@ ---- -title: Booleans -description: - Delve into the Boolean data type in Noir. Understand its methods, practical examples, and best practices for using Booleans in your Noir programs. -keywords: - [ - noir, - boolean type, - methods, - examples, - logical operations, - ] -sidebar_position: 2 ---- - - -The `bool` type in Noir has two possible values: `true` and `false`: - -```rust -fn main() { - let t = true; - let f: bool = false; -} -``` - -The boolean type is most commonly used in conditionals like `if` expressions and `assert` -statements. More about conditionals is covered in the [Control Flow](../control_flow.md) and -[Assert Function](../assert.md) sections. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/fields.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/fields.md deleted file mode 100644 index a10a4810788..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/fields.md +++ /dev/null @@ -1,192 +0,0 @@ ---- -title: Fields -description: - Dive deep into the Field data type in Noir. Understand its methods, practical examples, and best practices to effectively use Fields in your Noir programs. -keywords: - [ - noir, - field type, - methods, - examples, - best practices, - ] -sidebar_position: 0 ---- - -The field type corresponds to the native field type of the proving backend. - -The size of a Noir field depends on the elliptic curve's finite field for the proving backend -adopted. For example, a field would be a 254-bit integer when paired with the default backend that -spans the Grumpkin curve. - -Fields support integer arithmetic and are often used as the default numeric type in Noir: - -```rust -fn main(x : Field, y : Field) { - let z = x + y; -} -``` - -`x`, `y` and `z` are all private fields in this example. Using the `let` keyword we defined a new -private value `z` constrained to be equal to `x + y`. - -If proving efficiency is of priority, fields should be used as a default for solving problems. -Smaller integer types (e.g. `u64`) incur extra range constraints. - -## Methods - -After declaring a Field, you can use these common methods on it: - -### to_le_bits - -Transforms the field into an array of bits, Little Endian. - -```rust -fn to_le_bits(_x : Field, _bit_size: u32) -> [u1] -``` - -example: - -```rust -fn main() { - let field = 2; - let bits = field.to_le_bits(32); -} -``` - -### to_be_bits - -Transforms the field into an array of bits, Big Endian. - -```rust -fn to_be_bits(_x : Field, _bit_size: u32) -> [u1] -``` - -example: - -```rust -fn main() { - let field = 2; - let bits = field.to_be_bits(32); -} -``` - -### to_le_bytes - -Transforms into an array of bytes, Little Endian - -```rust -fn to_le_bytes(_x : Field, byte_size: u32) -> [u8] -``` - -example: - -```rust -fn main() { - let field = 2; - let bytes = field.to_le_bytes(4); -} -``` - -### to_be_bytes - -Transforms into an array of bytes, Big Endian - -```rust -fn to_be_bytes(_x : Field, byte_size: u32) -> [u8] -``` - -example: - -```rust -fn main() { - let field = 2; - let bytes = field.to_be_bytes(4); -} -``` - -### to_le_radix - -Decomposes into a vector over the specified base, Little Endian - -```rust -fn to_le_radix(_x : Field, _radix: u32, _result_len: u32) -> [u8] -``` - -example: - -```rust -fn main() { - let field = 2; - let radix = field.to_le_radix(256, 4); -} -``` - -### to_be_radix - -Decomposes into a vector over the specified base, Big Endian - -```rust -fn to_be_radix(_x : Field, _radix: u32, _result_len: u32) -> [u8] -``` - -example: - -```rust -fn main() { - let field = 2; - let radix = field.to_be_radix(256, 4); -} -``` - -### pow_32 - -Returns the value to the power of the specified exponent - -```rust -fn pow_32(self, exponent: Field) -> Field -``` - -example: - -```rust -fn main() { - let field = 2 - let pow = field.pow_32(4); - assert(pow == 16); -} -``` - -### assert_max_bit_size - -Adds a constraint to specify that the field can be represented with `bit_size` number of bits - -```rust -fn assert_max_bit_size(self, bit_size: u32) -``` - -example: - -```rust -fn main() { - let field = 2 - field.assert_max_bit_size(32); -} -``` - -### sgn0 - -Parity of (prime) Field element, i.e. sgn0(x mod p) = 0 if x ∈ \{0, ..., p-1\} is even, otherwise sgn0(x mod p) = 1. - -```rust -fn sgn0(self) -> u1 -``` - - -### lt - -Returns true if the field is less than the other field - -```rust -pub fn lt(self, another: Field) -> bool -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/function_types.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/function_types.md deleted file mode 100644 index f6121af17e2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/function_types.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Function types -sidebar_position: 10 ---- - -Noir supports higher-order functions. The syntax for a function type is as follows: - -```rust -fn(arg1_type, arg2_type, ...) -> return_type -``` - -Example: - -```rust -fn assert_returns_100(f: fn() -> Field) { // f takes no args and returns a Field - assert(f() == 100); -} - -fn main() { - assert_returns_100(|| 100); // ok - assert_returns_100(|| 150); // fails -} -``` - -A function type also has an optional capture environment - this is necessary to support closures. -See [Lambdas](../lambdas.md) for more details. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/index.md deleted file mode 100644 index 3eadb2dc8a4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/index.md +++ /dev/null @@ -1,118 +0,0 @@ ---- -title: Data Types -description: - Get a clear understanding of the two categories of Noir data types - primitive types and compound - types. Learn about their characteristics, differences, and how to use them in your Noir - programming. -keywords: - [ - noir, - data types, - primitive types, - compound types, - private types, - public types, - ] ---- - -Every value in Noir has a type, which determines which operations are valid for it. - -All values in Noir are fundamentally composed of `Field` elements. For a more approachable -developing experience, abstractions are added on top to introduce different data types in Noir. - -Noir has two category of data types: primitive types (e.g. `Field`, integers, `bool`) and compound -types that group primitive types (e.g. arrays, tuples, structs). Each value can either be private or -public. - -## Private & Public Types - -A **private value** is known only to the Prover, while a **public value** is known by both the -Prover and Verifier. Mark values as `private` when the value should only be known to the prover. All -primitive types (including individual fields of compound types) in Noir are private by default, and -can be marked public when certain values are intended to be revealed to the Verifier. - -> **Note:** For public values defined in Noir programs paired with smart contract verifiers, once -> the proofs are verified on-chain the values can be considered known to everyone that has access to -> that blockchain. - -Public data types are treated no differently to private types apart from the fact that their values -will be revealed in proofs generated. Simply changing the value of a public type will not change the -circuit (where the same goes for changing values of private types as well). - -_Private values_ are also referred to as _witnesses_ sometimes. - -> **Note:** The terms private and public when applied to a type (e.g. `pub Field`) have a different -> meaning than when applied to a function (e.g. `pub fn foo() {}`). -> -> The former is a visibility modifier for the Prover to interpret if a value should be made known to -> the Verifier, while the latter is a visibility modifier for the compiler to interpret if a -> function should be made accessible to external Noir programs like in other languages. - -### pub Modifier - -All data types in Noir are private by default. Types are explicitly declared as public using the -`pub` modifier: - -```rust -fn main(x : Field, y : pub Field) -> pub Field { - x + y -} -``` - -In this example, `x` is **private** while `y` and `x + y` (the return value) are **public**. Note -that visibility is handled **per variable**, so it is perfectly valid to have one input that is -private and another that is public. - -> **Note:** Public types can only be declared through parameters on `main`. - -## Type Aliases - -A type alias is a new name for an existing type. Type aliases are declared with the keyword `type`: - -```rust -type Id = u8; - -fn main() { - let id: Id = 1; - let zero: u8 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can also be used with [generics](../generics.md): - -```rust -type Id = Size; - -fn main() { - let id: Id = 1; - let zero: u32 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can even refer to other aliases. An error will be issued if they form a cycle: - -```rust -// Ok! -type A = B; -type B = Field; - -type Bad1 = Bad2; - -// error: Dependency cycle found -type Bad2 = Bad1; -// ^^^^^^^^^^^ 'Bad2' recursively depends on itself: Bad2 -> Bad1 -> Bad2 -``` - -## Wildcard Type -Noir can usually infer the type of the variable from the context, so specifying the type of a variable is only required when it cannot be inferred. However, specifying a complex type can be tedious, especially when it has multiple generic arguments. Often some of the generic types can be inferred from the context, and Noir only needs a hint to properly infer the other types. We can partially specify a variable's type by using `_` as a marker, indicating where we still want the compiler to infer the type. - -```rust -let a: [_; 4] = foo(b); -``` - - -### BigInt - -You can achieve BigInt functionality using the [Noir BigInt](https://github.com/shuklaayush/noir-bigint) library. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/references.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/references.md deleted file mode 100644 index a5293d11cfb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/references.md +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: References -sidebar_position: 9 ---- - -Noir supports first-class references. References are a bit like pointers: they point to a specific address that can be followed to access the data stored at that address. You can use Rust-like syntax to use pointers in Noir: the `&` operator references the variable, the `*` operator dereferences it. - -Example: - -```rust -fn main() { - let mut x = 2; - - // you can reference x as &mut and pass it to multiplyBy2 - multiplyBy2(&mut x); -} - -// you can access &mut here -fn multiplyBy2(x: &mut Field) { - // and dereference it with * - *x = *x * 2; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/slices.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/slices.mdx deleted file mode 100644 index 95da2030843..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/slices.mdx +++ /dev/null @@ -1,358 +0,0 @@ ---- -title: Slices -description: Explore the Slice data type in Noir. Understand its methods, see real-world examples, and learn how to effectively use Slices in your Noir programs. -keywords: [noir, slice type, methods, examples, subarrays] -sidebar_position: 5 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A slice is a dynamically-sized view into a sequence of elements. They can be resized at runtime, but because they don't own the data, they cannot be returned from a circuit. You can treat slices as arrays without a constrained size. - -```rust -fn main() -> pub u32 { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -To write a slice literal, use a preceeding ampersand as in: `&[0; 2]` or -`&[1, 2, 3]`. - -It is important to note that slices are not references to arrays. In Noir, -`&[..]` is more similar to an immutable, growable vector. - -View the corresponding test file [here][test-file]. - -[test-file]: https://github.com/noir-lang/noir/blob/f387ec1475129732f72ba294877efdf6857135ac/crates/nargo_cli/tests/test_data_ssa_refactor/slices/src/main.nr - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for slices: - -### push_back - -Pushes a new element to the end of the slice, returning a new slice with a length one greater than the original unmodified slice. - -```rust -fn push_back(_self: [T], _elem: T) -> [T] -``` - -example: - -```rust -fn main() -> pub Field { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -View the corresponding test file [here][test-file]. - -### push_front - -Returns a new array with the specified element inserted at index 0. The existing elements indexes are incremented by 1. - -```rust -fn push_front(_self: Self, _elem: T) -> Self -``` - -Example: - -```rust -let mut new_slice: [Field] = &[]; -new_slice = new_slice.push_front(20); -assert(new_slice[0] == 20); // returns true -``` - -View the corresponding test file [here][test-file]. - -### pop_front - -Returns a tuple of two items, the first element of the array and the rest of the array. - -```rust -fn pop_front(_self: Self) -> (T, Self) -``` - -Example: - -```rust -let (first_elem, rest_of_slice) = slice.pop_front(); -``` - -View the corresponding test file [here][test-file]. - -### pop_back - -Returns a tuple of two items, the beginning of the array with the last element omitted and the last element. - -```rust -fn pop_back(_self: Self) -> (Self, T) -``` - -Example: - -```rust -let (popped_slice, last_elem) = slice.pop_back(); -``` - -View the corresponding test file [here][test-file]. - -### append - -Loops over a slice and adds it to the end of another. - -```rust -fn append(mut self, other: Self) -> Self -``` - -Example: - -```rust -let append = &[1, 2].append(&[3, 4, 5]); -``` - -### insert - -Inserts an element at a specified index and shifts all following elements by 1. - -```rust -fn insert(_self: Self, _index: Field, _elem: T) -> Self -``` - -Example: - -```rust -new_slice = rest_of_slice.insert(2, 100); -assert(new_slice[2] == 100); -``` - -View the corresponding test file [here][test-file]. - -### remove - -Remove an element at a specified index, shifting all elements after it to the left, returning the altered slice and the removed element. - -```rust -fn remove(_self: Self, _index: Field) -> (Self, T) -``` - -Example: - -```rust -let (remove_slice, removed_elem) = slice.remove(3); -``` - -### len - -Returns the length of a slice - -```rust -fn len(self) -> Field -``` - -Example: - -```rust -fn main() { - let slice = &[42, 42]; - assert(slice.len() == 2); -} -``` - -### as_array - -Converts this slice into an array. - -Make sure to specify the size of the resulting array. -Panics if the resulting array length is different than the slice's length. - -```rust -fn as_array(self) -> [T; N] -``` - -Example: - -```rust -fn main() { - let slice = &[5, 6]; - - // Always specify the length of the resulting array! - let array: [Field; 2] = slice.as_array(); - - assert(array[0] == slice[0]); - assert(array[1] == slice[1]); -} -``` - -### map - -Applies a function to each element of the slice, returning a new slice containing the mapped elements. - -```rust -fn map(self, f: fn[Env](T) -> U) -> [U] -``` - -example - -```rust -let a = &[1, 2, 3]; -let b = a.map(|a| a * 2); // b is now &[2, 4, 6] -``` - -### fold - -Applies a function to each element of the slice, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn[Env](U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the slice, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = &[1]; -let a2 = &[1, 2]; -let a3 = &[1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let folded = slice.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn[Env](T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let reduced = slice.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### filter - -Returns a new slice containing only elements for which the given predicate returns true. - -```rust -fn filter(self, f: fn[Env](T) -> bool) -> Self -``` - -example: - -```rust -fn main() { - let slice = &[1, 2, 3, 4, 5]; - let odds = slice.filter(|x| x % 2 == 1); - assert_eq(odds, &[1, 3, 5]); -} -``` - -### join - -Flatten each element in the slice into one value, separated by `separator`. - -Note that although slices implement `Append`, `join` cannot be used on slice -elements since nested slices are prohibited. - -```rust -fn join(self, separator: T) -> T where T: Append -``` - -example: - -```rust -struct Accumulator { - total: Field, -} - -// "Append" two accumulators by adding them -impl Append for Accumulator { - fn empty() -> Self { - Self { total: 0 } - } - - fn append(self, other: Self) -> Self { - Self { total: self.total + other.total } - } -} - -fn main() { - let slice = &[1, 2, 3, 4, 5].map(|total| Accumulator { total }); - - let result = slice.join(Accumulator::empty()); - assert_eq(result, Accumulator { total: 15 }); - - // We can use a non-empty separator to insert additional elements to sum: - let separator = Accumulator { total: 10 }; - let result = slice.join(separator); - assert_eq(result, Accumulator { total: 55 }); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let all = slice.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 5]; - let any = slice.any(|a| a == 5); - assert(any); -} - -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/strings.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/strings.md deleted file mode 100644 index 1fdee42425e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/strings.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Strings -description: - Discover the String data type in Noir. Learn about its methods, see real-world examples, and understand how to effectively manipulate and use Strings in Noir. -keywords: - [ - noir, - string type, - methods, - examples, - concatenation, - ] -sidebar_position: 3 ---- - - -The string type is a fixed length value defined with `str`. - -You can use strings in `assert()` functions or print them with -`println()`. See more about [Logging](../../standard_library/logging.md). - -```rust - -fn main(message : pub str<11>, hex_as_string : str<4>) { - println(message); - assert(message == "hello world"); - assert(hex_as_string == "0x41"); -} -``` - -You can convert a `str` to a byte array by calling `as_bytes()` -or a vector by calling `as_bytes_vec()`. - -```rust -fn main() { - let message = "hello world"; - let message_bytes = message.as_bytes(); - let mut message_vec = message.as_bytes_vec(); - assert(message_bytes.len() == 11); - assert(message_bytes[0] == 104); - assert(message_bytes[0] == message_vec.get(0)); -} -``` - -## Escape characters - -You can use escape characters for your strings: - -| Escape Sequence | Description | -|-----------------|-----------------| -| `\r` | Carriage Return | -| `\n` | Newline | -| `\t` | Tab | -| `\0` | Null Character | -| `\"` | Double Quote | -| `\\` | Backslash | - -Example: - -```rust -let s = "Hello \"world" // prints "Hello "world" -let s = "hey \tyou"; // prints "hey you" -``` - -## Raw strings - -A raw string begins with the letter `r` and is optionally delimited by a number of hashes `#`. - -Escape characters are *not* processed within raw strings. All contents are interpreted literally. - -Example: - -```rust -let s = r"Hello world"; -let s = r#"Simon says "hello world""#; - -// Any number of hashes may be used (>= 1) as long as the string also terminates with the same number of hashes -let s = r#####"One "#, Two "##, Three "###, Four "####, Five will end the string."#####; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/structs.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/structs.md deleted file mode 100644 index dbf68c99813..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/structs.md +++ /dev/null @@ -1,70 +0,0 @@ ---- -title: Structs -description: - Explore the Struct data type in Noir. Learn about its methods, see real-world examples, and grasp how to effectively define and use Structs in your Noir programs. -keywords: - [ - noir, - struct type, - methods, - examples, - data structures, - ] -sidebar_position: 8 ---- - -A struct also allows for grouping multiple values of different types. Unlike tuples, we can also -name each field. - -> **Note:** The usage of _field_ here refers to each element of the struct and is unrelated to the -> field type of Noir. - -Defining a struct requires giving it a name and listing each field within as `: ` pairs: - -```rust -struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` - -An instance of a struct can then be created with actual values in `: ` pairs in any -order. Struct fields are accessible using their given names: - -```rust -fn main() { - let legs = 4; - - let dog = Animal { - eyes: 2, - hands: 0, - legs, - }; - - let zero = dog.hands; -} -``` - -Structs can also be destructured in a pattern, binding each field to a new variable: - -```rust -fn main() { - let Animal { hands, legs: feet, eyes } = get_octopus(); - - let ten = hands + feet + eyes as u8; -} - -fn get_octopus() -> Animal { - let octopus = Animal { - hands: 0, - legs: 8, - eyes: 2, - }; - - octopus -} -``` - -The new variables can be bound with names different from the original struct field names, as -showcased in the `legs --> feet` binding in the example above. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/tuples.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/tuples.md deleted file mode 100644 index 2ec5c9c4113..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/tuples.md +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Tuples -description: - Dive into the Tuple data type in Noir. Understand its methods, practical examples, and best practices for efficiently using Tuples in your Noir code. -keywords: - [ - noir, - tuple type, - methods, - examples, - multi-value containers, - ] -sidebar_position: 7 ---- - -A tuple collects multiple values like an array, but with the added ability to collect values of -different types: - -```rust -fn main() { - let tup: (u8, u64, Field) = (255, 500, 1000); -} -``` - -One way to access tuple elements is via destructuring using pattern matching: - -```rust -fn main() { - let tup = (1, 2); - - let (one, two) = tup; - - let three = one + two; -} -``` - -Another way to access tuple elements is via direct member access, using a period (`.`) followed by -the index of the element we want to access. Index `0` corresponds to the first tuple element, `1` to -the second and so on: - -```rust -fn main() { - let tup = (5, 6, 7, 8); - - let five = tup.0; - let eight = tup.3; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/functions.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/functions.md deleted file mode 100644 index f656cdfd97a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/functions.md +++ /dev/null @@ -1,226 +0,0 @@ ---- -title: Functions -description: - Learn how to declare functions and methods in Noir, a programming language with Rust semantics. - This guide covers parameter declaration, return types, call expressions, and more. -keywords: [Noir, Rust, functions, methods, parameter declaration, return types, call expressions] -sidebar_position: 1 ---- - -Functions in Noir follow the same semantics of Rust, though Noir does not support early returns. - -To declare a function the `fn` keyword is used. - -```rust -fn foo() {} -``` - -By default, functions are visible only within the package they are defined. To make them visible outside of that package (for example, as part of a [library](../modules_packages_crates/crates_and_packages.md#libraries)), you should mark them as `pub`: - -```rust -pub fn foo() {} -``` - -You can also restrict the visibility of the function to only the crate it was defined in, by specifying `pub(crate)`: - -```rust -pub(crate) fn foo() {} //foo can only be called within its crate -``` - -All parameters in a function must have a type and all types are known at compile time. The parameter -is pre-pended with a colon and the parameter type. Multiple parameters are separated using a comma. - -```rust -fn foo(x : Field, y : Field){} -``` - -The return type of a function can be stated by using the `->` arrow notation. The function below -states that the foo function must return a `Field`. If the function returns no value, then the arrow -is omitted. - -```rust -fn foo(x : Field, y : Field) -> Field { - x + y -} -``` - -Note that a `return` keyword is unneeded in this case - the last expression in a function's body is -returned. - -## Main function - -If you're writing a binary, the `main` function is the starting point of your program. You can pass all types of expressions to it, as long as they have a fixed size at compile time: - -```rust -fn main(x : Field) // this is fine: passing a Field -fn main(x : [Field; 2]) // this is also fine: passing a Field with known size at compile-time -fn main(x : (Field, bool)) // 👌: passing a (Field, bool) tuple means size 2 -fn main(x : str<5>) // this is fine, as long as you pass a string of size 5 - -fn main(x : Vec) // can't compile, has variable size -fn main(x : [Field]) // can't compile, has variable size -fn main(....// i think you got it by now -``` - -Keep in mind [tests](../../tooling/testing.md) don't differentiate between `main` and any other function. The following snippet passes tests, but won't compile or prove: - -```rust -fn main(x : [Field]) { - assert(x[0] == 1); -} - -#[test] -fn test_one() { - main(&[1, 2]); -} -``` - -```bash -$ nargo test -[testing] Running 1 test functions -[testing] Testing test_one... ok -[testing] All tests passed - -$ nargo check -The application panicked (crashed). -Message: Cannot have variable sized arrays as a parameter to main -``` - -## Call Expressions - -Calling a function in Noir is executed by using the function name and passing in the necessary -arguments. - -Below we show how to call the `foo` function from the `main` function using a call expression: - -```rust -fn main(x : Field, y : Field) { - let z = foo(x); -} - -fn foo(x : Field) -> Field { - x + x -} -``` - -## Methods - -You can define methods in Noir on any struct type in scope. - -```rust -struct MyStruct { - foo: Field, - bar: Field, -} - -impl MyStruct { - fn new(foo: Field) -> MyStruct { - MyStruct { - foo, - bar: 2, - } - } - - fn sum(self) -> Field { - self.foo + self.bar - } -} - -fn main() { - let s = MyStruct::new(40); - assert(s.sum() == 42); -} -``` - -Methods are just syntactic sugar for functions, so if we wanted to we could also call `sum` as -follows: - -```rust -assert(MyStruct::sum(s) == 42); -``` - -It is also possible to specialize which method is chosen depending on the [generic](./generics.md) type that is used. In this example, the `foo` function returns different values depending on its type: - -```rust -struct Foo {} - -impl Foo { - fn foo(self) -> Field { 1 } -} - -impl Foo { - fn foo(self) -> Field { 2 } -} - -fn main() { - let f1: Foo = Foo{}; - let f2: Foo = Foo{}; - assert(f1.foo() + f2.foo() == 3); -} -``` - -Also note that impls with the same method name defined in them cannot overlap. For example, if we already have `foo` defined for `Foo` and `Foo` like we do above, we cannot also define `foo` in an `impl Foo` since it would be ambiguous which version of `foo` to choose. - -```rust -// Including this impl in the same project as the above snippet would -// cause an overlapping impls error -impl Foo { - fn foo(self) -> Field { 3 } -} -``` - -## Lambdas - -Lambdas are anonymous functions. They follow the syntax of Rust - `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -See [Lambdas](./lambdas.md) for more details. - -## Attributes - -Attributes are metadata that can be applied to a function, using the following syntax: `#[attribute(value)]`. - -Supported attributes include: - -- **builtin**: the function is implemented by the compiler, for efficiency purposes. -- **deprecated**: mark the function as _deprecated_. Calling the function will generate a warning: `warning: use of deprecated function` -- **field**: Used to enable conditional compilation of code depending on the field size. See below for more details -- **oracle**: mark the function as _oracle_; meaning it is an external unconstrained function, implemented in noir_js. See [Unconstrained](./unconstrained.md) and [NoirJS](../../reference/NoirJS/noir_js/index.md) for more details. -- **test**: mark the function as unit tests. See [Tests](../../tooling/testing.md) for more details - -### Field Attribute - -The field attribute defines which field the function is compatible for. The function is conditionally compiled, under the condition that the field attribute matches the Noir native field. -The field can be defined implicitly, by using the name of the elliptic curve usually associated to it - for instance bn254, bls12_381 - or explicitly by using the field (prime) order, in decimal or hexadecimal form. -As a result, it is possible to define multiple versions of a function with each version specialized for a different field attribute. This can be useful when a function requires different parameters depending on the underlying elliptic curve. - -Example: we define the function `foo()` three times below. Once for the default Noir bn254 curve, once for the field $\mathbb F_{23}$, which will normally never be used by Noir, and once again for the bls12_381 curve. - -```rust -#[field(bn254)] -fn foo() -> u32 { - 1 -} - -#[field(23)] -fn foo() -> u32 { - 2 -} - -// This commented code would not compile as foo would be defined twice because it is the same field as bn254 -// #[field(21888242871839275222246405745257275088548364400416034343698204186575808495617)] -// fn foo() -> u32 { -// 2 -// } - -#[field(bls12_381)] -fn foo() -> u32 { - 3 -} -``` - -If the field name is not known to Noir, it will discard the function. Field names are case insensitive. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/generics.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/generics.md deleted file mode 100644 index 3e416eee093..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/generics.md +++ /dev/null @@ -1,163 +0,0 @@ ---- -title: Generics -description: Learn how to use Generics in Noir -keywords: [Noir, Rust, generics, functions, structs] -sidebar_position: 7 ---- - -Generics allow you to use the same functions with multiple different concrete data types. You can -read more about the concept of generics in the Rust documentation -[here](https://doc.rust-lang.org/book/ch10-01-syntax.html). - -Here is a trivial example showing the identity function that supports any type. In Rust, it is -common to refer to the most general type as `T`. We follow the same convention in Noir. - -```rust -fn id(x: T) -> T { - x -} -``` - -## In Structs - -Generics are useful for specifying types in structs. For example, we can specify that a field in a -struct will be of a certain generic type. In this case `value` is of type `T`. - -```rust -struct RepeatedValue { - value: T, - count: Field, -} - -impl RepeatedValue { - fn print(self) { - for _i in 0 .. self.count { - println(self.value); - } - } -} - -fn main() { - let repeated = RepeatedValue { value: "Hello!", count: 2 }; - repeated.print(); -} -``` - -The `print` function will print `Hello!` an arbitrary number of times, twice in this case. - -If we want to be generic over array lengths (which are type-level integers), we can use numeric -generics. Using these looks just like using regular generics, but these generics can resolve to -integers at compile-time, rather than resolving to types. Here's an example of a struct that is -generic over the size of the array it contains internally: - -```rust -struct BigInt { - limbs: [u32; N], -} - -impl BigInt { - // `N` is in scope of all methods in the impl - fn first(first: BigInt, second: BigInt) -> Self { - assert(first.limbs != second.limbs); - first - - fn second(first: BigInt, second: Self) -> Self { - assert(first.limbs != second.limbs); - second - } -} -``` - -## Calling functions on generic parameters - -Since a generic type `T` can represent any type, how can we call functions on the underlying type? -In other words, how can we go from "any type `T`" to "any type `T` that has certain methods available?" - -This is what [traits](../concepts/traits.md) are for in Noir. Here's an example of a function generic over -any type `T` that implements the `Eq` trait for equality: - -```rust -fn first_element_is_equal(array1: [T; N], array2: [T; N]) -> bool - where T: Eq -{ - if (array1.len() == 0) | (array2.len() == 0) { - true - } else { - array1[0] == array2[0] - } -} - -fn main() { - assert(first_element_is_equal([1, 2, 3], [1, 5, 6])); - - // We can use first_element_is_equal for arrays of any type - // as long as we have an Eq impl for the types we pass in - let array = [MyStruct::new(), MyStruct::new()]; - assert(array_eq(array, array, MyStruct::eq)); -} - -impl Eq for MyStruct { - fn eq(self, other: MyStruct) -> bool { - self.foo == other.foo - } -} -``` - -You can find more details on traits and trait implementations on the [traits page](../concepts/traits.md). - -## Manually Specifying Generics with the Turbofish Operator - -There are times when the compiler cannot reasonably infer what type should be used for a generic, or when the developer themselves may want to manually distinguish generic type parameters. This is where the `::<>` turbofish operator comes into play. - -The `::<>` operator can follow a variable or path and can be used to manually specify generic arguments within the angle brackets. -The name "turbofish" comes from that `::<>` looks like a little fish. - -Examples: -```rust -fn main() { - let mut slice = []; - slice = slice.push_back(1); - slice = slice.push_back(2); - // Without turbofish a type annotation would be needed on the left hand side - let array = slice.as_array::<2>(); -} -``` -```rust -fn double() -> u32 { - N * 2 -} -fn example() { - assert(double::<9>() == 18); - assert(double::<7 + 8>() == 30); -} -``` -```rust -trait MyTrait { - fn ten() -> Self; -} - -impl MyTrait for Field { - fn ten() -> Self { 10 } -} - -struct Foo { - inner: T -} - -impl Foo { - fn generic_method(_self: Self) -> U where U: MyTrait { - U::ten() - } -} - -fn example() { - let foo: Foo = Foo { inner: 1 }; - // Using a type other than `Field` here (e.g. u32) would fail as - // there is no matching impl for `u32: MyTrait`. - // - // Substituting the `10` on the left hand side of this assert - // with `10 as u32` would also fail with a type mismatch as we - // are expecting a `Field` from the right hand side. - assert(10 as u32 == foo.generic_method::()); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/globals.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/globals.md deleted file mode 100644 index 063a3d89248..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/globals.md +++ /dev/null @@ -1,72 +0,0 @@ ---- -title: Global Variables -description: - Learn about global variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, globals, global variables, constants] -sidebar_position: 8 ---- - -## Globals - - -Noir supports global variables. The global's type can be inferred by the compiler entirely: - -```rust -global N = 5; // Same as `global N: Field = 5` - -global TUPLE = (3, 2); - -fn main() { - assert(N == 5); - assert(N == TUPLE.0 + TUPLE.1); -} -``` - -:::info - -Globals can be defined as any expression, so long as they don't depend on themselves - otherwise there would be a dependency cycle! For example: - -```rust -global T = foo(T); // dependency error -``` - -::: - - -If they are initialized to a literal integer, globals can be used to specify an array's length: - -```rust -global N: Field = 2; - -fn main(y : [Field; N]) { - assert(y[0] == y[1]) -} -``` - -A global from another module can be imported or referenced externally like any other name: - -```rust -global N = 20; - -fn main() { - assert(my_submodule::N != N); -} - -mod my_submodule { - global N: Field = 10; -} -``` - -When a global is used, Noir replaces the name with its definition on each occurrence. -This means globals defined using function calls will repeat the call each time they're used: - -```rust -global RESULT = foo(); - -fn foo() -> [Field; 100] { ... } -``` - -This is usually fine since Noir will generally optimize any function call that does not -refer to a program input into a constant. It should be kept in mind however, if the called -function performs side-effects like `println`, as these will still occur on each use. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/lambdas.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/lambdas.md deleted file mode 100644 index be3c7e0b5ca..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/lambdas.md +++ /dev/null @@ -1,81 +0,0 @@ ---- -title: Lambdas -description: Learn how to use anonymous functions in Noir programming language. -keywords: [Noir programming language, lambda, closure, function, anonymous function] -sidebar_position: 9 ---- - -## Introduction - -Lambdas are anonymous functions. The syntax is `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -A block can be used as the body of a lambda, allowing you to declare local variables inside it: - -```rust -let cool = || { - let x = 100; - let y = 100; - x + y -} - -assert(cool() == 200); -``` - -## Closures - -Inside the body of a lambda, you can use variables defined in the enclosing function. Such lambdas are called **closures**. In this example `x` is defined inside `main` and is accessed from within the lambda: - -```rust -fn main() { - let x = 100; - let closure = || x + 150; - assert(closure() == 250); -} -``` - -## Passing closures to higher-order functions - -It may catch you by surprise that the following code fails to compile: - -```rust -fn foo(f: fn () -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // error :( -} -``` - -The reason is that the closure's capture environment affects its type - we have a closure that captures two Fields and `foo` -expects a regular function as an argument - those are incompatible. -:::note - -Variables contained within the `||` are the closure's parameters, and the expression that follows it is the closure's body. The capture environment is comprised of any variables used in the closure's body that are not parameters. - -E.g. in |x| x + y, y would be a captured variable, but x would not be, since it is a parameter of the closure. - -::: -The syntax for the type of a closure is `fn[env](args) -> ret_type`, where `env` is the capture environment of the closure - -in this example that's `(Field, Field)`. - -The best solution in our case is to make `foo` generic over the environment type of its parameter, so that it can be called -with closures with any environment, as well as with regular functions: - -```rust -fn foo(f: fn[Env]() -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // compiles fine - assert(foo(|| 60) == 60); // compiles fine -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/mutability.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/mutability.md deleted file mode 100644 index fdeef6a87c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/mutability.md +++ /dev/null @@ -1,121 +0,0 @@ ---- -title: Mutability -description: - Learn about mutable variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, mutability in noir, mutable variables] -sidebar_position: 8 ---- - -Variables in noir can be declared mutable via the `mut` keyword. Mutable variables can be reassigned -to via an assignment expression. - -```rust -let x = 2; -x = 3; // error: x must be mutable to be assigned to - -let mut y = 3; -let y = 4; // OK -``` - -The `mut` modifier can also apply to patterns: - -```rust -let (a, mut b) = (1, 2); -a = 11; // error: a must be mutable to be assigned to -b = 12; // OK - -let mut (c, d) = (3, 4); -c = 13; // OK -d = 14; // OK - -// etc. -let MyStruct { x: mut y } = MyStruct { x: a }; -// y is now in scope -``` - -Note that mutability in noir is local and everything is passed by value, so if a called function -mutates its parameters then the parent function will keep the old value of the parameters. - -```rust -fn main() -> pub Field { - let x = 3; - helper(x); - x // x is still 3 -} - -fn helper(mut x: i32) { - x = 4; -} -``` - -## Non-local mutability - -Non-local mutability can be achieved through the mutable reference type `&mut T`: - -```rust -fn set_to_zero(x: &mut Field) { - *x = 0; -} - -fn main() { - let mut y = 42; - set_to_zero(&mut y); - assert(*y == 0); -} -``` - -When creating a mutable reference, the original variable being referred to (`y` in this -example) must also be mutable. Since mutable references are a reference type, they must -be explicitly dereferenced via `*` to retrieve the underlying value. Note that this yields -a copy of the value, so mutating this copy will not change the original value behind the -reference: - -```rust -fn main() { - let mut x = 1; - let x_ref = &mut x; - - let mut y = *x_ref; - let y_ref = &mut y; - - x = 2; - *x_ref = 3; - - y = 4; - *y_ref = 5; - - assert(x == 3); - assert(*x_ref == 3); - assert(y == 5); - assert(*y_ref == 5); -} -``` - -Note that types in Noir are actually deeply immutable so the copy that occurs when -dereferencing is only a conceptual copy - no additional constraints will occur. - -Mutable references can also be stored within structs. Note that there is also -no lifetime parameter on these unlike rust. This is because the allocated memory -always lasts the entire program - as if it were an array of one element. - -```rust -struct Foo { - x: &mut Field -} - -impl Foo { - fn incr(mut self) { - *self.x += 1; - } -} - -fn main() { - let foo = Foo { x: &mut 0 }; - foo.incr(); - assert(*foo.x == 1); -} -``` - -In general, you should avoid non-local & shared mutability unless it is needed. Sticking -to only local mutability will improve readability and potentially improve compiler optimizations as well. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/ops.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/ops.md deleted file mode 100644 index c35c36c38a9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/ops.md +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: Logical Operations -description: - Learn about the supported arithmetic and logical operations in the Noir programming language. - Discover how to perform operations on private input types, integers, and booleans. -keywords: - [ - Noir programming language, - supported operations, - arithmetic operations, - logical operations, - predicate operators, - bitwise operations, - short-circuiting, - backend, - ] -sidebar_position: 3 ---- - -# Operations - -## Table of Supported Operations - -| Operation | Description | Requirements | -| :-------- | :------------------------------------------------------------: | -------------------------------------: | -| + | Adds two private input types together | Types must be private input | -| - | Subtracts two private input types together | Types must be private input | -| \* | Multiplies two private input types together | Types must be private input | -| / | Divides two private input types together | Types must be private input | -| ^ | XOR two private input types together | Types must be integer | -| & | AND two private input types together | Types must be integer | -| \| | OR two private input types together | Types must be integer | -| \<\< | Left shift an integer by another integer amount | Types must be integer, shift must be u8 | -| >> | Right shift an integer by another integer amount | Types must be integer, shift must be u8 | -| ! | Bitwise not of a value | Type must be integer or boolean | -| \< | returns a bool if one value is less than the other | Upper bound must have a known bit size | -| \<= | returns a bool if one value is less than or equal to the other | Upper bound must have a known bit size | -| > | returns a bool if one value is more than the other | Upper bound must have a known bit size | -| >= | returns a bool if one value is more than or equal to the other | Upper bound must have a known bit size | -| == | returns a bool if one value is equal to the other | Both types must not be constants | -| != | returns a bool if one value is not equal to the other | Both types must not be constants | - -### Predicate Operators - -`<,<=, !=, == , >, >=` are known as predicate/comparison operations because they compare two values. -This differs from the operations such as `+` where the operands are used in _computation_. - -### Bitwise Operations Example - -```rust -fn main(x : Field) { - let y = x as u32; - let z = y & y; -} -``` - -`z` is implicitly constrained to be the result of `y & y`. The `&` operand is used to denote bitwise -`&`. - -> `x & x` would not compile as `x` is a `Field` and not an integer type. - -### Logical Operators - -Noir has no support for the logical operators `||` and `&&`. This is because encoding the -short-circuiting that these operators require can be inefficient for Noir's backend. Instead you can -use the bitwise operators `|` and `&` which operate identically for booleans, just without the -short-circuiting. - -```rust -let my_val = 5; - -let mut flag = 1; -if (my_val > 6) | (my_val == 0) { - flag = 0; -} -assert(flag == 1); - -if (my_val != 10) & (my_val < 50) { - flag = 0; -} -assert(flag == 0); -``` - -### Shorthand operators - -Noir shorthand operators for most of the above operators, namely `+=, -=, *=, /=, %=, &=, |=, ^=, <<=`, and `>>=`. These allow for more concise syntax. For example: - -```rust -let mut i = 0; -i = i + 1; -``` - -could be written as: - -```rust -let mut i = 0; -i += 1; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/oracles.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/oracles.mdx deleted file mode 100644 index 77a2ac1550a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/oracles.mdx +++ /dev/null @@ -1,29 +0,0 @@ ---- -title: Oracles -description: Dive into how Noir supports Oracles via RPC calls, and learn how to declare an Oracle in Noir with our comprehensive guide. -keywords: - - Noir - - Oracles - - RPC Calls - - Unconstrained Functions - - Programming - - Blockchain -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -Noir has support for Oracles via RPC calls. This means Noir will make an RPC call and use the return value for proof generation. - -Since Oracles are not resolved by Noir, they are [`unconstrained` functions](./unconstrained.md) - -You can declare an Oracle through the `#[oracle()]` flag. Example: - -```rust -#[oracle(get_number_sequence)] -unconstrained fn get_number_sequence(_size: Field) -> [Field] {} -``` - -The timeout for when using an external RPC oracle resolver can be set with the `NARGO_FOREIGN_CALL_TIMEOUT` environment variable. This timeout is in units of milliseconds. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/shadowing.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/shadowing.md deleted file mode 100644 index 5ce6130d201..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/shadowing.md +++ /dev/null @@ -1,44 +0,0 @@ ---- -title: Shadowing -sidebar_position: 12 ---- - -Noir allows for inheriting variables' values and re-declaring them with the same name similar to Rust, known as shadowing. - -For example, the following function is valid in Noir: - -```rust -fn main() { - let x = 5; - - { - let x = x * 2; - assert (x == 10); - } - - assert (x == 5); -} -``` - -In this example, a variable x is first defined with the value 5. - -The local scope that follows shadows the original x, i.e. creates a local mutable x based on the value of the original x. It is given a value of 2 times the original x. - -When we return to the main scope, x once again refers to just the original x, which stays at the value of 5. - -## Temporal mutability - -One way that shadowing is useful, in addition to ergonomics across scopes, is for temporarily mutating variables. - -```rust -fn main() { - let age = 30; - // age = age + 5; // Would error as `age` is immutable by default. - - let mut age = age + 5; // Temporarily mutates `age` with a new value. - - let age = age; // Locks `age`'s mutability again. - - assert (age == 35); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/traits.md deleted file mode 100644 index 51305b38c16..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/traits.md +++ /dev/null @@ -1,405 +0,0 @@ ---- -title: Traits -description: - Traits in Noir can be used to abstract out a common interface for functions across - several data types. -keywords: [noir programming language, traits, interfaces, generic, protocol] -sidebar_position: 14 ---- - -## Overview - -Traits in Noir are a useful abstraction similar to interfaces or protocols in other languages. Each trait defines -the interface of several methods contained within the trait. Types can then implement this trait by providing -implementations for these methods. For example in the program: - -```rust -struct Rectangle { - width: Field, - height: Field, -} - -impl Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -fn log_area(r: Rectangle) { - println(r.area()); -} -``` - -We have a function `log_area` to log the area of a `Rectangle`. Now how should we change the program if we want this -function to work on `Triangle`s as well?: - -```rust -struct Triangle { - width: Field, - height: Field, -} - -impl Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Making `log_area` generic over all types `T` would be invalid since not all types have an `area` method. Instead, we can -introduce a new `Area` trait and make `log_area` generic over all types `T` that implement `Area`: - -```rust -trait Area { - fn area(self) -> Field; -} - -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -We also need to explicitly implement `Area` for `Rectangle` and `Triangle`. We can do that by changing their existing -impls slightly. Note that the parameter types and return type of each of our `area` methods must match those defined -by the `Area` trait. - -```rust -impl Area for Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -impl Area for Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Now we have a working program that is generic over any type of Shape that is used! Others can even use this program -as a library with their own types - such as `Circle` - as long as they also implement `Area` for these types. - -## Where Clauses - -As seen in `log_area` above, when we want to create a function or method that is generic over any type that implements -a trait, we can add a where clause to the generic function. - -```rust -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -It is also possible to apply multiple trait constraints on the same variable at once by combining traits with the `+` -operator. Similarly, we can have multiple trait constraints by separating each with a comma: - -```rust -fn foo(elements: [T], thing: U) where - T: Default + Add + Eq, - U: Bar, -{ - let mut sum = T::default(); - - for element in elements { - sum += element; - } - - if sum == T::default() { - thing.bar(); - } -} -``` - -## Generic Implementations - -You can add generics to a trait implementation by adding the generic list after the `impl` keyword: - -```rust -trait Second { - fn second(self) -> Field; -} - -impl Second for (T, Field) { - fn second(self) -> Field { - self.1 - } -} -``` - -You can also implement a trait for every type this way: - -```rust -trait Debug { - fn debug(self); -} - -impl Debug for T { - fn debug(self) { - println(self); - } -} - -fn main() { - 1.debug(); -} -``` - -### Generic Trait Implementations With Where Clauses - -Where clauses can be placed on trait implementations themselves to restrict generics in a similar way. -For example, while `impl Foo for T` implements the trait `Foo` for every type, `impl Foo for T where T: Bar` -will implement `Foo` only for types that also implement `Bar`. This is often used for implementing generic types. -For example, here is the implementation for array equality: - -```rust -impl Eq for [T; N] where T: Eq { - // Test if two arrays have the same elements. - // Because both arrays must have length N, we know their lengths already match. - fn eq(self, other: Self) -> bool { - let mut result = true; - - for i in 0 .. self.len() { - // The T: Eq constraint is needed to call == on the array elements here - result &= self[i] == other[i]; - } - - result - } -} -``` - -Where clauses can also be placed on struct implementations. -For example, here is a method utilizing a generic type that implements the equality trait. - -```rust -struct Foo { - a: u32, - b: T, -} - -impl Foo where T: Eq { - fn eq(self, other: Self) -> bool { - (self.a == other.a) & self.b.eq(other.b) - } -} -``` - -## Generic Traits - -Traits themselves can also be generic by placing the generic arguments after the trait name. These generics are in -scope of every item within the trait. - -```rust -trait Into { - // Convert `self` to type `T` - fn into(self) -> T; -} -``` - -When implementing generic traits the generic arguments of the trait must be specified. This is also true anytime -when referencing a generic trait (e.g. in a `where` clause). - -```rust -struct MyStruct { - array: [Field; 2], -} - -impl Into<[Field; 2]> for MyStruct { - fn into(self) -> [Field; 2] { - self.array - } -} - -fn as_array(x: T) -> [Field; 2] - where T: Into<[Field; 2]> -{ - x.into() -} - -fn main() { - let array = [1, 2]; - let my_struct = MyStruct { array }; - - assert_eq(as_array(my_struct), array); -} -``` - -## Trait Methods With No `self` - -A trait can contain any number of methods, each of which have access to the `Self` type which represents each type -that eventually implements the trait. Similarly, the `self` variable is available as well but is not required to be used. -For example, we can define a trait to create a default value for a type. This trait will need to return the `Self` type -but doesn't need to take any parameters: - -```rust -trait Default { - fn default() -> Self; -} -``` - -Implementing this trait can be done similarly to any other trait: - -```rust -impl Default for Field { - fn default() -> Field { - 0 - } -} - -struct MyType {} - -impl Default for MyType { - fn default() -> Field { - MyType {} - } -} -``` - -However, since there is no `self` parameter, we cannot call it via the method call syntax `object.method()`. -Instead, we'll need to refer to the function directly. This can be done either by referring to the -specific impl `MyType::default()` or referring to the trait itself `Default::default()`. In the later -case, type inference determines the impl that is selected. - -```rust -let my_struct = MyStruct::default(); - -let x: Field = Default::default(); -let result = x + Default::default(); -``` - -:::warning - -```rust -let _ = Default::default(); -``` - -If type inference cannot select which impl to use because of an ambiguous `Self` type, an impl will be -arbitrarily selected. This occurs most often when the result of a trait function call with no parameters -is unused. To avoid this, when calling a trait function with no `self` or `Self` parameters or return type, -always refer to it via the implementation type's namespace - e.g. `MyType::default()`. -This is set to change to an error in future Noir versions. - -::: - -## Default Method Implementations - -A trait can also have default implementations of its methods by giving a body to the desired functions. -Note that this body must be valid for all types that may implement the trait. As a result, the only -valid operations on `self` will be operations valid for any type or other operations on the trait itself. - -```rust -trait Numeric { - fn add(self, other: Self) -> Self; - - // Default implementation of double is (self + self) - fn double(self) -> Self { - self.add(self) - } -} -``` - -When implementing a trait with default functions, a type may choose to implement only the required functions: - -```rust -impl Numeric for Field { - fn add(self, other: Field) -> Field { - self + other - } -} -``` - -Or it may implement the optional methods as well: - -```rust -impl Numeric for u32 { - fn add(self, other: u32) -> u32 { - self + other - } - - fn double(self) -> u32 { - self * 2 - } -} -``` - -## Impl Specialization - -When implementing traits for a generic type it is possible to implement the trait for only a certain combination -of generics. This can be either as an optimization or because those specific generics are required to implement the trait. - -```rust -trait Sub { - fn sub(self, other: Self) -> Self; -} - -struct NonZero { - value: T, -} - -impl Sub for NonZero { - fn sub(self, other: Self) -> Self { - let value = self.value - other.value; - assert(value != 0); - NonZero { value } - } -} -``` - -## Overlapping Implementations - -Overlapping implementations are disallowed by Noir to ensure Noir's decision on which impl to select is never ambiguous. -This means if a trait `Foo` is already implemented -by a type `Bar` for all `T`, then we cannot also have a separate impl for `Bar` (or any other -type argument). Similarly, if there is an impl for all `T` such as `impl Debug for T`, we cannot create -any more impls to `Debug` for other types since it would be ambiguous which impl to choose for any given -method call. - -```rust -trait Trait {} - -// Previous impl defined here -impl Trait for (A, B) {} - -// error: Impl for type `(Field, Field)` overlaps with existing impl -impl Trait for (Field, Field) {} -``` - -## Trait Coherence - -Another restriction on trait implementations is coherence. This restriction ensures other crates cannot create -impls that may overlap with other impls, even if several unrelated crates are used as dependencies in the same -program. - -The coherence restriction is: to implement a trait, either the trait itself or the object type must be declared -in the crate the impl is in. - -In practice this often comes up when using types provided by libraries. If a library provides a type `Foo` that does -not implement a trait in the standard library such as `Default`, you may not `impl Default for Foo` in your own crate. -While restrictive, this prevents later issues or silent changes in the program if the `Foo` library later added its -own impl for `Default`. If you are a user of the `Foo` library in this scenario and need a trait not implemented by the -library your choices are to either submit a patch to the library or use the newtype pattern. - -### The Newtype Pattern - -The newtype pattern gets around the coherence restriction by creating a new wrapper type around the library type -that we cannot create `impl`s for. Since the new wrapper type is defined in our current crate, we can create -impls for any trait we need on it. - -```rust -struct Wrapper { - foo: some_library::Foo, -} - -impl Default for Wrapper { - fn default() -> Wrapper { - Wrapper { - foo: some_library::Foo::new(), - } - } -} -``` - -Since we have an impl for our own type, the behavior of this code will not change even if `some_library` is updated -to provide its own `impl Default for Foo`. The downside of this pattern is that it requires extra wrapping and -unwrapping of values when converting to and from the `Wrapper` and `Foo` types. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/unconstrained.md deleted file mode 100644 index 96f824c5e42..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/unconstrained.md +++ /dev/null @@ -1,99 +0,0 @@ ---- -title: Unconstrained Functions -description: "Learn about what unconstrained functions in Noir are, how to use them and when you'd want to." - -keywords: [Noir programming language, unconstrained, open] -sidebar_position: 5 ---- - -Unconstrained functions are functions which do not constrain any of the included computation and allow for non-deterministic computation. - -## Why? - -Zero-knowledge (ZK) domain-specific languages (DSL) enable developers to generate ZK proofs from their programs by compiling code down to the constraints of an NP complete language (such as R1CS or PLONKish languages). However, the hard bounds of a constraint system can be very limiting to the functionality of a ZK DSL. - -Enabling a circuit language to perform unconstrained execution is a powerful tool. Said another way, unconstrained execution lets developers generate witnesses from code that does not generate any constraints. Being able to execute logic outside of a circuit is critical for both circuit performance and constructing proofs on information that is external to a circuit. - -Fetching information from somewhere external to a circuit can also be used to enable developers to improve circuit efficiency. - -A ZK DSL does not just prove computation, but proves that some computation was handled correctly. Thus, it is necessary that when we switch from performing some operation directly inside of a circuit to inside of an unconstrained environment that the appropriate constraints are still laid down elsewhere in the circuit. - -## Example - -An in depth example might help drive the point home. This example comes from the excellent [post](https://discord.com/channels/1113924620781883405/1124022445054111926/1128747641853972590) by Tom in the Noir Discord. - -Let's look at how we can optimize a function to turn a `u72` into an array of `u8`s. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u72 & 0xff) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 91 -Backend circuit size: 3619 -``` - -A lot of the operations in this function are optimized away by the compiler (all the bit-shifts turn into divisions by constants). However we can save a bunch of gates by casting to u8 a bit earlier. This automatically truncates the bit-shifted value to fit in a u8 which allows us to remove the AND against 0xff. This saves us ~480 gates in total. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 75 -Backend circuit size: 3143 -``` - -Those are some nice savings already but we can do better. This code is all constrained so we're proving every step of calculating out using num, but we don't actually care about how we calculate this, just that it's correct. This is where brillig comes in. - -It turns out that truncating a u72 into a u8 is hard to do inside a snark, each time we do as u8 we lay down 4 ACIR opcodes which get converted into multiple gates. It's actually much easier to calculate num from out than the other way around. All we need to do is multiply each element of out by a constant and add them all together, both relatively easy operations inside a snark. - -We can then run u72_to_u8 as unconstrained brillig code in order to calculate out, then use that result in our constrained function and assert that if we were to do the reverse calculation we'd get back num. This looks a little like the below: - -```rust -fn main(num: u72) -> pub [u8; 8] { - let out = u72_to_u8(num); - - let mut reconstructed_num: u72 = 0; - for i in 0..8 { - reconstructed_num += (out[i] as u72 << (56 - (8 * i))); - } - assert(num == reconstructed_num); - out -} - -unconstrained fn u72_to_u8(num: u72) -> [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8))) as u8; - } - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 78 -Backend circuit size: 2902 -``` - -This ends up taking off another ~250 gates from our circuit! We've ended up with more ACIR opcodes than before but they're easier for the backend to prove (resulting in fewer gates). - -Generally we want to use brillig whenever there's something that's easy to verify but hard to compute within the circuit. For example, if you wanted to calculate a square root of a number it'll be a much better idea to calculate this in brillig and then assert that if you square the result you get back your number. - -## Break and Continue - -In addition to loops over runtime bounds, `break` and `continue` are also available in unconstrained code. See [break and continue](../concepts/control_flow.md#break-and-continue) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/_category_.json deleted file mode 100644 index 1debcfe7675..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Modules, Packages and Crates", - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/crates_and_packages.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/crates_and_packages.md deleted file mode 100644 index 95ee9f52ab2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/crates_and_packages.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Crates and Packages -description: Learn how to use Crates and Packages in your Noir project -keywords: [Nargo, dependencies, package management, crates, package] -sidebar_position: 0 ---- - -## Crates - -A crate is the smallest amount of code that the Noir compiler considers at a time. -Crates can contain modules, and the modules may be defined in other files that get compiled with the crate, as we’ll see in the coming sections. - -### Crate Types - -A Noir crate can come in several forms: binaries, libraries or contracts. - -#### Binaries - -_Binary crates_ are programs which you can compile to an ACIR circuit which you can then create proofs against. Each must have a function called `main` that defines the ACIR circuit which is to be proved. - -#### Libraries - -_Library crates_ don't have a `main` function and they don't compile down to ACIR. Instead they define functionality intended to be shared with multiple projects, and eventually included in a binary crate. - -#### Contracts - -Contract crates are similar to binary crates in that they compile to ACIR which you can create proofs against. They are different in that they do not have a single `main` function, but are a collection of functions to be deployed to the [Aztec network](https://aztec.network). You can learn more about the technical details of Aztec in the [monorepo](https://github.com/AztecProtocol/aztec-packages) or contract [examples](https://github.com/AztecProtocol/aztec-packages/tree/master/noir-projects/noir-contracts/contracts). - -### Crate Root - -Every crate has a root, which is the source file that the compiler starts, this is also known as the root module. The Noir compiler does not enforce any conditions on the name of the file which is the crate root, however if you are compiling via Nargo the crate root must be called `lib.nr` or `main.nr` for library or binary crates respectively. - -## Packages - -A Nargo _package_ is a collection of one of more crates that provides a set of functionality. A package must include a Nargo.toml file. - -A package _must_ contain either a library or a binary crate, but not both. - -### Differences from Cargo Packages - -One notable difference between Rust's Cargo and Noir's Nargo is that while Cargo allows a package to contain an unlimited number of binary crates and a single library crate, Nargo currently only allows a package to contain a single crate. - -In future this restriction may be lifted to allow a Nargo package to contain both a binary and library crate or multiple binary crates. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/modules.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/modules.md deleted file mode 100644 index 16b6307d2fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/modules.md +++ /dev/null @@ -1,185 +0,0 @@ ---- -title: Modules -description: - Learn how to organize your files using modules in Noir, following the same convention as Rust's - module system. Examples included. -keywords: [Noir, Rust, modules, organizing files, sub-modules] -sidebar_position: 2 ---- - -Noir's module system follows the same convention as the _newer_ version of Rust's module system. - -## Purpose of Modules - -Modules are used to organize files. Without modules all of your code would need to live in a single -file. In Noir, the compiler does not automatically scan all of your files to detect modules. This -must be done explicitly by the developer. - -## Examples - -### Importing a module in the crate root - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo.nr` - -```rust -fn from_foo() {} -``` - -In the above snippet, the crate root is the `src/main.nr` file. The compiler sees the module -declaration `mod foo` which prompts it to look for a foo.nr file. - -Visually this module hierarchy looks like the following : - -``` -crate - ├── main - │ - └── foo - └── from_foo - -``` - -The module filename may also be the name of the module as a directory with the contents in a -file named `mod.nr` within that directory. The above example can alternatively be expressed like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -fn from_foo() {} -``` - -Note that it's an error to have both files `src/foo.nr` and `src/foo/mod.nr` in the filesystem. - -### Importing a module throughout the tree - -All modules are accessible from the `crate::` namespace. - -``` -crate - ├── bar - ├── foo - └── main - -``` - -In the above snippet, if `bar` would like to use functions in `foo`, it can do so by `use crate::foo::function_name`. - -### Sub-modules - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -fn from_bar() {} -``` - -In the above snippet, we have added an extra module to the module tree; `bar`. `bar` is a submodule -of `foo` hence we declare bar in `foo.nr` with `mod bar`. Since `foo` is not the crate root, the -compiler looks for the file associated with the `bar` module in `src/foo/bar.nr` - -Visually the module hierarchy looks as follows: - -``` -crate - ├── main - │ - └── foo - ├── from_foo - └── bar - └── from_bar -``` - -Similar to importing a module in the crate root, modules can be placed in a `mod.nr` file, like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar/mod.nr` - -```rust -fn from_bar() {} -``` - -### Referencing a parent module - -Given a submodule, you can refer to its parent module using the `super` keyword. - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; - -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -// Same as bar::from_foo -use super::from_foo; - -fn from_bar() { - from_foo(); // invokes super::from_foo(), which is bar::from_foo() - super::from_foo(); // also invokes bar::from_foo() -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/workspaces.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/workspaces.md deleted file mode 100644 index 513497f12bf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/workspaces.md +++ /dev/null @@ -1,42 +0,0 @@ ---- -title: Workspaces -sidebar_position: 3 ---- - -Workspaces are a feature of nargo that allow you to manage multiple related Noir packages in a single repository. A workspace is essentially a group of related projects that share common build output directories and configurations. - -Each Noir project (with it's own Nargo.toml file) can be thought of as a package. Each package is expected to contain exactly one "named circuit", being the "name" defined in Nargo.toml with the program logic defined in `./src/main.nr`. - -For a project with the following structure: - -```tree -├── crates -│ ├── a -│ │ ├── Nargo.toml -│ │ └── Prover.toml -│ │ └── src -│ │ └── main.nr -│ └── b -│ ├── Nargo.toml -│ └── Prover.toml -│ └── src -│ └── main.nr -│ -└── Nargo.toml -``` - -You can define a workspace in Nargo.toml like so: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -`members` indicates which packages are included in the workspace. As such, all member packages of a workspace will be processed when the `--workspace` flag is used with various commands or if a `default-member` is not specified. - -`default-member` indicates which package various commands process by default. - -Libraries can be defined in a workspace. Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/_category_.json deleted file mode 100644 index af04c0933fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Standard Library", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/bigint.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/bigint.md deleted file mode 100644 index 2bfdeec6631..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/bigint.md +++ /dev/null @@ -1,122 +0,0 @@ ---- -title: Big Integers -description: How to use big integers from Noir standard library -keywords: - [ - Big Integer, - Noir programming language, - Noir libraries, - ] ---- - -The BigInt module in the standard library exposes some class of integers which do not fit (well) into a Noir native field. It implements modulo arithmetic, modulo a 'big' prime number. - -:::note - -The module can currently be considered as `Field`s with fixed modulo sizes used by a set of elliptic curves, in addition to just the native curve. [More work](https://github.com/noir-lang/noir/issues/510) is needed to achieve arbitrarily sized big integers. - -::: - -Currently 6 classes of integers (i.e 'big' prime numbers) are available in the module, namely: - -- BN254 Fq: Bn254Fq -- BN254 Fr: Bn254Fr -- Secp256k1 Fq: Secpk1Fq -- Secp256k1 Fr: Secpk1Fr -- Secp256r1 Fr: Secpr1Fr -- Secp256r1 Fq: Secpr1Fq - -Where XXX Fq and XXX Fr denote respectively the order of the base and scalar field of the (usual) elliptic curve XXX. -For instance the big integer 'Secpk1Fq' in the standard library refers to integers modulo $2^{256}-2^{32}-977$. - -Feel free to explore the source code for the other primes: - -```rust title="big_int_definition" showLineNumbers -struct BigInt { - pointer: u32, - modulus: u32, -} -``` -> Source code: noir_stdlib/src/bigint.nr#L14-L19 - - -## Example usage - -A common use-case is when constructing a big integer from its bytes representation, and performing arithmetic operations on it: - -```rust title="big_int_example" showLineNumbers -fn big_int_example(x: u8, y: u8) { - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - let b = Secpk1Fq::from_le_bytes(&[y, x, 9]); - let c = (a + b) * b / a; - let d = c.to_le_bytes(); - println(d[0]); -} -``` -> Source code: test_programs/execution_success/bigint/src/main.nr#L70-L78 - - -## Methods - -The available operations for each big integer are: - -### from_le_bytes - -Construct a big integer from its little-endian bytes representation. Example: - -```rust - // Construct a big integer from a slice of bytes - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - // Construct a big integer from an array of 32 bytes - let a = Secpk1Fq::from_le_bytes_32([1;32]); - ``` - -Sure, here's the formatted version of the remaining methods: - -### to_le_bytes - -Return the little-endian bytes representation of a big integer. Example: - -```rust -let bytes = a.to_le_bytes(); -``` - -### add - -Add two big integers. Example: - -```rust -let sum = a + b; -``` - -### sub - -Subtract two big integers. Example: - -```rust -let difference = a - b; -``` - -### mul - -Multiply two big integers. Example: - -```rust -let product = a * b; -``` - -### div - -Divide two big integers. Note that division is field division and not euclidean division. Example: - -```rust -let quotient = a / b; -``` - -### eq - -Compare two big integers. Example: - -```rust -let are_equal = a == b; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/black_box_fns.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/black_box_fns.md deleted file mode 100644 index d6079ab182c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/black_box_fns.md +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Black Box Functions -description: Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. -keywords: [noir, black box functions] ---- - -Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. This makes certain zk-snark unfriendly computations cheaper than if they were implemented in Noir. - -The ACVM spec defines a set of blackbox functions which backends will be expected to implement. This allows backends to use optimized implementations of these constraints if they have them, however they may also fallback to less efficient naive implementations if not. - -## Function list - -Here is a list of the current black box functions: - -- [AES128](./cryptographic_primitives/ciphers.mdx#aes128) -- [SHA256](./cryptographic_primitives/hashes.mdx#sha256) -- [Schnorr signature verification](./cryptographic_primitives/schnorr.mdx) -- [Blake2s](./cryptographic_primitives/hashes.mdx#blake2s) -- [Blake3](./cryptographic_primitives/hashes.mdx#blake3) -- [Pedersen Hash](./cryptographic_primitives/hashes.mdx#pedersen_hash) -- [Pedersen Commitment](./cryptographic_primitives/hashes.mdx#pedersen_commitment) -- [ECDSA signature verification](./cryptographic_primitives/ecdsa_sig_verification.mdx) -- [Embedded curve operations (MSM, addition, ...)](./cryptographic_primitives/embedded_curve_ops.mdx) -- AND -- XOR -- RANGE -- [Keccak256](./cryptographic_primitives/hashes.mdx#keccak256) -- [Recursive proof verification](./recursion.mdx) - -Most black box functions are included as part of the Noir standard library, however `AND`, `XOR` and `RANGE` are used as part of the Noir language syntax. For instance, using the bitwise operator `&` will invoke the `AND` black box function. - -You can view the black box functions defined in the ACVM code [here](https://github.com/noir-lang/noir/blob/master/acvm-repo/acir/src/circuit/black_box_functions.rs). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/bn254.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/bn254.md deleted file mode 100644 index 3294f005dbb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/bn254.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Bn254 Field Library ---- - -Noir provides a module in standard library with some optimized functions for bn254 Fr in `std::field::bn254`. - -## decompose - -```rust -fn decompose(x: Field) -> (Field, Field) {} -``` - -Decomposes a single field into two fields, low and high. The low field contains the lower 16 bytes of the input field and the high field contains the upper 16 bytes of the input field. Both field results are range checked to 128 bits. - - -## assert_gt - -```rust -fn assert_gt(a: Field, b: Field) {} -``` - -Asserts that a > b. This will generate less constraints than using `assert(gt(a, b))`. - -## assert_lt - -```rust -fn assert_lt(a: Field, b: Field) {} -``` - -Asserts that a < b. This will generate less constraints than using `assert(lt(a, b))`. - -## gt - -```rust -fn gt(a: Field, b: Field) -> bool {} -``` - -Returns true if a > b. - -## lt - -```rust -fn lt(a: Field, b: Field) -> bool {} -``` - -Returns true if a < b. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/boundedvec.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/boundedvec.md deleted file mode 100644 index 604d84d5ba4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/boundedvec.md +++ /dev/null @@ -1,419 +0,0 @@ ---- -title: Bounded Vectors -keywords: [noir, vector, bounded vector, slice] -sidebar_position: 1 ---- - -A `BoundedVec` is a growable storage similar to a `Vec` except that it -is bounded with a maximum possible length. Unlike `Vec`, `BoundedVec` is not implemented -via slices and thus is not subject to the same restrictions slices are (notably, nested -slices - and thus nested vectors as well - are disallowed). - -Since a BoundedVec is backed by a normal array under the hood, growing the BoundedVec by -pushing an additional element is also more efficient - the length only needs to be increased -by one. - -For these reasons `BoundedVec` should generally be preferred over `Vec` when there -is a reasonable maximum bound that can be placed on the vector. - -Example: - -```rust -let mut vector: BoundedVec = BoundedVec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -assert(vector.max_len() == 10); -``` - -## Methods - -### new - -```rust -pub fn new() -> Self -``` - -Creates a new, empty vector of length zero. - -Since this container is backed by an array internally, it still needs an initial value -to give each element. To resolve this, each element is zeroed internally. This value -is guaranteed to be inaccessible unless `get_unchecked` is used. - -Example: - -```rust -let empty_vector: BoundedVec = BoundedVec::new(); -assert(empty_vector.len() == 0); -``` - -Note that whenever calling `new` the maximum length of the vector should always be specified -via a type signature: - -```rust title="new_example" showLineNumbers -fn foo() -> BoundedVec { - // Ok! MaxLen is specified with a type annotation - let v1: BoundedVec = BoundedVec::new(); - let v2 = BoundedVec::new(); - - // Ok! MaxLen is known from the type of foo's return value - v2 -} - -fn bad() { - let mut v3 = BoundedVec::new(); - - // Not Ok! We don't know if v3's MaxLen is at least 1, and the compiler often infers 0 by default. - v3.push(5); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L11-L27 - - -This defaulting of `MaxLen` (and numeric generics in general) to zero may change in future noir versions -but for now make sure to use type annotations when using bounded vectors. Otherwise, you will receive a constraint failure at runtime when the vec is pushed to. - -### get - -```rust -pub fn get(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this -will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - let last = v.get(v.len() - 1); - assert(first != last); -} -``` - -### get_unchecked - -```rust -pub fn get_unchecked(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero, without -performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, -it is unsafe! Use at your own risk! - -Example: - -```rust title="get_unchecked_example" showLineNumbers -fn sum_of_first_three(v: BoundedVec) -> u32 { - // Always ensure the length is larger than the largest - // index passed to get_unchecked - assert(v.len() > 2); - let first = v.get_unchecked(0); - let second = v.get_unchecked(1); - let third = v.get_unchecked(2); - first + second + third -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L54-L64 - - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - assert(first != 42); - v.set(0, 42); - let new_first = v.get(0); - assert(new_first == 42); -} -``` - -### set_unchecked - -```rust -pub fn set_unchecked(&mut self: Self, index: u64, value: T) -> T { -``` - -Writes an element to the vector at the given index, starting from zero, without performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, it is unsafe! Use at your own risk! - -Example: - -```rust title="set_unchecked_example" showLineNumbers -fn set_unchecked_example() { - let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([1, 2]); - - // Here we're safely writing within the valid range of `vec` - // `vec` now has the value [42, 2] - vec.set_unchecked(0, 42); - - // We can then safely read this value back out of `vec`. - // Notice that we use the checked version of `get` which would prevent reading unsafe values. - assert_eq(vec.get(0), 42); - - // We've now written past the end of `vec`. - // As this index is still within the maximum potential length of `v`, - // it won't cause a constraint failure. - vec.set_unchecked(2, 42); - println(vec); - - // This will write past the end of the maximum potential length of `vec`, - // it will then trigger a constraint failure. - vec.set_unchecked(5, 42); - println(vec); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L67-L91 - - - -### push - -```rust -pub fn push(&mut self, elem: T) { -``` - -Pushes an element to the end of the vector. This increases the length -of the vector by one. - -Panics if the new length of the vector will be greater than the max length. - -Example: - -```rust title="bounded-vec-push-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - v.push(1); - v.push(2); - - // Panics with failed assertion "push out of bounds" - v.push(3); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L95-L103 - - -### pop - -```rust -pub fn pop(&mut self) -> T -``` - -Pops the element at the end of the vector. This will decrease the length -of the vector by one. - -Panics if the vector is empty. - -Example: - -```rust title="bounded-vec-pop-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.push(1); - v.push(2); - - let two = v.pop(); - let one = v.pop(); - - assert(two == 2); - assert(one == 1); - // error: cannot pop from an empty vector - // let _ = v.pop(); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L108-L120 - - -### len - -```rust -pub fn len(self) -> u64 { -``` - -Returns the current length of this vector - -Example: - -```rust title="bounded-vec-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - assert(v.len() == 0); - - v.push(100); - assert(v.len() == 1); - - v.push(200); - v.push(300); - v.push(400); - assert(v.len() == 4); - - let _ = v.pop(); - let _ = v.pop(); - assert(v.len() == 2); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L125-L140 - - -### max_len - -```rust -pub fn max_len(_self: BoundedVec) -> u64 { -``` - -Returns the maximum length of this vector. This is always -equal to the `MaxLen` parameter this vector was initialized with. - -Example: - -```rust title="bounded-vec-max-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.max_len() == 5); - v.push(10); - assert(v.max_len() == 5); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L145-L151 - - -### storage - -```rust -pub fn storage(self) -> [T; MaxLen] { -``` - -Returns the internal array within this vector. -Since arrays in Noir are immutable, mutating the returned storage array will not mutate -the storage held internally by this vector. - -Note that uninitialized elements may be zeroed out! - -Example: - -```rust title="bounded-vec-storage-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.storage() == [0, 0, 0, 0, 0]); - - v.push(57); - assert(v.storage() == [57, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L156-L163 - - -### extend_from_array - -```rust -pub fn extend_from_array(&mut self, array: [T; Len]) -``` - -Pushes each element from the given array to this vector. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-array-example" showLineNumbers -let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([2, 4]); - - assert(vec.len == 2); - assert(vec.get(0) == 2); - assert(vec.get(1) == 4); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L168-L175 - - -### extend_from_bounded_vec - -```rust -pub fn extend_from_bounded_vec(&mut self, vec: BoundedVec) -``` - -Pushes each element from the other vector to this vector. The length of -the other vector is left unchanged. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-bounded-vec-example" showLineNumbers -let mut v1: BoundedVec = BoundedVec::new(); - let mut v2: BoundedVec = BoundedVec::new(); - - v2.extend_from_array([1, 2, 3]); - v1.extend_from_bounded_vec(v2); - - assert(v1.storage() == [1, 2, 3, 0, 0]); - assert(v2.storage() == [1, 2, 3, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L180-L189 - - -### from_array - -```rust -pub fn from_array(array: [T; Len]) -> Self -``` - -Creates a new vector, populating it with values derived from an array input. -The maximum length of the vector is determined based on the type signature. - -Example: -```rust -let bounded_vec: BoundedVec = BoundedVec::from_array([1, 2, 3]) -``` - -### map - -```rust -pub fn map(self, f: fn[Env](T) -> U) -> BoundedVec -``` - -Creates a new vector of equal size by calling a closure on each element in this vector. - -Example: - -```rust title="bounded-vec-map-example" showLineNumbers -let vec: BoundedVec = BoundedVec::from_array([1, 2, 3, 4]); - let result = vec.map(|value| value * 2); -``` -> Source code: noir_stdlib/src/collections/bounded_vec.nr#L205-L208 - - -### any - -```rust -pub fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -Returns true if the given predicate returns true for any element -in this vector. - -Example: - -```rust title="bounded-vec-any-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.extend_from_array([2, 4, 6]); - - let all_even = !v.any(|elem: u32| elem % 2 != 0); - assert(all_even); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L256-L262 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/hashmap.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/hashmap.md deleted file mode 100644 index 8c50c7e774c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/hashmap.md +++ /dev/null @@ -1,570 +0,0 @@ ---- -title: HashMap -keywords: [noir, map, hash, hashmap] -sidebar_position: 1 ---- - -`HashMap` is used to efficiently store and look up key-value pairs. - -`HashMap` is a bounded type which can store anywhere from zero to `MaxLen` total elements. -Note that due to hash collisions, the actual maximum number of elements stored by any particular -hashmap is likely lower than `MaxLen`. This is true even with cryptographic hash functions since -every hash value will be performed modulo `MaxLen`. - -When creating `HashMap`s, the `MaxLen` generic should always be specified if it is not already -known. Otherwise, the compiler may infer a different value for `MaxLen` (such as zero), which -will likely change the result of the program. This behavior is set to become an error in future -versions instead. - -Example: - -```rust -// Create a mapping from Fields to u32s with a maximum length of 12 -// using a poseidon2 hasher -use std::hash::poseidon2::Poseidon2Hasher; -let mut map: HashMap> = HashMap::default(); - -map.insert(1, 2); -map.insert(3, 4); - -let two = map.get(1).unwrap(); -``` - -## Methods - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default -{ - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L462-L469 - - -Creates a fresh, empty HashMap. - -When using this function, always make sure to specify the maximum size of the hash map. - -This is the same `default` from the `Default` implementation given further below. It is -repeated here for convenience since it is the recommended way to create a hashmap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L204 - - -Because `HashMap` has so many generic arguments that are likely to be the same throughout -your program, it may be helpful to create a type alias: - -```rust title="type_alias" showLineNumbers -type MyMap = HashMap>; -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L195-L197 - - -### with_hasher - -```rust title="with_hasher" showLineNumbers -pub fn with_hasher(_build_hasher: B) -> Self - where - B: BuildHasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L82-L86 - - -Creates a hashmap with an existing `BuildHasher`. This can be used to ensure multiple -hashmaps are created with the same hasher instance. - -Example: - -```rust title="with_hasher_example" showLineNumbers -let my_hasher: BuildHasherDefault = Default::default(); - let hashmap: HashMap> = HashMap::with_hasher(my_hasher); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L206-L210 - - -### get - -```rust title="get" showLineNumbers -pub fn get( - self, - key: K - ) -> Option - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L278-L287 - - -Retrieves a value from the hashmap, returning `Option::none()` if it was not found. - -Example: - -```rust title="get_example" showLineNumbers -fn get_example(map: HashMap>) { - let x = map.get(12); - - if x.is_some() { - assert(x.unwrap() == 42); - } -} -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L298-L306 - - -### insert - -```rust title="insert" showLineNumbers -pub fn insert( - &mut self, - key: K, - value: V - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L313-L323 - - -Inserts a new key-value pair into the map. If the key was already in the map, its -previous value will be overridden with the newly provided one. - -Example: - -```rust title="insert_example" showLineNumbers -let mut map: HashMap> = HashMap::default(); - map.insert(12, 42); - assert(map.len() == 1); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L212-L216 - - -### remove - -```rust title="remove" showLineNumbers -pub fn remove( - &mut self, - key: K - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L356-L365 - - -Removes the given key-value pair from the map. If the key was not already present -in the map, this does nothing. - -Example: - -```rust title="remove_example" showLineNumbers -map.remove(12); - assert(map.is_empty()); - - // If a key was not present in the map, remove does nothing - map.remove(12); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L220-L227 - - -### is_empty - -```rust title="is_empty" showLineNumbers -pub fn is_empty(self) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L115-L117 - - -True if the length of the hash map is empty. - -Example: - -```rust title="is_empty_example" showLineNumbers -assert(map.is_empty()); - - map.insert(1, 2); - assert(!map.is_empty()); - - map.remove(1); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L229-L237 - - -### len - -```rust title="len" showLineNumbers -pub fn len(self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L264-L266 - - -Returns the current length of this hash map. - -Example: - -```rust title="len_example" showLineNumbers -// This is equivalent to checking map.is_empty() - assert(map.len() == 0); - - map.insert(1, 2); - map.insert(3, 4); - map.insert(5, 6); - assert(map.len() == 3); - - // 3 was already present as a key in the hash map, so the length is unchanged - map.insert(3, 7); - assert(map.len() == 3); - - map.remove(1); - assert(map.len() == 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L239-L254 - - -### capacity - -```rust title="capacity" showLineNumbers -pub fn capacity(_self: Self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L271-L273 - - -Returns the maximum capacity of this hashmap. This is always equal to the capacity -specified in the hashmap's type. - -Unlike hashmaps in general purpose programming languages, hashmaps in Noir have a -static capacity that does not increase as the map grows larger. Thus, this capacity -is also the maximum possible element count that can be inserted into the hashmap. -Due to hash collisions (modulo the hashmap length), it is likely the actual maximum -element count will be lower than the full capacity. - -Example: - -```rust title="capacity_example" showLineNumbers -let empty_map: HashMap> = HashMap::default(); - assert(empty_map.len() == 0); - assert(empty_map.capacity() == 42); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L256-L260 - - -### clear - -```rust title="clear" showLineNumbers -pub fn clear(&mut self) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L93-L95 - - -Clears the hashmap, removing all key-value pairs from it. - -Example: - -```rust title="clear_example" showLineNumbers -assert(!map.is_empty()); - map.clear(); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L262-L266 - - -### contains_key - -```rust title="contains_key" showLineNumbers -pub fn contains_key( - self, - key: K - ) -> bool - where - K: Hash + Eq, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L101-L110 - - -True if the hashmap contains the given key. Unlike `get`, this will not also return -the value associated with the key. - -Example: - -```rust title="contains_key_example" showLineNumbers -if map.contains_key(7) { - let value = map.get(7); - assert(value.is_some()); - } else { - println("No value for key 7!"); - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L268-L275 - - -### entries - -```rust title="entries" showLineNumbers -pub fn entries(self) -> BoundedVec<(K, V), N> { -``` -> Source code: noir_stdlib/src/collections/map.nr#L123-L125 - - -Returns a vector of each key-value pair present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="entries_example" showLineNumbers -let entries = map.entries(); - - // The length of a hashmap may not be compile-time known, so we - // need to loop over its capacity instead - for i in 0..map.capacity() { - if i < entries.len() { - let (key, value) = entries.get(i); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L309-L320 - - -### keys - -```rust title="keys" showLineNumbers -pub fn keys(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L144-L146 - - -Returns a vector of each key present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="keys_example" showLineNumbers -let keys = map.keys(); - - for i in 0..keys.max_len() { - if i < keys.len() { - let key = keys.get_unchecked(i); - let value = map.get(key).unwrap_unchecked(); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L322-L332 - - -### values - -```rust title="values" showLineNumbers -pub fn values(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L164-L166 - - -Returns a vector of each value present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="values_example" showLineNumbers -let values = map.values(); - - for i in 0..values.max_len() { - if i < values.len() { - let value = values.get_unchecked(i); - println(f"Found value {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L334-L343 - - -### iter_mut - -```rust title="iter_mut" showLineNumbers -pub fn iter_mut( - &mut self, - f: fn(K, V) -> (K, V) - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L183-L192 - - -Iterates through each key-value pair of the HashMap, setting each key-value pair to the -result returned from the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If this is not desired, use `iter_values_mut` if only values need to be mutated, -or `entries` if neither keys nor values need to be mutated. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_mut_example" showLineNumbers -// Add 1 to each key in the map, and double the value associated with that key. - map.iter_mut(|k, v| (k + 1, v * 2)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L347-L350 - - -### iter_keys_mut - -```rust title="iter_keys_mut" showLineNumbers -pub fn iter_keys_mut( - &mut self, - f: fn(K) -> K - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L208-L217 - - -Iterates through the HashMap, mutating each key to the result returned from -the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If only iteration is desired and the keys are not intended to be mutated, -prefer using `entries` instead. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_keys_mut_example" showLineNumbers -// Double each key, leaving the value associated with that key untouched - map.iter_keys_mut(|k| k * 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L352-L355 - - -### iter_values_mut - -```rust title="iter_values_mut" showLineNumbers -pub fn iter_values_mut(&mut self, f: fn(V) -> V) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L233-L235 - - -Iterates through the HashMap, applying the given function to each value and mutating the -value to equal the result. This function is more efficient than `iter_mut` and `iter_keys_mut` -because the keys are untouched and the underlying hashmap thus does not need to be reordered. - -Example: - -```rust title="iter_values_mut_example" showLineNumbers -// Halve each value - map.iter_values_mut(|v| v / 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L357-L360 - - -### retain - -```rust title="retain" showLineNumbers -pub fn retain(&mut self, f: fn(K, V) -> bool) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L247-L249 - - -Retains only the key-value pairs for which the given function returns true. -Any key-value pairs for which the function returns false will be removed from the map. - -Example: - -```rust title="retain_example" showLineNumbers -map.retain(|k, v| (k != 0) & (v != 0)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L280-L282 - - -## Trait Implementations - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default -{ - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L462-L469 - - -Constructs an empty HashMap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L204 - - -### eq - -```rust title="eq" showLineNumbers -impl Eq for HashMap -where - K: Eq + Hash, - V: Eq, - B: BuildHasher, - H: Hasher -{ - fn eq(self, other: HashMap) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L426-L435 - - -Checks if two HashMaps are equal. - -Example: - -```rust title="eq_example" showLineNumbers -let mut map1: HashMap> = HashMap::default(); - let mut map2: HashMap> = HashMap::default(); - - map1.insert(1, 2); - map1.insert(3, 4); - - map2.insert(3, 4); - map2.insert(1, 2); - - assert(map1 == map2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L284-L295 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/index.md deleted file mode 100644 index ea84c6d5c21..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/index.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -title: Containers -description: Container types provided by Noir's standard library for storing and retrieving data -keywords: [containers, data types, vec, hashmap] ---- diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/vec.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/vec.mdx deleted file mode 100644 index 475011922f8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/containers/vec.mdx +++ /dev/null @@ -1,170 +0,0 @@ ---- -title: Vectors -description: Delve into the Vec data type in Noir. Learn about its methods, practical examples, and best practices for using Vectors in your Noir code. -keywords: [noir, vector type, methods, examples, dynamic arrays] -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A vector is a collection type similar to Rust's `Vec` type. In Noir, it is a convenient way to use slices as mutable arrays. - -Example: - -```rust -let mut vector: Vec = Vec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -``` - -## Methods - -### new - -Creates a new, empty vector. - -```rust -pub fn new() -> Self -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### from_slice - -Creates a vector containing each element from a given slice. Mutations to the resulting vector will not affect the original slice. - -```rust -pub fn from_slice(slice: [T]) -> Self -``` - -Example: - -```rust -let slice: [Field] = &[1, 2, 3]; -let vector_from_slice = Vec::from_slice(slice); -assert(vector_from_slice.len() == 3); -``` - -### len - -Returns the number of elements in the vector. - -```rust -pub fn len(self) -> Field -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### get - -Retrieves an element from the vector at a given index. Panics if the index points beyond the vector's end. - -```rust -pub fn get(self, index: Field) -> T -``` - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -``` - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -Panics if the index points beyond the vector's end. - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -vector.set(1, 42); -assert(vector.get(1) == 42); -``` - -### push - -Adds a new element to the vector's end, returning a new vector with a length one greater than the original unmodified vector. - -```rust -pub fn push(&mut self, elem: T) -``` - -Example: - -```rust -let mut vector: Vec = Vec::new(); -vector.push(10); -assert(vector.len() == 1); -``` - -### pop - -Removes an element from the vector's end, returning a new vector with a length one less than the original vector, along with the removed element. Panics if the vector's length is zero. - -```rust -pub fn pop(&mut self) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20]); -let popped_elem = vector.pop(); -assert(popped_elem == 20); -assert(vector.len() == 1); -``` - -### insert - -Inserts an element at a specified index, shifting subsequent elements to the right. - -```rust -pub fn insert(&mut self, index: Field, elem: T) -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 30]); -vector.insert(1, 20); -assert(vector.get(1) == 20); -``` - -### remove - -Removes an element at a specified index, shifting subsequent elements to the left, and returns the removed element. - -```rust -pub fn remove(&mut self, index: Field) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20, 30]); -let removed_elem = vector.remove(1); -assert(removed_elem == 20); -assert(vector.len() == 2); -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ciphers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ciphers.mdx deleted file mode 100644 index d6a5e1a79eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ciphers.mdx +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Ciphers -description: - Learn about the implemented ciphers ready to use for any Noir project -keywords: - [ciphers, Noir project, aes128, encrypt] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## aes128 - -Given a plaintext as an array of bytes, returns the corresponding aes128 ciphertext (CBC mode). Input padding is automatically performed using PKCS#7, so that the output length is `input.len() + (16 - input.len() % 16)`. - -```rust title="aes128" showLineNumbers -pub fn aes128_encrypt(input: [u8; N], iv: [u8; 16], key: [u8; 16]) -> [u8] {} -``` -> Source code: noir_stdlib/src/aes128.nr#L2-L4 - - -```rust -fn main() { - let input: [u8; 4] = [0, 12, 3, 15] // Random bytes, will be padded to 16 bytes. - let iv: [u8; 16] = [0; 16]; // Initialisation vector - let key: [u8; 16] = [0; 16] // AES key - let ciphertext = std::aes128::aes128_encrypt(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()); // In this case, the output length will be 16 bytes. -} -``` - - - \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ec_primitives.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ec_primitives.md deleted file mode 100644 index f262d8160d6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ec_primitives.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Elliptic Curve Primitives -keywords: [cryptographic primitives, Noir project] -sidebar_position: 4 ---- - -Data structures and methods on them that allow you to carry out computations involving elliptic -curves over the (mathematical) field corresponding to `Field`. For the field currently at our -disposal, applications would involve a curve embedded in BN254, e.g. the -[Baby Jubjub curve](https://eips.ethereum.org/EIPS/eip-2494). - -## Data structures - -### Elliptic curve configurations - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Curve`), i.e. the specific elliptic -curve you want to use, which would be specified using any one of the methods -`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::new` which take the coefficients in the -defining equation together with a generator point as parameters. You can find more detail in the -comments in -[`noir_stdlib/src/ec/mod.nr`](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr), but -the gist of it is that the elliptic curves of interest are usually expressed in one of the standard -forms implemented here (Twisted Edwards, Montgomery and Short Weierstraß), and in addition to that, -you could choose to use `affine` coordinates (Cartesian coordinates - the usual (x,y) - possibly -together with a point at infinity) or `curvegroup` coordinates (some form of projective coordinates -requiring more coordinates but allowing for more efficient implementations of elliptic curve -operations). Conversions between all of these forms are provided, and under the hood these -conversions are done whenever an operation is more efficient in a different representation (or a -mixed coordinate representation is employed). - -### Points - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Point`), i.e. points lying on the -elliptic curve. For a curve configuration `c` and a point `p`, it may be checked that `p` -does indeed lie on `c` by calling `c.contains(p1)`. - -## Methods - -(given a choice of curve representation, e.g. use `std::ec::tecurve::affine::Curve` and use -`std::ec::tecurve::affine::Point`) - -- The **zero element** is given by `Point::zero()`, and we can verify whether a point `p: Point` is - zero by calling `p.is_zero()`. -- **Equality**: Points `p1: Point` and `p2: Point` may be checked for equality by calling - `p1.eq(p2)`. -- **Addition**: For `c: Curve` and points `p1: Point` and `p2: Point` on the curve, adding these two - points is accomplished by calling `c.add(p1,p2)`. -- **Negation**: For a point `p: Point`, `p.negate()` is its negation. -- **Subtraction**: For `c` and `p1`, `p2` as above, subtracting `p2` from `p1` is accomplished by - calling `c.subtract(p1,p2)`. -- **Scalar multiplication**: For `c` as above, `p: Point` a point on the curve and `n: Field`, - scalar multiplication is given by `c.mul(n,p)`. If instead `n :: [u1; N]`, i.e. `n` is a bit - array, the `bit_mul` method may be used instead: `c.bit_mul(n,p)` -- **Multi-scalar multiplication**: For `c` as above and arrays `n: [Field; N]` and `p: [Point; N]`, - multi-scalar multiplication is given by `c.msm(n,p)`. -- **Coordinate representation conversions**: The `into_group` method converts a point or curve - configuration in the affine representation to one in the CurveGroup representation, and - `into_affine` goes in the other direction. -- **Curve representation conversions**: `tecurve` and `montcurve` curves and points are equivalent - and may be converted between one another by calling `into_montcurve` or `into_tecurve` on their - configurations or points. `swcurve` is more general and a curve c of one of the other two types - may be converted to this representation by calling `c.into_swcurve()`, whereas a point `p` lying - on the curve given by `c` may be mapped to its corresponding `swcurve` point by calling - `c.map_into_swcurve(p)`. -- **Map-to-curve methods**: The Elligator 2 method of mapping a field element `n: Field` into a - `tecurve` or `montcurve` with configuration `c` may be called as `c.elligator2_map(n)`. For all of - the curve configurations, the SWU map-to-curve method may be called as `c.swu_map(z,n)`, where - `z: Field` depends on `Field` and `c` and must be chosen by the user (the conditions it needs to - satisfy are specified in the comments - [here](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr)). - -## Examples - -The -[ec_baby_jubjub test](https://github.com/noir-lang/noir/blob/master/test_programs/compile_success_empty/ec_baby_jubjub/src/main.nr) -illustrates all of the above primitives on various forms of the Baby Jubjub curve. A couple of more -interesting examples in Noir would be: - -Public-key cryptography: Given an elliptic curve and a 'base point' on it, determine the public key -from the private key. This is a matter of using scalar multiplication. In the case of Baby Jubjub, -for example, this code would do: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; - -fn bjj_pub_key(priv_key: Field) -> Point -{ - - let bjj = Curve::new(168700, 168696, G::new(995203441582195749578291179787384436505546430278305826713579947235728471134,5472060717959818805561601436314318772137091100104008585924551046643952123905)); - - let base_pt = Point::new(5299619240641551281634865583518297030282874472190772894086521144482721001553, 16950150798460657717958625567821834550301663161624707787222815936182638968203); - - bjj.mul(priv_key,base_pt) -} -``` - -This would come in handy in a Merkle proof. - -- EdDSA signature verification: This is a matter of combining these primitives with a suitable hash - function. See - [feat(stdlib): EdDSA sig verification noir#1136](https://github.com/noir-lang/noir/pull/1136) for - the case of Baby Jubjub and the Poseidon hash function. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx deleted file mode 100644 index 4c22e70e8de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: ECDSA Signature Verification -description: Learn about the cryptographic primitives regarding ECDSA over the secp256k1 and secp256r1 curves -keywords: [cryptographic primitives, Noir project, ecdsa, secp256k1, secp256r1, signatures] -sidebar_position: 3 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports ECDSA signatures verification over the secp256k1 and secp256r1 curves. - -## ecdsa_secp256k1::verify_signature - -Verifier for ECDSA Secp256k1 signatures. -See ecdsa_secp256k1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256k1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256k1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256k1::verify_signature_slice - -Verifier for ECDSA Secp256k1 signatures where the message is a slice. - -```rust title="ecdsa_secp256k1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L13-L20 - - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures. -See ecdsa_secp256r1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256r1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256r1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures where the message is a slice. - -```rust title="ecdsa_secp256r1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/eddsa.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/eddsa.mdx deleted file mode 100644 index b283de693c8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/eddsa.mdx +++ /dev/null @@ -1,37 +0,0 @@ ---- -title: EdDSA Verification -description: Learn about the cryptographic primitives regarding EdDSA -keywords: [cryptographic primitives, Noir project, eddsa, signatures] -sidebar_position: 5 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## eddsa::eddsa_poseidon_verify - -Verifier for EdDSA signatures - -```rust -fn eddsa_poseidon_verify(public_key_x : Field, public_key_y : Field, signature_s: Field, signature_r8_x: Field, signature_r8_y: Field, message: Field) -> bool -``` - -It is also possible to specify the hash algorithm used for the signature by using the `eddsa_verify` function by passing a type implementing the Hasher trait with the turbofish operator. -For instance, if you want to use Poseidon2 instead, you can do the following: -```rust -use std::hash::poseidon2::Poseidon2Hasher; - -eddsa_verify::(pub_key_a.x, pub_key_a.y, s_a, r8_a.x, r8_a.y, msg); -``` - - - -## eddsa::eddsa_to_pub - -Private to public key conversion. - -Returns `(pub_key_x, pub_key_y)` - -```rust -fn eddsa_to_pub(secret : Field) -> (Field, Field) -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx deleted file mode 100644 index 69e0265c81a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx +++ /dev/null @@ -1,95 +0,0 @@ ---- -title: Scalar multiplication -description: See how you can perform scalar multiplication in Noir -keywords: [cryptographic primitives, Noir project, scalar multiplication] -sidebar_position: 1 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -The following functions perform operations over the embedded curve whose coordinates are defined by the configured noir field. -For the BN254 scalar field, this is BabyJubJub or Grumpkin. - -:::note -Suffixes `_low` and `_high` denote low and high limbs of a scalar. -::: - -## embedded_curve_ops::multi_scalar_mul - -Performs multi scalar multiplication over the embedded curve. -The function accepts arbitrary amount of point-scalar pairs on the input, it multiplies the individual pairs over -the curve and returns a sum of the resulting points. - -Points represented as x and y coordinates [x1, y1, x2, y2, ...], scalars as low and high limbs [low1, high1, low2, high2, ...]. - -```rust title="multi_scalar_mul" showLineNumbers -pub fn multi_scalar_mul( - points: [EmbeddedCurvePoint; N], - scalars: [EmbeddedCurveScalar; N] -) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L91-L96 - - -example - -```rust -fn main(point_x: Field, point_y: Field, scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::multi_scalar_mul([point_x, point_y], [scalar_low, scalar_high]); - println(point); -} -``` - -## embedded_curve_ops::fixed_base_scalar_mul - -Performs fixed base scalar multiplication over the embedded curve (multiplies input scalar with a generator point). -The function accepts a single scalar on the input represented as 2 fields. - -```rust title="fixed_base_scalar_mul" showLineNumbers -pub fn fixed_base_scalar_mul(scalar: EmbeddedCurveScalar) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L108-L110 - - -example - -```rust -fn main(scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::fixed_base_scalar_mul(scalar_low, scalar_high); - println(point); -} -``` - -## embedded_curve_ops::embedded_curve_add - -Adds two points on the embedded curve. -This function takes two `EmbeddedCurvePoint` structures as parameters, representing points on the curve, and returns a new `EmbeddedCurvePoint` structure that represents their sum. - -### Parameters: -- `point1` (`EmbeddedCurvePoint`): The first point to add. -- `point2` (`EmbeddedCurvePoint`): The second point to add. - -### Returns: -- `EmbeddedCurvePoint`: The resulting point after the addition of `point1` and `point2`. - -```rust title="embedded_curve_add" showLineNumbers -fn embedded_curve_add( - point1: EmbeddedCurvePoint, - point2: EmbeddedCurvePoint -) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L118-L123 - - -example - -```rust -fn main() { - let point1 = EmbeddedCurvePoint { x: 1, y: 2 }; - let point2 = EmbeddedCurvePoint { x: 3, y: 4 }; - let result = std::embedded_curve_ops::embedded_curve_add(point1, point2); - println!("Resulting Point: ({}, {})", result.x, result.y); -} -``` - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/hashes.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/hashes.mdx deleted file mode 100644 index 797ff8cc22c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/hashes.mdx +++ /dev/null @@ -1,253 +0,0 @@ ---- -title: Hash methods -description: - Learn about the cryptographic primitives ready to use for any Noir project, including sha256, - blake2s, pedersen, mimc_bn254 and mimc -keywords: - [cryptographic primitives, Noir project, sha256, blake2s, pedersen, mimc_bn254, mimc, hash] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## sha256 - -Given an array of bytes, returns the resulting sha256 hash. -Specify a message_size to hash only the first `message_size` bytes of the input. - -```rust title="sha256" showLineNumbers -pub fn sha256(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L14-L16 - - -example: -```rust title="sha256_var" showLineNumbers -let digest = std::hash::sha256_var([x as u8], 1); -``` -> Source code: test_programs/execution_success/sha256/src/main.nr#L16-L18 - - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::sha256::sha256_var(x, 4); -} -``` - - - - -## blake2s - -Given an array of bytes, returns an array with the Blake2 hash - -```rust title="blake2s" showLineNumbers -pub fn blake2s(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L20-L22 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake2s(x); -} -``` - - - -## blake3 - -Given an array of bytes, returns an array with the Blake3 hash - -```rust title="blake3" showLineNumbers -pub fn blake3(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L26-L28 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake3(x); -} -``` - - - -## pedersen_hash - -Given an array of Fields, returns the Pedersen hash. - -```rust title="pedersen_hash" showLineNumbers -pub fn pedersen_hash(input: [Field; N]) -> Field -``` -> Source code: noir_stdlib/src/hash/mod.nr#L79-L81 - - -example: - -```rust title="pedersen-hash" showLineNumbers -fn main(x: Field, y: Field, expected_hash: Field) { - let hash = std::hash::pedersen_hash([x, y]); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/pedersen_hash/src/main.nr#L1-L7 - - - - -## pedersen_commitment - -Given an array of Fields, returns the Pedersen commitment. - -```rust title="pedersen_commitment" showLineNumbers -pub fn pedersen_commitment(input: [Field; N]) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/hash/mod.nr#L31-L33 - - -example: - -```rust title="pedersen-commitment" showLineNumbers -fn main(x: Field, y: Field, expected_commitment: std::embedded_curve_ops::EmbeddedCurvePoint) { - let commitment = std::hash::pedersen_commitment([x, y]); - assert_eq(commitment.x, expected_commitment.x); - assert_eq(commitment.y, expected_commitment.y); -} -``` -> Source code: test_programs/execution_success/pedersen_commitment/src/main.nr#L1-L8 - - - - -## keccak256 - -Given an array of bytes (`u8`), returns the resulting keccak hash as an array of -32 bytes (`[u8; 32]`). Specify a message_size to hash only the first -`message_size` bytes of the input. - -```rust title="keccak256" showLineNumbers -pub fn keccak256(input: [u8; N], message_size: u32) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L128-L130 - - -example: - -```rust title="keccak256" showLineNumbers -fn main(x: Field, result: [u8; 32]) { - // We use the `as` keyword here to denote the fact that we want to take just the first byte from the x Field - // The padding is taken care of by the program - let digest = std::hash::keccak256([x as u8], 1); - assert(digest == result); - - //#1399: variable message size - let message_size = 4; - let hash_a = std::hash::keccak256([1, 2, 3, 4], message_size); - let hash_b = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size); - - assert(hash_a == hash_b); - - let message_size_big = 8; - let hash_c = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size_big); - - assert(hash_a != hash_c); -} -``` -> Source code: test_programs/execution_success/keccak256/src/main.nr#L1-L21 - - - - -## poseidon - -Given an array of Fields, returns a new Field with the Poseidon Hash. Mind that you need to specify -how many inputs are there to your Poseidon function. - -```rust -// example for hash_1, hash_2 accepts an array of length 2, etc -fn hash_1(input: [Field; 1]) -> Field -``` - -example: - -```rust title="poseidon" showLineNumbers -use std::hash::poseidon; - -fn main(x1: [Field; 2], y1: pub Field, x2: [Field; 4], y2: pub Field) { - let hash1 = poseidon::bn254::hash_2(x1); - assert(hash1 == y1); - - let hash2 = poseidon::bn254::hash_4(x2); - assert(hash2 == y2); -} -``` -> Source code: test_programs/execution_success/poseidon_bn254_hash/src/main.nr#L1-L11 - - -## poseidon 2 - -Given an array of Fields, returns a new Field with the Poseidon2 Hash. Contrary to the Poseidon -function, there is only one hash and you can specify a message_size to hash only the first -`message_size` bytes of the input, - -```rust -// example for hashing the first three elements of the input -Poseidon2::hash(input, 3); -``` - -example: - -```rust title="poseidon2" showLineNumbers -use std::hash::poseidon2; - -fn main(inputs: [Field; 4], expected_hash: Field) { - let hash = poseidon2::Poseidon2::hash(inputs, inputs.len()); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/poseidon2/src/main.nr#L1-L8 - - -## mimc_bn254 and mimc - -`mimc_bn254` is `mimc`, but with hardcoded parameters for the BN254 curve. You can use it by -providing an array of Fields, and it returns a Field with the hash. You can use the `mimc` method if -you're willing to input your own constants: - -```rust -fn mimc(x: Field, k: Field, constants: [Field; N], exp : Field) -> Field -``` - -otherwise, use the `mimc_bn254` method: - -```rust -fn mimc_bn254(array: [Field; N]) -> Field -``` - -example: - -```rust - -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::mimc::mimc_bn254(x); -} -``` - -## hash_to_field - -```rust -fn hash_to_field(_input : [Field]) -> Field {} -``` - -Calculates the `blake2s` hash of the inputs and returns the hash modulo the field modulus to return -a value which can be represented as a `Field`. - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/index.md deleted file mode 100644 index 650f30165d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/index.md +++ /dev/null @@ -1,14 +0,0 @@ ---- -title: Cryptographic Primitives -description: - Learn about the cryptographic primitives ready to use for any Noir project -keywords: - [ - cryptographic primitives, - Noir project, - ] ---- - -The Noir team is progressively adding new cryptographic primitives to the standard library. Reach out for news or if you would be interested in adding more of these calculations in Noir. - -Some methods are available thanks to the Aztec backend, not being performed using Noir. When using other backends, these methods may or may not be supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/schnorr.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/schnorr.mdx deleted file mode 100644 index 00e7f257612..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/cryptographic_primitives/schnorr.mdx +++ /dev/null @@ -1,64 +0,0 @@ ---- -title: Schnorr Signatures -description: Learn how you can verify Schnorr signatures using Noir -keywords: [cryptographic primitives, Noir project, schnorr, signatures] -sidebar_position: 2 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## schnorr::verify_signature - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin). -See schnorr::verify_signature_slice for a version that works directly on slices. - -```rust title="schnorr_verify" showLineNumbers -pub fn verify_signature( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L2-L9 - - -where `_signature` can be generated like so using the npm package -[@noir-lang/barretenberg](https://www.npmjs.com/package/@noir-lang/barretenberg) - -```js -const { BarretenbergWasm } = require('@noir-lang/barretenberg/dest/wasm'); -const { Schnorr } = require('@noir-lang/barretenberg/dest/crypto/schnorr'); - -... - -const barretenberg = await BarretenbergWasm.new(); -const schnorr = new Schnorr(barretenberg); -const pubKey = schnorr.computePublicKey(privateKey); -const message = ... -const signature = Array.from( - schnorr.constructSignature(hash, privateKey).toBuffer() -); - -... -``` - - - -## schnorr::verify_signature_slice - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin) -where the message is a slice. - -```rust title="schnorr_verify_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/is_unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/is_unconstrained.md deleted file mode 100644 index 51bb1bda8f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/is_unconstrained.md +++ /dev/null @@ -1,69 +0,0 @@ ---- -title: Is Unconstrained Function -description: - The is_unconstrained function returns wether the context at that point of the program is unconstrained or not. -keywords: - [ - unconstrained - ] ---- - -It's very common for functions in circuits to take unconstrained hints of an expensive computation and then verify it. This is done by running the hint in an unconstrained context and then verifying the result in a constrained context. - -When a function is marked as unconstrained, any subsequent functions that it calls will also be run in an unconstrained context. However, if we are implementing a library function, other users might call it within an unconstrained context or a constrained one. Generally, in an unconstrained context we prefer just computing the result instead of taking a hint of it and verifying it, since that'd mean doing the same computation twice: - -```rust - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - my_expensive_computation_hint(); - // verify my_expensive_computation: If external_interface is called from unconstrained, this is redundant - ... -} - -``` - -In order to improve the performance in an unconstrained context you can use the function at `std::runtime::is_unconstrained() -> bool`: - - -```rust -use dep::std::runtime::is_unconstrained; - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - if is_unconstrained() { - my_expensive_computation(); - } else { - my_expensive_computation_hint(); - // verify my_expensive_computation - ... - } -} - -``` - -The is_unconstrained result is resolved at compile time, so in unconstrained contexts the compiler removes the else branch, and in constrained contexts the compiler removes the if branch, reducing the amount of compute necessary to run external_interface. - -Note that using `is_unconstrained` in a `comptime` context will also return `true`: - -``` -fn main() { - comptime { - assert(is_unconstrained()); - } -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/logging.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/logging.md deleted file mode 100644 index db75ef9f86f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/logging.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Logging -description: - Learn how to use the println statement for debugging in Noir with this tutorial. Understand the - basics of logging in Noir and how to implement it in your code. -keywords: - [ - noir logging, - println statement, - print statement, - debugging in noir, - noir std library, - logging tutorial, - basic logging in noir, - noir logging implementation, - noir debugging techniques, - rust, - ] ---- - -The standard library provides two familiar statements you can use: `println` and `print`. Despite being a limited implementation of rust's `println!` and `print!` macros, these constructs can be useful for debugging. - -You can print the output of both statements in your Noir code by using the `nargo execute` command or the `--show-output` flag when using `nargo test` (provided there are print statements in your tests). - -It is recommended to use `nargo execute` if you want to debug failing constraints with `println` or `print` statements. This is due to every input in a test being a constant rather than a witness, so we issue an error during compilation while we only print during execution (which comes after compilation). Neither `println`, nor `print` are callable for failed constraints caught at compile time. - -Both `print` and `println` are generic functions which can work on integers, fields, strings, and even structs or expressions. Note however, that slices are currently unsupported. For example: - -```rust -struct Person { - age: Field, - height: Field, -} - -fn main(age: Field, height: Field) { - let person = Person { - age: age, - height: height, - }; - println(person); - println(age + height); - println("Hello world!"); -} -``` - -You can print different types in the same statement (including strings) with a type called `fmtstr`. It can be specified in the same way as a normal string, just prepended with an "f" character: - -```rust - let fmt_str = f"i: {i}, j: {j}"; - println(fmt_str); - - let s = myStruct { y: x, x: y }; - println(s); - - println(f"i: {i}, s: {s}"); - - println(x); - println([x, y]); - - let foo = fooStruct { my_struct: s, foo: 15 }; - println(f"s: {s}, foo: {foo}"); - - println(15); // prints 0x0f, implicit Field - println(-1 as u8); // prints 255 - println(-1 as i8); // prints -1 -``` - -Examples shown above are interchangeable between the two `print` statements: - -```rust -let person = Person { age : age, height : height }; - -println(person); -print(person); - -println("Hello world!"); // Prints with a newline at the end of the input -print("Hello world!"); // Prints the input and keeps cursor on the same line -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/merkle_trees.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/merkle_trees.md deleted file mode 100644 index 6a9ebf72ada..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/merkle_trees.md +++ /dev/null @@ -1,58 +0,0 @@ ---- -title: Merkle Trees -description: Learn about Merkle Trees in Noir with this tutorial. Explore the basics of computing a merkle root using a proof, with examples. -keywords: - [ - Merkle trees in Noir, - Noir programming language, - check membership, - computing root from leaf, - Noir Merkle tree implementation, - Merkle tree tutorial, - Merkle tree code examples, - Noir libraries, - pedersen hash., - ] ---- - -## compute_merkle_root - -Returns the root of the tree from the provided leaf and its hash path, using a [Pedersen hash](./cryptographic_primitives/hashes.mdx#pedersen_hash). - -```rust -fn compute_merkle_root(leaf : Field, index : Field, hash_path: [Field]) -> Field -``` - -example: - -```rust -/** - // these values are for this example only - index = "0" - priv_key = "0x000000000000000000000000000000000000000000000000000000616c696365" - secret = "0x1929ea3ab8d9106a899386883d9428f8256cfedb3c4f6b66bf4aa4d28a79988f" - note_hash_path = [ - "0x1e61bdae0f027b1b2159e1f9d3f8d00fa668a952dddd822fda80dc745d6f65cc", - "0x0e4223f3925f98934393c74975142bd73079ab0621f4ee133cee050a3c194f1a", - "0x2fd7bb412155bf8693a3bd2a3e7581a679c95c68a052f835dddca85fa1569a40" - ] - */ -fn main(index: Field, priv_key: Field, secret: Field, note_hash_path: [Field; 3]) { - - let pubkey = std::scalar_mul::fixed_base_embedded_curve(priv_key); - let pubkey_x = pubkey[0]; - let pubkey_y = pubkey[1]; - let note_commitment = std::hash::pedersen(&[pubkey_x, pubkey_y, secret]); - - let root = std::merkle::compute_merkle_root(note_commitment[0], index, note_hash_path.as_slice()); - println(root); -} -``` - -To check merkle tree membership: - -1. Include a merkle root as a program input. -2. Compute the merkle root of a given leaf, index and hash path. -3. Assert the merkle roots are equal. - -For more info about merkle trees, see the Wikipedia [page](https://en.wikipedia.org/wiki/Merkle_tree). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/options.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/options.md deleted file mode 100644 index a1bd4e1de5f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/options.md +++ /dev/null @@ -1,101 +0,0 @@ ---- -title: Option Type ---- - -The `Option` type is a way to express that a value might be present (`Some(T))` or absent (`None`). It's a safer way to handle potential absence of values, compared to using nulls in many other languages. - -```rust -struct Option { - None, - Some(T), -} -``` - -The `Option` type, already imported into your Noir program, can be used directly: - -```rust -fn main() { - let none = Option::none(); - let some = Option::some(3); -} -``` - -See [this test](https://github.com/noir-lang/noir/blob/5cbfb9c4a06c8865c98ff2b594464b037d821a5c/crates/nargo_cli/tests/test_data/option/src/main.nr) for a more comprehensive set of examples of each of the methods described below. - -## Methods - -### none - -Constructs a none value. - -### some - -Constructs a some wrapper around a given value. - -### is_none - -Returns true if the Option is None. - -### is_some - -Returns true of the Option is Some. - -### unwrap - -Asserts `self.is_some()` and returns the wrapped value. - -### unwrap_unchecked - -Returns the inner value without asserting `self.is_some()`. This method can be useful within an if condition when we already know that `option.is_some()`. If the option is None, there is no guarantee what value will be returned, only that it will be of type T for an `Option`. - -### unwrap_or - -Returns the wrapped value if `self.is_some()`. Otherwise, returns the given default value. - -### unwrap_or_else - -Returns the wrapped value if `self.is_some()`. Otherwise, calls the given function to return a default value. - -### expect - -Asserts `self.is_some()` with a provided custom message and returns the contained `Some` value. The custom message is expected to be a format string. - -### map - -If self is `Some(x)`, this returns `Some(f(x))`. Otherwise, this returns `None`. - -### map_or - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns the given default value. - -### map_or_else - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns `default()`. - -### and - -Returns None if self is None. Otherwise, this returns `other`. - -### and_then - -If self is None, this returns None. Otherwise, this calls the given function with the Some value contained within self, and returns the result of that call. In some languages this function is called `flat_map` or `bind`. - -### or - -If self is Some, return self. Otherwise, return `other`. - -### or_else - -If self is Some, return self. Otherwise, return `default()`. - -### xor - -If only one of the two Options is Some, return that option. Otherwise, if both options are Some or both are None, None is returned. - -### filter - -Returns `Some(x)` if self is `Some(x)` and `predicate(x)` is true. Otherwise, this returns `None`. - -### flatten - -Flattens an `Option>` into a `Option`. This returns `None` if the outer Option is None. Otherwise, this returns the inner Option. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/recursion.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/recursion.mdx deleted file mode 100644 index 60414a2fa51..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/recursion.mdx +++ /dev/null @@ -1,85 +0,0 @@ ---- -title: Recursive Proofs -description: Learn about how to write recursive proofs in Noir. -keywords: [recursion, recursive proofs, verification_key, verify_proof] ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports recursively verifying proofs, meaning you verify the proof of a Noir program in another Noir program. This enables creating proofs of arbitrary size by doing step-wise verification of smaller components of a large proof. - -Read [the explainer on recursion](../../explainers/explainer-recursion.md) to know more about this function and the [guide on how to use it.](../../how_to/how-to-recursion.md) - -## The `#[recursive]` Attribute - -In Noir, the `#[recursive]` attribute is used to indicate that a circuit is designed for recursive proof generation. When applied, it informs the compiler and the tooling that the circuit should be compiled in a way that makes its proofs suitable for recursive verification. This attribute eliminates the need for manual flagging of recursion at the tooling level, streamlining the proof generation process for recursive circuits. - -### Example usage with `#[recursive]` - -```rust -#[recursive] -fn main(x: Field, y: pub Field) { - assert(x == y, "x and y are not equal"); -} - -// This marks the circuit as recursion-friendly and indicates that proofs generated from this circuit -// are intended for recursive verification. -``` - -By incorporating this attribute directly in the circuit's definition, tooling like Nargo and NoirJS can automatically execute recursive-specific duties for Noir programs (e.g. recursive-friendly proof artifact generation) without additional flags or configurations. - -## Verifying Recursive Proofs - -```rust -#[foreign(recursive_aggregation)] -pub fn verify_proof(verification_key: [Field], proof: [Field], public_inputs: [Field], key_hash: Field) {} -``` - - - -## Example usage - -```rust - -fn main( - verification_key : [Field; 114], - proof : [Field; 93], - public_inputs : [Field; 1], - key_hash : Field, - proof_b : [Field; 93], -) { - std::verify_proof( - verification_key, - proof, - public_inputs, - key_hash - ); - - std::verify_proof( - verification_key, - proof_b, - public_inputs, - key_hash - ); -} -``` - -You can see a full example of recursive proofs in [this example recursion demo repo](https://github.com/noir-lang/noir-examples/tree/master/recursion). - -## Parameters - -### `verification_key` - -The verification key for the zk program that is being verified. - -### `proof` - -The proof for the zk program that is being verified. - -### `public_inputs` - -These represent the public inputs of the proof we are verifying. - -### `key_hash` - -A key hash is used to check the validity of the verification key. The circuit implementing this opcode can use this hash to ensure that the key provided to the circuit matches the key produced by the circuit creator. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/traits.md deleted file mode 100644 index 850cc129e73..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/traits.md +++ /dev/null @@ -1,501 +0,0 @@ ---- -title: Traits -description: Noir's stdlib provides a few commonly used traits. -keywords: [traits, trait, interface, protocol, default, add, eq] ---- - -## `std::default` - -### `std::default::Default` - -```rust title="default-trait" showLineNumbers -trait Default { - fn default() -> Self; -} -``` -> Source code: noir_stdlib/src/default.nr#L4-L8 - - -Constructs a default value of a type. - -Implementations: -```rust -impl Default for Field { .. } - -impl Default for i8 { .. } -impl Default for i16 { .. } -impl Default for i32 { .. } -impl Default for i64 { .. } - -impl Default for u8 { .. } -impl Default for u16 { .. } -impl Default for u32 { .. } -impl Default for u64 { .. } - -impl Default for () { .. } -impl Default for bool { .. } - -impl Default for [T; N] - where T: Default { .. } - -impl Default for [T] { .. } - -impl Default for (A, B) - where A: Default, B: Default { .. } - -impl Default for (A, B, C) - where A: Default, B: Default, C: Default { .. } - -impl Default for (A, B, C, D) - where A: Default, B: Default, C: Default, D: Default { .. } - -impl Default for (A, B, C, D, E) - where A: Default, B: Default, C: Default, D: Default, E: Default { .. } -``` - -For primitive integer types, the return value of `default` is `0`. Container -types such as arrays are filled with default values of their element type, -except slices whose length is unknown and thus defaulted to zero. - ---- - -## `std::convert` - -### `std::convert::From` - -```rust title="from-trait" showLineNumbers -trait From { - fn from(input: T) -> Self; -} -``` -> Source code: noir_stdlib/src/convert.nr#L1-L5 - - -The `From` trait defines how to convert from a given type `T` to the type on which the trait is implemented. - -The Noir standard library provides a number of implementations of `From` between primitive types. -```rust title="from-impls" showLineNumbers -// Unsigned integers - -impl From for u32 { fn from(value: u8) -> u32 { value as u32 } } - -impl From for u64 { fn from(value: u8) -> u64 { value as u64 } } -impl From for u64 { fn from(value: u32) -> u64 { value as u64 } } - -impl From for Field { fn from(value: u8) -> Field { value as Field } } -impl From for Field { fn from(value: u32) -> Field { value as Field } } -impl From for Field { fn from(value: u64) -> Field { value as Field } } - -// Signed integers - -impl From for i32 { fn from(value: i8) -> i32 { value as i32 } } - -impl From for i64 { fn from(value: i8) -> i64 { value as i64 } } -impl From for i64 { fn from(value: i32) -> i64 { value as i64 } } - -// Booleans -impl From for u8 { fn from(value: bool) -> u8 { value as u8 } } -impl From for u32 { fn from(value: bool) -> u32 { value as u32 } } -impl From for u64 { fn from(value: bool) -> u64 { value as u64 } } -impl From for i8 { fn from(value: bool) -> i8 { value as i8 } } -impl From for i32 { fn from(value: bool) -> i32 { value as i32 } } -impl From for i64 { fn from(value: bool) -> i64 { value as i64 } } -impl From for Field { fn from(value: bool) -> Field { value as Field } } -``` -> Source code: noir_stdlib/src/convert.nr#L25-L52 - - -#### When to implement `From` - -As a general rule of thumb, `From` may be implemented in the [situations where it would be suitable in Rust](https://doc.rust-lang.org/std/convert/trait.From.html#when-to-implement-from): - -- The conversion is *infallible*: Noir does not provide an equivalent to Rust's `TryFrom`, if the conversion can fail then provide a named method instead. -- The conversion is *lossless*: semantically, it should not lose or discard information. For example, `u32: From` can losslessly convert any `u16` into a valid `u32` such that the original `u16` can be recovered. On the other hand, `u16: From` should not be implemented as `2**16` is a `u32` which cannot be losslessly converted into a `u16`. -- The conversion is *value-preserving*: the conceptual kind and meaning of the resulting value is the same, even though the Noir type and technical representation might be different. While it's possible to infallibly and losslessly convert a `u8` into a `str<2>` hex representation, `4u8` and `"04"` are too different for `str<2>: From` to be implemented. -- The conversion is *obvious*: it's the only reasonable conversion between the two types. If there's ambiguity on how to convert between them such that the same input could potentially map to two different values then a named method should be used. For instance rather than implementing `U128: From<[u8; 16]>`, the methods `U128::from_le_bytes` and `U128::from_be_bytes` are used as otherwise the endianness of the array would be ambiguous, resulting in two potential values of `U128` from the same byte array. - -One additional recommendation specific to Noir is: -- The conversion is *efficient*: it's relatively cheap to convert between the two types. Due to being a ZK DSL, it's more important to avoid unnecessary computation compared to Rust. If the implementation of `From` would encourage users to perform unnecessary conversion, resulting in additional proving time, then it may be preferable to expose functionality such that this conversion may be avoided. - -### `std::convert::Into` - -The `Into` trait is defined as the reciprocal of `From`. It should be easy to convince yourself that if we can convert to type `A` from type `B`, then it's possible to convert type `B` into type `A`. - -For this reason, implementing `From` on a type will automatically generate a matching `Into` implementation. One should always prefer implementing `From` over `Into` as implementing `Into` will not generate a matching `From` implementation. - -```rust title="into-trait" showLineNumbers -trait Into { - fn into(self) -> T; -} - -impl Into for U where T: From { - fn into(self) -> T { - T::from(self) - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L13-L23 - - -`Into` is most useful when passing function arguments where the types don't quite match up with what the function expects. In this case, the compiler has enough type information to perform the necessary conversion by just appending `.into()` onto the arguments in question. - ---- - -## `std::cmp` - -### `std::cmp::Eq` - -```rust title="eq-trait" showLineNumbers -trait Eq { - fn eq(self, other: Self) -> bool; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L4-L8 - - -Returns `true` if `self` is equal to `other`. Implementing this trait on a type -allows the type to be used with `==` and `!=`. - -Implementations: -```rust -impl Eq for Field { .. } - -impl Eq for i8 { .. } -impl Eq for i16 { .. } -impl Eq for i32 { .. } -impl Eq for i64 { .. } - -impl Eq for u8 { .. } -impl Eq for u16 { .. } -impl Eq for u32 { .. } -impl Eq for u64 { .. } - -impl Eq for () { .. } -impl Eq for bool { .. } - -impl Eq for [T; N] - where T: Eq { .. } - -impl Eq for [T] - where T: Eq { .. } - -impl Eq for (A, B) - where A: Eq, B: Eq { .. } - -impl Eq for (A, B, C) - where A: Eq, B: Eq, C: Eq { .. } - -impl Eq for (A, B, C, D) - where A: Eq, B: Eq, C: Eq, D: Eq { .. } - -impl Eq for (A, B, C, D, E) - where A: Eq, B: Eq, C: Eq, D: Eq, E: Eq { .. } -``` - -### `std::cmp::Ord` - -```rust title="ord-trait" showLineNumbers -trait Ord { - fn cmp(self, other: Self) -> Ordering; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L113-L117 - - -`a.cmp(b)` compares two values returning `Ordering::less()` if `a < b`, -`Ordering::equal()` if `a == b`, or `Ordering::greater()` if `a > b`. -Implementing this trait on a type allows `<`, `<=`, `>`, and `>=` to be -used on values of the type. - -`std::cmp` also provides `max` and `min` functions for any type which implements the `Ord` trait. - -Implementations: - -```rust -impl Ord for u8 { .. } -impl Ord for u16 { .. } -impl Ord for u32 { .. } -impl Ord for u64 { .. } - -impl Ord for i8 { .. } -impl Ord for i16 { .. } -impl Ord for i32 { .. } - -impl Ord for i64 { .. } - -impl Ord for () { .. } -impl Ord for bool { .. } - -impl Ord for [T; N] - where T: Ord { .. } - -impl Ord for [T] - where T: Ord { .. } - -impl Ord for (A, B) - where A: Ord, B: Ord { .. } - -impl Ord for (A, B, C) - where A: Ord, B: Ord, C: Ord { .. } - -impl Ord for (A, B, C, D) - where A: Ord, B: Ord, C: Ord, D: Ord { .. } - -impl Ord for (A, B, C, D, E) - where A: Ord, B: Ord, C: Ord, D: Ord, E: Ord { .. } -``` - ---- - -## `std::ops` - -### `std::ops::Add`, `std::ops::Sub`, `std::ops::Mul`, and `std::ops::Div` - -These traits abstract over addition, subtraction, multiplication, and division respectively. -Implementing these traits for a given type will also allow that type to be used with the corresponding operator -for that trait (`+` for Add, etc) in addition to the normal method names. - -```rust title="add-trait" showLineNumbers -trait Add { - fn add(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L1-L5 - -```rust title="sub-trait" showLineNumbers -trait Sub { - fn sub(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L19-L23 - -```rust title="mul-trait" showLineNumbers -trait Mul { - fn mul(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L37-L41 - -```rust title="div-trait" showLineNumbers -trait Div { - fn div(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L55-L59 - - -The implementations block below is given for the `Add` trait, but the same types that implement -`Add` also implement `Sub`, `Mul`, and `Div`. - -Implementations: -```rust -impl Add for Field { .. } - -impl Add for i8 { .. } -impl Add for i16 { .. } -impl Add for i32 { .. } -impl Add for i64 { .. } - -impl Add for u8 { .. } -impl Add for u16 { .. } -impl Add for u32 { .. } -impl Add for u64 { .. } -``` - -### `std::ops::Rem` - -```rust title="rem-trait" showLineNumbers -trait Rem{ - fn rem(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L73-L77 - - -`Rem::rem(a, b)` is the remainder function returning the result of what is -left after dividing `a` and `b`. Implementing `Rem` allows the `%` operator -to be used with the implementation type. - -Unlike other numeric traits, `Rem` is not implemented for `Field`. - -Implementations: -```rust -impl Rem for u8 { fn rem(self, other: u8) -> u8 { self % other } } -impl Rem for u16 { fn rem(self, other: u16) -> u16 { self % other } } -impl Rem for u32 { fn rem(self, other: u32) -> u32 { self % other } } -impl Rem for u64 { fn rem(self, other: u64) -> u64 { self % other } } - -impl Rem for i8 { fn rem(self, other: i8) -> i8 { self % other } } -impl Rem for i16 { fn rem(self, other: i16) -> i16 { self % other } } -impl Rem for i32 { fn rem(self, other: i32) -> i32 { self % other } } -impl Rem for i64 { fn rem(self, other: i64) -> i64 { self % other } } -``` - -### `std::ops::Neg` - -```rust title="neg-trait" showLineNumbers -trait Neg { - fn neg(self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L89-L93 - - -`Neg::neg` is equivalent to the unary negation operator `-`. - -Implementations: -```rust title="neg-trait-impls" showLineNumbers -impl Neg for Field { fn neg(self) -> Field { -self } } - -impl Neg for i8 { fn neg(self) -> i8 { -self } } -impl Neg for i16 { fn neg(self) -> i16 { -self } } -impl Neg for i32 { fn neg(self) -> i32 { -self } } -impl Neg for i64 { fn neg(self) -> i64 { -self } } -``` -> Source code: noir_stdlib/src/ops/arith.nr#L95-L102 - - -### `std::ops::Not` - -```rust title="not-trait" showLineNumbers -trait Not { - fn not(self: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L1-L5 - - -`Not::not` is equivalent to the unary bitwise NOT operator `!`. - -Implementations: -```rust title="not-trait-impls" showLineNumbers -impl Not for bool { fn not(self) -> bool { !self } } - -impl Not for u64 { fn not(self) -> u64 { !self } } -impl Not for u32 { fn not(self) -> u32 { !self } } -impl Not for u16 { fn not(self) -> u16 { !self } } -impl Not for u8 { fn not(self) -> u8 { !self } } -impl Not for u1 { fn not(self) -> u1 { !self } } - -impl Not for i8 { fn not(self) -> i8 { !self } } -impl Not for i16 { fn not(self) -> i16 { !self } } -impl Not for i32 { fn not(self) -> i32 { !self } } -impl Not for i64 { fn not(self) -> i64 { !self } } -``` -> Source code: noir_stdlib/src/ops/bit.nr#L7-L20 - - -### `std::ops::{ BitOr, BitAnd, BitXor }` - -```rust title="bitor-trait" showLineNumbers -trait BitOr { - fn bitor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L22-L26 - -```rust title="bitand-trait" showLineNumbers -trait BitAnd { - fn bitand(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L40-L44 - -```rust title="bitxor-trait" showLineNumbers -trait BitXor { - fn bitxor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L58-L62 - - -Traits for the bitwise operations `|`, `&`, and `^`. - -Implementing `BitOr`, `BitAnd` or `BitXor` for a type allows the `|`, `&`, or `^` operator respectively -to be used with the type. - -The implementations block below is given for the `BitOr` trait, but the same types that implement -`BitOr` also implement `BitAnd` and `BitXor`. - -Implementations: -```rust -impl BitOr for bool { fn bitor(self, other: bool) -> bool { self | other } } - -impl BitOr for u8 { fn bitor(self, other: u8) -> u8 { self | other } } -impl BitOr for u16 { fn bitor(self, other: u16) -> u16 { self | other } } -impl BitOr for u32 { fn bitor(self, other: u32) -> u32 { self | other } } -impl BitOr for u64 { fn bitor(self, other: u64) -> u64 { self | other } } - -impl BitOr for i8 { fn bitor(self, other: i8) -> i8 { self | other } } -impl BitOr for i16 { fn bitor(self, other: i16) -> i16 { self | other } } -impl BitOr for i32 { fn bitor(self, other: i32) -> i32 { self | other } } -impl BitOr for i64 { fn bitor(self, other: i64) -> i64 { self | other } } -``` - -### `std::ops::{ Shl, Shr }` - -```rust title="shl-trait" showLineNumbers -trait Shl { - fn shl(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L76-L80 - -```rust title="shr-trait" showLineNumbers -trait Shr { - fn shr(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L93-L97 - - -Traits for a bit shift left and bit shift right. - -Implementing `Shl` for a type allows the left shift operator (`<<`) to be used with the implementation type. -Similarly, implementing `Shr` allows the right shift operator (`>>`) to be used with the type. - -Note that bit shifting is not currently implemented for signed types. - -The implementations block below is given for the `Shl` trait, but the same types that implement -`Shl` also implement `Shr`. - -Implementations: -```rust -impl Shl for u8 { fn shl(self, other: u8) -> u8 { self << other } } -impl Shl for u16 { fn shl(self, other: u16) -> u16 { self << other } } -impl Shl for u32 { fn shl(self, other: u32) -> u32 { self << other } } -impl Shl for u64 { fn shl(self, other: u64) -> u64 { self << other } } -``` - ---- - -## `std::append` - -### `std::append::Append` - -`Append` can abstract over types that can be appended to - usually container types: - -```rust title="append-trait" showLineNumbers -trait Append { - fn empty() -> Self; - fn append(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/append.nr#L9-L14 - - -`Append` requires two methods: - -- `empty`: Constructs an empty value of `Self`. -- `append`: Append two values together, returning the result. - -Additionally, it is expected that for any implementation: - -- `T::empty().append(x) == x` -- `x.append(T::empty()) == x` - -Implementations: -```rust -impl Append for [T] -impl Append for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/zeroed.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/zeroed.md deleted file mode 100644 index f450fecdd36..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/standard_library/zeroed.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Zeroed Function -description: - The zeroed function returns a zeroed value of any type. -keywords: - [ - zeroed - ] ---- - -Implements `fn zeroed() -> T` to return a zeroed value of any type. This function is generally unsafe to use as the zeroed bit pattern is not guaranteed to be valid for all types. It can however, be useful in cases when the value is guaranteed not to be used such as in a BoundedVec library implementing a growable vector, up to a certain length, backed by an array. The array can be initialized with zeroed values which are guaranteed to be inaccessible until the vector is pushed to. Similarly, enumerations in noir can be implemented using this method by providing zeroed values for the unused variants. - -You can access the function at `std::unsafe::zeroed`. - -This function currently supports the following types: - -- Field -- Bool -- Uint -- Array -- Slice -- String -- Tuple -- Function - -Using it on other types could result in unexpected behavior. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md deleted file mode 100644 index 42f065f4a4e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md +++ /dev/null @@ -1,141 +0,0 @@ -# BarretenbergBackend - -## Implements - -- [`Backend`](../index.md#backend) -- [`Backend`](../index.md#backend) - -## Constructors - -### new BarretenbergBackend(acirCircuit, options) - -```ts -new BarretenbergBackend(acirCircuit, options): BarretenbergBackend -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `acirCircuit` | `CompiledCircuit` | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`BarretenbergBackend`](BarretenbergBackend.md) - -## Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `acirComposer` | `any` | - | -| `acirUncompressedBytecode` | `Uint8Array` | - | -| `api` | `Barretenberg` | - | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - | - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### generateProof() - -```ts -generateProof(compressedWitness): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `compressedWitness` | `Uint8Array` | - -#### Returns - -`Promise`\<`ProofData`\> - -#### Description - -Generates a proof - -*** - -### generateRecursiveProofArtifacts() - -```ts -generateRecursiveProofArtifacts(proofData, numOfPublicInputs): Promise -``` - -Generates artifacts that will be passed to a circuit that will verify this proof. - -Instead of passing the proof and verification key as a byte array, we pass them -as fields which makes it cheaper to verify in a circuit. - -The proof that is passed here will have been created using a circuit -that has the #[recursive] attribute on its `main` method. - -The number of public inputs denotes how many public inputs are in the inner proof. - -#### Parameters - -| Parameter | Type | Default value | -| :------ | :------ | :------ | -| `proofData` | `ProofData` | `undefined` | -| `numOfPublicInputs` | `number` | `0` | - -#### Returns - -`Promise`\<`object`\> - -#### Example - -```typescript -const artifacts = await backend.generateRecursiveProofArtifacts(proof, numOfPublicInputs); -``` - -*** - -### getVerificationKey() - -```ts -getVerificationKey(): Promise -``` - -#### Returns - -`Promise`\<`Uint8Array`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md deleted file mode 100644 index 500276ea748..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md +++ /dev/null @@ -1,58 +0,0 @@ -# BarretenbergVerifier - -## Constructors - -### new BarretenbergVerifier(options) - -```ts -new BarretenbergVerifier(options): BarretenbergVerifier -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`BarretenbergVerifier`](BarretenbergVerifier.md) - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData, verificationKey): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | -| `verificationKey` | `Uint8Array` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md deleted file mode 100644 index be1cd9ad465..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md +++ /dev/null @@ -1,116 +0,0 @@ -# UltraHonkBackend - -## Implements - -- [`Backend`](../index.md#backend) -- [`Backend`](../index.md#backend) - -## Constructors - -### new UltraHonkBackend(acirCircuit, options) - -```ts -new UltraHonkBackend(acirCircuit, options): UltraHonkBackend -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `acirCircuit` | `CompiledCircuit` | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`UltraHonkBackend`](UltraHonkBackend.md) - -## Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `acirUncompressedBytecode` | `Uint8Array` | - | -| `api` | `Barretenberg` | - | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - | - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### generateProof() - -```ts -generateProof(decompressedWitness): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `decompressedWitness` | `Uint8Array` | - -#### Returns - -`Promise`\<`ProofData`\> - -*** - -### generateRecursiveProofArtifacts() - -```ts -generateRecursiveProofArtifacts(_proofData, _numOfPublicInputs): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `_proofData` | `ProofData` | -| `_numOfPublicInputs` | `number` | - -#### Returns - -`Promise`\<`object`\> - -*** - -### getVerificationKey() - -```ts -getVerificationKey(): Promise -``` - -#### Returns - -`Promise`\<`Uint8Array`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | - -#### Returns - -`Promise`\<`boolean`\> - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md deleted file mode 100644 index aee9460153f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md +++ /dev/null @@ -1,58 +0,0 @@ -# UltraHonkVerifier - -## Constructors - -### new UltraHonkVerifier(options) - -```ts -new UltraHonkVerifier(options): UltraHonkVerifier -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`UltraHonkVerifier`](UltraHonkVerifier.md) - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData, verificationKey): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | -| `verificationKey` | `Uint8Array` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/index.md deleted file mode 100644 index 4699e16dee6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/index.md +++ /dev/null @@ -1,42 +0,0 @@ -# backend_barretenberg - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [BarretenbergBackend](classes/BarretenbergBackend.md) | - | -| [BarretenbergVerifier](classes/BarretenbergVerifier.md) | - | -| [UltraHonkBackend](classes/UltraHonkBackend.md) | - | -| [UltraHonkVerifier](classes/UltraHonkVerifier.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [BackendOptions](type-aliases/BackendOptions.md) | - | - -## References - -### CompiledCircuit - -Renames and re-exports [Backend](index.md#backend) - -*** - -### ProofData - -Renames and re-exports [Backend](index.md#backend) - -## Variables - -### Backend - -```ts -Backend: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md deleted file mode 100644 index b49a479f4f4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md +++ /dev/null @@ -1,21 +0,0 @@ -# BackendOptions - -```ts -type BackendOptions: object; -``` - -## Description - -An options object, currently only used to specify the number of threads to use. - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `memory` | `object` | - | -| `memory.maximum` | `number` | - | -| `threads` | `number` | **Description**

Number of threads | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs deleted file mode 100644 index 8ecf05c0163..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend","label":"BarretenbergBackend"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier","label":"BarretenbergVerifier"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend","label":"UltraHonkBackend"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier","label":"UltraHonkVerifier"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions","label":"BackendOptions"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/classes/Noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/classes/Noir.md deleted file mode 100644 index ead255bc504..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/classes/Noir.md +++ /dev/null @@ -1,52 +0,0 @@ -# Noir - -## Constructors - -### new Noir(circuit) - -```ts -new Noir(circuit): Noir -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `circuit` | `CompiledCircuit` | - -#### Returns - -[`Noir`](Noir.md) - -## Methods - -### execute() - -```ts -execute(inputs, foreignCallHandler?): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `inputs` | `InputMap` | -| `foreignCallHandler`? | [`ForeignCallHandler`](../type-aliases/ForeignCallHandler.md) | - -#### Returns - -`Promise`\<`object`\> - -#### Description - -Allows to execute a circuit to get its witness and return value. - -#### Example - -```typescript -async execute(inputs) -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/and.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/and.md deleted file mode 100644 index c783283e396..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/and.md +++ /dev/null @@ -1,22 +0,0 @@ -# and() - -```ts -and(lhs, rhs): string -``` - -Performs a bitwise AND operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/blake2s256.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/blake2s256.md deleted file mode 100644 index 7882d0da8d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/blake2s256.md +++ /dev/null @@ -1,21 +0,0 @@ -# blake2s256() - -```ts -blake2s256(inputs): Uint8Array -``` - -Calculates the Blake2s256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md deleted file mode 100644 index 5e3cd53e9d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256k1\_verify() - -```ts -ecdsa_secp256k1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256k1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md deleted file mode 100644 index 0b20ff68957..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256r1\_verify() - -```ts -ecdsa_secp256r1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256r1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/keccak256.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/keccak256.md deleted file mode 100644 index d10f155ce86..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/keccak256.md +++ /dev/null @@ -1,21 +0,0 @@ -# keccak256() - -```ts -keccak256(inputs): Uint8Array -``` - -Calculates the Keccak256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/sha256.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/sha256.md deleted file mode 100644 index 6ba4ecac022..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/sha256.md +++ /dev/null @@ -1,21 +0,0 @@ -# sha256() - -```ts -sha256(inputs): Uint8Array -``` - -Calculates the SHA256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/xor.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/xor.md deleted file mode 100644 index 8d762b895d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/functions/xor.md +++ /dev/null @@ -1,22 +0,0 @@ -# xor() - -```ts -xor(lhs, rhs): string -``` - -Performs a bitwise XOR operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/index.md deleted file mode 100644 index 166508f7124..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_js - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [Noir](classes/Noir.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [ErrorWithPayload](type-aliases/ErrorWithPayload.md) | - | -| [ForeignCallHandler](type-aliases/ForeignCallHandler.md) | A callback which performs an foreign call and returns the response. | -| [ForeignCallInput](type-aliases/ForeignCallInput.md) | - | -| [ForeignCallOutput](type-aliases/ForeignCallOutput.md) | - | -| [WitnessMap](type-aliases/WitnessMap.md) | - | - -### Functions - -| Function | Description | -| :------ | :------ | -| [and](functions/and.md) | Performs a bitwise AND operation between `lhs` and `rhs` | -| [blake2s256](functions/blake2s256.md) | Calculates the Blake2s256 hash of the input bytes | -| [ecdsa\_secp256k1\_verify](functions/ecdsa_secp256k1_verify.md) | Verifies a ECDSA signature over the secp256k1 curve. | -| [ecdsa\_secp256r1\_verify](functions/ecdsa_secp256r1_verify.md) | Verifies a ECDSA signature over the secp256r1 curve. | -| [keccak256](functions/keccak256.md) | Calculates the Keccak256 hash of the input bytes | -| [sha256](functions/sha256.md) | Calculates the SHA256 hash of the input bytes | -| [xor](functions/xor.md) | Performs a bitwise XOR operation between `lhs` and `rhs` | - -## References - -### CompiledCircuit - -Renames and re-exports [InputMap](index.md#inputmap) - -## Variables - -### InputMap - -```ts -InputMap: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md deleted file mode 100644 index e8c2f4aef3d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md +++ /dev/null @@ -1,15 +0,0 @@ -# ErrorWithPayload - -```ts -type ErrorWithPayload: ExecutionError & object; -``` - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `decodedAssertionPayload` | `any` | - | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md deleted file mode 100644 index 812b8b16481..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md +++ /dev/null @@ -1,24 +0,0 @@ -# ForeignCallHandler - -```ts -type ForeignCallHandler: (name, inputs) => Promise; -``` - -A callback which performs an foreign call and returns the response. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | The identifier for the type of foreign call being performed. | -| `inputs` | [`ForeignCallInput`](ForeignCallInput.md)[] | An array of hex encoded inputs to the foreign call. | - -## Returns - -`Promise`\<[`ForeignCallOutput`](ForeignCallOutput.md)[]\> - -outputs - An array of hex encoded outputs containing the results of the foreign call. - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md deleted file mode 100644 index dd95809186a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallInput - -```ts -type ForeignCallInput: string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md deleted file mode 100644 index b71fb78a946..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallOutput - -```ts -type ForeignCallOutput: string | string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md deleted file mode 100644 index 258c46f9d0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md +++ /dev/null @@ -1,9 +0,0 @@ -# WitnessMap - -```ts -type WitnessMap: Map; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs deleted file mode 100644 index b3156097df6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/noir_js/classes/Noir","label":"Noir"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ErrorWithPayload","label":"ErrorWithPayload"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallHandler","label":"ForeignCallHandler"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallInput","label":"ForeignCallInput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallOutput","label":"ForeignCallOutput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/WitnessMap","label":"WitnessMap"}]},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_js/functions/and","label":"and"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/blake2s256","label":"blake2s256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify","label":"ecdsa_secp256k1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify","label":"ecdsa_secp256r1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/keccak256","label":"keccak256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/sha256","label":"sha256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/xor","label":"xor"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/compile.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/compile.md deleted file mode 100644 index 6faf763b37f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/compile.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile() - -```ts -compile( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ProgramCompilationArtifacts`](../index.md#programcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_program(fm); -``` - -```typescript -// Browser - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_program(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/compile_contract.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/compile_contract.md deleted file mode 100644 index 7d0b39a43ef..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/compile_contract.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile\_contract() - -```ts -compile_contract( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ContractCompilationArtifacts`](../index.md#contractcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_contract(fm); -``` - -```typescript -// Browser - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_contract(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/createFileManager.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/createFileManager.md deleted file mode 100644 index 7e65c1d69c7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/createFileManager.md +++ /dev/null @@ -1,21 +0,0 @@ -# createFileManager() - -```ts -createFileManager(dataDir): FileManager -``` - -Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `dataDir` | `string` | root of the file system | - -## Returns - -`FileManager` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md deleted file mode 100644 index fcea9275341..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md +++ /dev/null @@ -1,21 +0,0 @@ -# inflateDebugSymbols() - -```ts -inflateDebugSymbols(debugSymbols): any -``` - -Decompresses and decodes the debug symbols - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `debugSymbols` | `string` | The base64 encoded debug symbols | - -## Returns - -`any` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/index.md deleted file mode 100644 index b6e0f9d1bc0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_wasm - -## Exports - -### Functions - -| Function | Description | -| :------ | :------ | -| [compile](functions/compile.md) | Compiles a Noir project | -| [compile\_contract](functions/compile_contract.md) | Compiles a Noir project | -| [createFileManager](functions/createFileManager.md) | Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) | -| [inflateDebugSymbols](functions/inflateDebugSymbols.md) | Decompresses and decodes the debug symbols | - -## References - -### compile\_program - -Renames and re-exports [compile](functions/compile.md) - -## Interfaces - -### ContractCompilationArtifacts - -The compilation artifacts of a given contract. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `contract` | `ContractArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -### ProgramCompilationArtifacts - -The compilation artifacts of a given program. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | not part of the compilation output, injected later | -| `program` | `ProgramArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs deleted file mode 100644 index e0870710349..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"doc","id":"reference/NoirJS/noir_wasm/index","label":"API"},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile","label":"compile"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile_contract","label":"compile_contract"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/createFileManager","label":"createFileManager"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/inflateDebugSymbols","label":"inflateDebugSymbols"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/_category_.json deleted file mode 100644 index 5b6a20a609a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 4, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/_category_.json deleted file mode 100644 index 27869205ad3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugger", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_known_limitations.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_known_limitations.md deleted file mode 100644 index 936d416ac4b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_known_limitations.md +++ /dev/null @@ -1,59 +0,0 @@ ---- -title: Known limitations -description: - An overview of known limitations of the current version of the Noir debugger -keywords: - [ - Nargo, - Noir Debugger, - VS Code, - ] -sidebar_position: 2 ---- - -# Debugger Known Limitations - -There are currently some limits to what the debugger can observe. - -## Mutable references - -The debugger is currently blind to any state mutated via a mutable reference. For example, in: - -``` -let mut x = 1; -let y = &mut x; -*y = 2; -``` - -The update on `x` will not be observed by the debugger. That means, when running `vars` from the debugger REPL, or inspecting the _local variables_ pane in the VS Code debugger, `x` will appear with value 1 despite having executed `*y = 2;`. - -## Variables of type function or mutable references are opaque - -When inspecting variables, any variable of type `Function` or `MutableReference` will render its value as `<>` or `<>`. - -## Debugger instrumentation affects resulting ACIR - -In order to make the state of local variables observable, the debugger compiles Noir circuits interleaving foreign calls that track any mutations to them. While this works (except in the cases described above) and doesn't introduce any behavior changes, it does as a side effect produce bigger bytecode. In particular, when running the command `opcodes` on the REPL debugger, you will notice Unconstrained VM blocks that look like this: - -``` -... -5 BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [], q_c: 2 }), Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(2))], q_c: 0 })] - | outputs=[] - 5.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 5.1 | Mov { destination: RegisterIndex(3), source: RegisterIndex(1) } - 5.2 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 5.3 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 5.4 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 5.5 | Mov { destination: RegisterIndex(3), source: RegisterIndex(3) } - 5.6 | Call { location: 8 } - 5.7 | Stop - 5.8 | ForeignCall { function: "__debug_var_assign", destinations: [], inputs: [RegisterIndex(RegisterIndex(2)), RegisterIndex(RegisterIndex(3))] } -... -``` - -If you are interested in debugging/inspecting compiled ACIR without these synthetic changes, you can invoke the REPL debugger with the `--skip-instrumentation` flag or launch the VS Code debugger with the `skipConfiguration` property set to true in its launch configuration. You can find more details about those in the [Debugger REPL reference](debugger_repl.md) and the [VS Code Debugger reference](debugger_vscode.md). - -:::note -Skipping debugger instrumentation means you won't be able to inspect values of local variables. -::: - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_repl.md deleted file mode 100644 index 46e2011304e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_repl.md +++ /dev/null @@ -1,360 +0,0 @@ ---- -title: REPL Debugger -description: - Noir Debugger REPL options and commands. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -## Running the REPL debugger - -`nargo debug [OPTIONS] [WITNESS_NAME]` - -Runs the Noir REPL debugger. If a `WITNESS_NAME` is provided the debugger writes the resulting execution witness to a `WITNESS_NAME` file. - -### Options - -| Option | Description | -| --------------------- | ------------------------------------------------------------ | -| `-p, --prover-name ` | The name of the toml file which contains the inputs for the prover [default: Prover]| -| `--package ` | The name of the package to debug | -| `--print-acir` | Display the ACIR for compiled circuit | -| `--deny-warnings` | Treat all warnings as errors | -| `--silence-warnings` | Suppress warnings | -| `-h, --help` | Print help | - -None of these options are required. - -:::note -Since the debugger starts by compiling the target package, all Noir compiler options are also available. Check out the [compiler reference](../nargo_commands.md#nargo-compile) to learn more about the compiler options. -::: - -## REPL commands - -Once the debugger is running, it accepts the following commands. - -#### `help` (h) - -Displays the menu of available commands. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) value - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -### Stepping through programs - -#### `next` (n) - -Step until the next Noir source code location. While other commands, such as [`into`](#into-i) and [`step`](#step-s), allow for finer grained control of the program's execution at the opcode level, `next` is source code centric. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `next` here would cause the debugger to jump to the definition of `deep_entry_point` (if available). - -If you want to step over `deep_entry_point` and go straight to line 8, use [the `over` command](#over) instead. - -#### `over` - -Step until the next source code location, without diving into function calls. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `over` here would cause the debugger to execute until line 8 (`multiple_values_entry_point(x);`). - -If you want to step into `deep_entry_point` instead, use [the `next` command](#next-n). - -#### `out` - -Step until the end of the current function call. For example: - -``` - 3 ... - 4 fn main(x: u32) { - 5 assert(entry_point(x) == 2); - 6 swap_entry_point(x, x + 1); - 7 -> assert(deep_entry_point(x) == 4); - 8 multiple_values_entry_point(x); - 9 } - 10 - 11 unconstrained fn returns_multiple_values(x: u32) -> (u32, u32, u32, u32) { - 12 ... - ... - 55 - 56 unconstrained fn deep_entry_point(x: u32) -> u32 { - 57 -> level_1(x + 1) - 58 } - -``` - -Running `out` here will resume execution until line 8. - -#### `step` (s) - -Skips to the next ACIR code. A compiled Noir program is a sequence of ACIR opcodes. However, an unconstrained VM opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `step` command at this point would result in the debugger stopping at ACIR opcode 2, `EXPR`, skipping unconstrained computation steps. - -Use [the `into` command](#into-i) instead if you want to follow unconstrained computation step by step. - -#### `into` (i) - -Steps into the next opcode. A compiled Noir program is a sequence of ACIR opcodes. However, a BRILLIG opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `into` command at this point would result in the debugger stopping at opcode 1.0, `Mov ...`, allowing the debugger user to follow unconstrained computation step by step. - -Use [the `step` command](#step-s) instead if you want to skip to the next ACIR code directly. - -#### `continue` (c) - -Continues execution until the next breakpoint, or the end of the program. - -#### `restart` (res) - -Interrupts execution, and restarts a new debugging session from scratch. - -#### `opcodes` (o) - -Display the program's ACIR opcode sequence. For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -### Breakpoints - -#### `break [Opcode]` (or shorthand `b [Opcode]`) - -Sets a breakpoint on the specified opcode index. To get a list of the program opcode numbers, see [the `opcode` command](#opcodes-o). For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -In this example, issuing a `break 1.2` command adds break on opcode 1.2, as denoted by the `*` character: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | * Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -Running [the `continue` command](#continue-c) at this point would cause the debugger to execute the program until opcode 1.2. - -#### `delete [Opcode]` (or shorthand `d [Opcode]`) - -Deletes a breakpoint at an opcode location. Usage is analogous to [the `break` command](#). - -### Variable inspection - -#### vars - -Show variable values available at this point in execution. - -:::note -The ability to inspect variable values from the debugger depends on compilation to be run in a special debug instrumentation mode. This instrumentation weaves variable tracing code with the original source code. - -So variable value inspection comes at the expense of making the resulting ACIR bytecode bigger and harder to understand and optimize. - -If you find this compromise unacceptable, you can run the debugger with the flag `--skip-debug-instrumentation`. This will compile your circuit without any additional debug information, so the resulting ACIR bytecode will be identical to the one produced by standard Noir compilation. However, if you opt for this, the `vars` command will not be available while debugging. -::: - - -### Stacktrace - -#### `stacktrace` - -Displays the current stack trace. - - -### Witness map - -#### `witness` (w) - -Show witness map. For example: - -``` -_0 = 0 -_1 = 2 -_2 = 1 -``` - -#### `witness [Witness Index]` - -Display a single witness from the witness map. For example: - -``` -> witness 1 -_1 = 2 -``` - -#### `witness [Witness Index] [New value]` - -Overwrite the given index with a new value. For example: - -``` -> witness 1 3 -_1 = 3 -``` - - -### Unconstrained VM memory - -#### `memory` - -Show unconstrained VM memory state. For example: - -``` -> memory -At opcode 1.13: Store { destination_pointer: RegisterIndex(0), source: RegisterIndex(3) } -... -> registers -0 = 0 -1 = 10 -2 = 0 -3 = 1 -4 = 1 -5 = 2³² -6 = 1 -> into -At opcode 1.14: Const { destination: RegisterIndex(5), value: Value { inner: 1 } } -... -> memory -0 = 1 -> -``` - -In the example above: we start with clean memory, then step through a `Store` opcode which stores the value of register 3 (1) into the memory address stored in register 0 (0). Thus now `memory` shows memory address 0 contains value 1. - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: - -#### `memset [Memory address] [New value]` - -Update a memory cell with the given value. For example: - -``` -> memory -0 = 1 -> memset 0 2 -> memory -0 = 2 -> memset 1 4 -> memory -0 = 2 -1 = 4 -> -``` - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_vscode.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_vscode.md deleted file mode 100644 index c027332b3b0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/debugger/debugger_vscode.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: VS Code Debugger -description: - VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -# VS Code Noir Debugger Reference - -The Noir debugger enabled by the vscode-noir extension ships with default settings such that the most common scenario should run without any additional configuration steps. - -These defaults can nevertheless be overridden by defining a launch configuration file. This page provides a reference for the properties you can override via a launch configuration file, as well as documenting the Nargo `dap` command, which is a dependency of the VS Code Noir debugger. - - -## Creating and editing launch configuration files - -To create a launch configuration file from VS Code, open the _debug pane_, and click on _create a launch.json file_. - -![Creating a launch configuration file](@site/static/img/debugger/ref1-create-launch.png) - -A `launch.json` file will be created, populated with basic defaults. - -### Noir Debugger launch.json properties - -#### projectFolder - -_String, optional._ - -Absolute path to the Nargo project to debug. By default, it is dynamically determined by looking for the nearest `Nargo.toml` file to the active file at the moment of launching the debugger. - -#### proverName - -_String, optional._ - -Name of the prover input to use. Defaults to `Prover`, which looks for a file named `Prover.toml` at the `projectFolder`. - -#### generateAcir - -_Boolean, optional._ - -If true, generate ACIR opcodes instead of unconstrained opcodes which will be closer to release binaries but less convenient for debugging. Defaults to `false`. - -#### skipInstrumentation - -_Boolean, optional._ - -Skips variables debugging instrumentation of code, making debugging less convenient but the resulting binary smaller and closer to production. Defaults to `false`. - -:::note -Skipping instrumentation causes the debugger to be unable to inspect local variables. -::: - -## `nargo dap [OPTIONS]` - -When run without any option flags, it starts the Nargo Debug Adapter Protocol server, which acts as the debugging backend for the VS Code Noir Debugger. - -All option flags are related to preflight checks. The Debug Adapter Protocol specifies how errors are to be informed from a running DAP server, but it doesn't specify mechanisms to communicate server initialization errors between the DAP server and its client IDE. - -Thus `nargo dap` ships with a _preflight check_ mode. If flag `--preflight-check` and the rest of the `--preflight-*` flags are provided, Nargo will run the same initialization routine except it will not start the DAP server. - -`vscode-noir` will then run `nargo dap` in preflight check mode first before a debugging session starts. If the preflight check ends in error, vscode-noir will present stderr and stdout output from this process through its own Output pane in VS Code. This makes it possible for users to diagnose what pieces of configuration might be wrong or missing in case of initialization errors. - -If the preflight check succeeds, `vscode-noir` proceeds to start the DAP server normally but running `nargo dap` without any additional flags. - -### Options - -| Option | Description | -| --------------------------------------------------------- | --------------------------------------------------------------------------------------------------------- | -| `--preflight-check` | If present, dap runs in preflight check mode. | -| `--preflight-project-folder ` | Absolute path to the project to debug for preflight check. | -| `--preflight-prover-name ` | Name of prover file to use for preflight check | -| `--preflight-generate-acir` | Optional. If present, compile in ACIR mode while running preflight check. | -| `--preflight-skip-instrumentation` | Optional. If present, compile without introducing debug instrumentation while running preflight check. | -| `-h, --help` | Print help. | diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/nargo_commands.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/nargo_commands.md deleted file mode 100644 index 8384788fa81..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/nargo_commands.md +++ /dev/null @@ -1,289 +0,0 @@ ---- -title: Nargo -description: - Noir CLI Commands for Noir Prover and Verifier to create, execute, prove and verify programs, - generate Solidity verifier smart contract and compile into JSON file containing ACIR - representation and ABI of circuit. -keywords: - [ - Nargo, - Noir CLI, - Noir Prover, - Noir Verifier, - generate Solidity verifier, - compile JSON file, - ACIR representation, - ABI of circuit, - TypeScript, - ] -sidebar_position: 0 ---- - -# Command-Line Help for `nargo` - -This document contains the help content for the `nargo` command-line program. - -**Command Overview:** - -* [`nargo`↴](#nargo) -* [`nargo check`↴](#nargo-check) -* [`nargo fmt`↴](#nargo-fmt) -* [`nargo compile`↴](#nargo-compile) -* [`nargo new`↴](#nargo-new) -* [`nargo init`↴](#nargo-init) -* [`nargo execute`↴](#nargo-execute) -* [`nargo debug`↴](#nargo-debug) -* [`nargo test`↴](#nargo-test) -* [`nargo info`↴](#nargo-info) -* [`nargo lsp`↴](#nargo-lsp) - -## `nargo` - -Noir's package manager - -**Usage:** `nargo ` - -###### **Subcommands:** - -* `check` — Checks the constraint system for errors -* `fmt` — Format the Noir files in a workspace -* `compile` — Compile the program and its secret execution trace into ACIR format -* `new` — Create a Noir project in a new directory -* `init` — Create a Noir project in the current directory -* `execute` — Executes a circuit to calculate its return value -* `debug` — Executes a circuit in debug mode -* `test` — Run the tests for this program -* `info` — Provides detailed information on each of a program's function (represented by a single circuit) -* `lsp` — Starts the Noir LSP server - -###### **Options:** - - - - -## `nargo check` - -Checks the constraint system for errors - -**Usage:** `nargo check [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to check -* `--workspace` — Check all packages in the workspace -* `--overwrite` — Force overwrite of existing files -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" - - - -## `nargo fmt` - -Format the Noir files in a workspace - -**Usage:** `nargo fmt [OPTIONS]` - -###### **Options:** - -* `--check` — Run noirfmt in check mode - - - -## `nargo compile` - -Compile the program and its secret execution trace into ACIR format - -**Usage:** `nargo compile [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to compile -* `--workspace` — Compile all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" - - - -## `nargo new` - -Create a Noir project in a new directory - -**Usage:** `nargo new [OPTIONS] ` - -###### **Arguments:** - -* `` — The path to save the new project - -###### **Options:** - -* `--name ` — Name of the package [default: package directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo init` - -Create a Noir project in the current directory - -**Usage:** `nargo init [OPTIONS]` - -###### **Options:** - -* `--name ` — Name of the package [default: current directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo execute` - -Executes a circuit to calculate its return value - -**Usage:** `nargo execute [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--workspace` — Execute all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo debug` - -Executes a circuit in debug mode - -**Usage:** `nargo debug [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--acir-mode` — Force ACIR output (disabling instrumentation) -* `--skip-instrumentation ` — Disable vars debug instrumentation (enabled by default) - - Possible values: `true`, `false` - - - - -## `nargo test` - -Run the tests for this program - -**Usage:** `nargo test [OPTIONS] [TEST_NAME]` - -###### **Arguments:** - -* `` — If given, only tests with names containing this string will be run - -###### **Options:** - -* `--show-output` — Display output of `println` statements -* `--exact` — Only run tests that match exactly -* `--package ` — The name of the package to test -* `--workspace` — Test all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo info` - -Provides detailed information on each of a program's function (represented by a single circuit) - -Current information provided per circuit: 1. The number of ACIR opcodes 2. Counts the final number gates in the circuit used by a backend - -**Usage:** `nargo info [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to detail -* `--workspace` — Detail all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" - - - -## `nargo lsp` - -Starts the Noir LSP server - -Starts an LSP server which allows IDEs such as VS Code to display diagnostics in Noir source. - -VS Code Noir Language Support: https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir - -**Usage:** `nargo lsp` - - - -
- - - This document was generated automatically by - clap-markdown. - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/noir_codegen.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/noir_codegen.md deleted file mode 100644 index db8f07dc22e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/reference/noir_codegen.md +++ /dev/null @@ -1,114 +0,0 @@ ---- -title: Noir Codegen for TypeScript -description: Learn how to use Noir codegen to generate TypeScript bindings -keywords: [Nargo, Noir, compile, TypeScript] -sidebar_position: 3 ---- - -When using TypeScript, it is extra work to interpret Noir program outputs in a type-safe way. Third party libraries may exist for popular Noir programs, but they are either hard to find or unmaintained. - -Now you can generate TypeScript bindings for your Noir programs in two steps: -1. Exporting Noir functions using `nargo export` -2. Using the TypeScript module `noir_codegen` to generate TypeScript binding - -**Note:** you can only export functions from a Noir *library* (not binary or contract program types). - -## Installation - -### Your TypeScript project - -If you don't already have a TypeScript project you can add the module with `yarn` (or `npm`), then initialize it: - -```bash -yarn add typescript -D -npx tsc --init -``` - -### Add TypeScript module - `noir_codegen` - -The following command will add the module to your project's devDependencies: - -```bash -yarn add @noir-lang/noir_codegen -D -``` - -### Nargo library -Make sure you have Nargo, v0.25.0 or greater, installed. If you don't, follow the [installation guide](../getting_started/installation/index.md). - -If you're in a new project, make a `circuits` folder and create a new Noir library: - -```bash -mkdir circuits && cd circuits -nargo new --lib myNoirLib -``` - -## Usage - -### Export ABI of specified functions - -First go to the `.nr` files in your Noir library, and add the `#[export]` macro to each function that you want to use in TypeScript. - -```rust -#[export] -fn your_function(... -``` - -From your Noir library (where `Nargo.toml` is), run the following command: - -```bash -nargo export -``` - -You will now have an `export` directory with a .json file per exported function. - -You can also specify the directory of Noir programs using `--program-dir`, for example: - -```bash -nargo export --program-dir=./circuits/myNoirLib -``` - -### Generate TypeScript bindings from exported functions - -To use the `noir-codegen` package we added to the TypeScript project: - -```bash -yarn noir-codegen ./export/your_function.json -``` - -This creates an `exports` directory with an `index.ts` file containing all exported functions. - -**Note:** adding `--out-dir` allows you to specify an output dir for your TypeScript bindings to go. Eg: - -```bash -yarn noir-codegen ./export/*.json --out-dir ./path/to/output/dir -``` - -## Example .nr function to .ts output - -Consider a Noir library with this function: - -```rust -#[export] -fn not_equal(x: Field, y: Field) -> bool { - x != y -} -``` - -After the export and codegen steps, you should have an `index.ts` like: - -```typescript -export type Field = string; - - -export const is_equal_circuit: CompiledCircuit = -{"abi":{"parameters":[{"name":"x","type":{"kind":"field"},"visibility":"private"},{"name":"y","type":{"kind":"field"},"visibility":"private"}],"return_type":{"abi_type":{"kind":"boolean"},"visibility":"private"}},"bytecode":"H4sIAAAAAAAA/7WUMQ7DIAxFQ0Krrr2JjSGYLVcpKrn/CaqqDQN12WK+hPBgmWd/wEyHbF1SS923uhOs3pfoChI+wKXMAXzIKyNj4PB0TFTYc0w5RUjoqeAeEu1wqK0F54RGkWvW44LPzExnlkbMEs4JNZmN8PxS42uHv82T8a3Jeyn2Ks+VLPcO558HmyLMCDOXAXXtpPt4R/Rt9T36ss6dS9HGPx/eG17nGegKBQAA"}; - -export async function is_equal(x: Field, y: Field, foreignCallHandler?: ForeignCallHandler): Promise { - const program = new Noir(is_equal_circuit); - const args: InputMap = { x, y }; - const { returnValue } = await program.execute(args, foreignCallHandler); - return returnValue as boolean; -} -``` - -Now the `is_equal()` function and relevant types are readily available for use in TypeScript. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/debugger.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/debugger.md deleted file mode 100644 index 9b7565ba9ff..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/debugger.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Debugger -description: Learn about the Noir Debugger, in its REPL or VS Code versions. -keywords: [Nargo, VSCode, Visual Studio Code, REPL, Debugger] -sidebar_position: 2 ---- - -# Noir Debugger - -There are currently two ways of debugging Noir programs: - -1. From VS Code, via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). -2. Via the REPL debugger, which ships with Nargo. - -In order to use either version of the debugger, you will need to install recent enough versions of Noir, [Nargo](../getting_started/installation/index.md) and vscode-noir: - -- Noir & Nargo ≥0.28.0 -- Noir's VS Code extension ≥0.0.11 - -:::info -At the moment, the debugger supports debugging binary projects, but not contracts. -::: - -We cover the VS Code Noir debugger more in depth in [its VS Code debugger how-to guide](../how_to/debugger/debugging_with_vs_code.md) and [the reference](../reference/debugger/debugger_vscode.md). - -The REPL debugger is discussed at length in [the REPL debugger how-to guide](../how_to/debugger/debugging_with_the_repl.md) and [the reference](../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/language_server.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/language_server.md deleted file mode 100644 index 81e0356ef8a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/language_server.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Language Server -description: Learn about the Noir Language Server, how to install the components, and configuration that may be required. -keywords: [Nargo, Language Server, LSP, VSCode, Visual Studio Code] -sidebar_position: 0 ---- - -This section helps you install and configure the Noir Language Server. - -The Language Server Protocol (LSP) has two components, the [Server](#language-server) and the [Client](#language-client). Below we describe each in the context of Noir. - -## Language Server - -The Server component is provided by the Nargo command line tool that you installed at the beginning of this guide. -As long as Nargo is installed and you've used it to run other commands in this guide, it should be good to go! - -If you'd like to verify that the `nargo lsp` command is available, you can run `nargo --help` and look for `lsp` in the list of commands. If you see it, you're using a version of Noir with LSP support. - -## Language Client - -The Client component is usually an editor plugin that launches the Server. It communicates LSP messages between the editor and the Server. For example, when you save a file, the Client will alert the Server, so it can try to compile the project and report any errors. - -Currently, Noir provides a Language Client for Visual Studio Code via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). - -> **Note:** Noir's Language Server Protocol support currently assumes users' VSCode workspace root to be the same as users' Noir project root (i.e. where Nargo.toml lies). -> -> If LSP features seem to be missing / malfunctioning, make sure you are opening your Noir project directly (instead of as a sub-folder) in your VSCode instance. - -When your language server is running correctly and the VSCode plugin is installed, you should see handy codelens buttons for compilation, measuring circuit size, execution, and tests: - -![Compile and Execute](@site/static/img/codelens_compile_execute.png) -![Run test](@site/static/img/codelens_run_test.png) - -You should also see your tests in the `testing` panel: - -![Testing panel](@site/static/img/codelens_testing_panel.png) - -### Configuration - -- **Noir: Enable LSP** - If checked, the extension will launch the Language Server via `nargo lsp` and communicate with it. -- **Noir: Nargo Flags** - Additional flags may be specified if you require them to be added when the extension calls `nargo lsp`. -- **Noir: Nargo Path** - An absolute path to a Nargo binary with the `lsp` command. This may be useful if Nargo is not within the `PATH` of your editor. -- **Noir > Trace: Server** - Setting this to `"messages"` or `"verbose"` will log LSP messages between the Client and Server. Useful for debugging. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/testing.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/testing.md deleted file mode 100644 index 866677da567..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tooling/testing.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Testing in Noir -description: Learn how to use Nargo to test your Noir program in a quick and easy way -keywords: [Nargo, testing, Noir, compile, test] -sidebar_position: 1 ---- - -You can test your Noir programs using Noir circuits. - -Nargo will automatically compile and run any functions which have the decorator `#[test]` on them if -you run `nargo test`. - -For example if you have a program like: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test] -fn test_add() { - assert(add(2,2) == 4); - assert(add(0,1) == 1); - assert(add(1,0) == 1); -} -``` - -Running `nargo test` will test that the `test_add` function can be executed while satisfying all -the constraints which allows you to test that add returns the expected values. Test functions can't -have any arguments currently. - -### Test fail - -You can write tests that are expected to fail by using the decorator `#[test(should_fail)]`. For example: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test(should_fail)] -fn test_add() { - assert(add(2,2) == 5); -} -``` - -You can be more specific and make it fail with a specific reason by using `should_fail_with = ""`: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "What is the airspeed velocity of an unladen swallow")] -fn test_bridgekeeper() { - main(32); -} -``` - -The string given to `should_fail_with` doesn't need to exactly match the failure reason, it just needs to be a substring of it: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "airspeed velocity")] -fn test_bridgekeeper() { - main(32); -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tutorials/noirjs_app.md deleted file mode 100644 index eac28168445..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/tutorials/noirjs_app.md +++ /dev/null @@ -1,362 +0,0 @@ ---- -title: Building a web app with NoirJS -description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. -keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] -sidebar_position: 0 -pagination_next: noir/concepts/data_types/index ---- - -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! - -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/installation/index.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! - -### Nargo - -Run: - -```bash -nargo new circuit -``` - -And... That's about it. Your program is ready to be compiled and run. - -To compile, let's `cd` into the `circuit` folder to enter our project, and call: - -```bash -nargo compile -``` - -This compiles our circuit into `json` format and add it to a new `target` folder. - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` - -::: - -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/hello_noir/index.md) guide. However, we want our app to run on the browser, so we need Vite. - -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. - -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". - -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: - -```bash -cd vite-project -``` - -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; - -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); -``` - -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: - -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev -``` - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... -``` - -::: - -#### Some cleanup - -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. - -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) - -## HTML - -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: - -```html - - - - - - -

Noir app

-
- - -
-
-

Logs

-

Proof

-
- - -``` - -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). - -Start by pasting in this boilerplate code: - -```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} - -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } -}); -``` - -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html -``` - -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - -::: - -## Some NoirJS - -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: - -```ts -import circuit from '../circuit/target/circuit.json'; -``` - -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: - -```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -import { Noir } from '@noir-lang/noir_js'; -``` - -And instantiate them inside our try-catch block: - -```ts -// try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); -// } -``` - -:::note - -For the remainder of the tutorial, everything will be happening inside the `try` block - -::: - -## Our app - -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: - -```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; -``` - -Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: - -```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); -const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); -``` - -You're probably eager to see stuff happening, so go and run your app now! - -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. - -![Getting Started 0](@site/static/img/noir_getting_started_1.png) - -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! - -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. - -## Verifying - -Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: - -```js -display('logs', 'Verifying proof... ⌛'); -const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); -``` - -You have successfully generated a client-side Noir web app! - -![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) - -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. - -## UltraHonk Backend - -Barretenberg has recently exposed a new UltraHonk backend. We can use UltraHonk in NoirJS after version 0.33.0. Everything will be the same as the tutorial above, except that the class we need to import will change: -```js -import { UltraHonkBackend, UltraHonkVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -``` -The backend will then be instantiated as such: -```js -const backend = new UltraHonkBackend(circuit); -``` -Then all the commands to prove and verify your circuit will be same. - -The only feature currently unsupported with UltraHonk are [recursive proofs](../explainers/explainer-recursion.md). \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/cspell.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/cspell.json deleted file mode 100644 index c60b0a597b1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/cspell.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "words": [ - "Cryptdoku" - ] -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-oracle.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-oracle.md deleted file mode 100644 index 821e1f95c04..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-oracle.md +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: Oracles -description: This guide provides an in-depth understanding of how Oracles work in Noir programming. Learn how to use outside calculations in your programs, constrain oracles, and understand their uses and limitations. -keywords: - - Noir Programming - - Oracles - - JSON-RPC - - Foreign Call Handlers - - Constrained Functions - - Blockchain Programming -sidebar_position: 1 ---- - -If you've seen "The Matrix" you may recall "The Oracle" as Gloria Foster smoking cigarettes and baking cookies. While she appears to "know things", she is actually providing a calculation of a pre-determined future. Noir Oracles are similar, in a way. They don't calculate the future (yet), but they allow you to use outside calculations in your programs. - -![matrix oracle prediction](@site/static/img/memes/matrix_oracle.jpeg) - -A Noir program is usually self-contained. You can pass certain inputs to it, and it will generate a deterministic output for those inputs. But what if you wanted to defer some calculation to an outside process or source? - -Oracles are functions that provide this feature. - -## Use cases - -An example usage for Oracles is proving something on-chain. For example, proving that the ETH-USDC quote was below a certain target at a certain block time. Or even making more complex proofs like proving the ownership of an NFT as an anonymous login method. - -Another interesting use case is to defer expensive calculations to be made outside of the Noir program, and then constraining the result; similar to the use of [unconstrained functions](../noir/concepts//unconstrained.md). - -In short, anything that can be constrained in a Noir program but needs to be fetched from an external source is a great candidate to be used in oracles. - -## Constraining oracles - -Just like in The Matrix, Oracles are powerful. But with great power, comes great responsibility. Just because you're using them in a Noir program doesn't mean they're true. Noir has no superpowers. If you want to prove that Portugal won the Euro Cup 2016, you're still relying on potentially untrusted information. - -To give a concrete example, Alice wants to login to the [NounsDAO](https://nouns.wtf/) forum with her username "noir_nouner" by proving she owns a noun without revealing her ethereum address. Her Noir program could have an oracle call like this: - -```rust -#[oracle(getNoun)] -unconstrained fn get_noun(address: Field) -> Field -``` - -This oracle could naively resolve with the number of Nouns she possesses. However, it is useless as a trusted source, as the oracle could resolve to anything Alice wants. In order to make this oracle call actually useful, Alice would need to constrain the response from the oracle, by proving her address and the noun count belongs to the state tree of the contract. - -In short, **Oracles don't prove anything. Your Noir program does.** - -:::danger - -If you don't constrain the return of your oracle, you could be clearly opening an attack vector on your Noir program. Make double-triple sure that the return of an oracle call is constrained! - -::: - -## How to use Oracles - -On CLI, Nargo resolves oracles by making JSON RPC calls, which means it would require an RPC node to be running. - -In JavaScript, NoirJS accepts and resolves arbitrary call handlers (that is, not limited to JSON) as long as they match the expected types the developer defines. Refer to [Foreign Call Handler](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) to learn more about NoirJS's call handling. - -If you want to build using oracles, follow through to the [oracle guide](../how_to/how-to-oracles.md) for a simple example on how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-recursion.md deleted file mode 100644 index df8529ef4e0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-recursion.md +++ /dev/null @@ -1,176 +0,0 @@ ---- -title: Recursive proofs -description: Explore the concept of recursive proofs in Zero-Knowledge programming. Understand how recursion works in Noir, a language for writing smart contracts on the EVM blockchain. Learn through practical examples like Alice and Bob's guessing game, Charlie's recursive merkle tree, and Daniel's reusable components. Discover how to use recursive proofs to optimize computational resources and improve efficiency. - -keywords: - [ - "Recursive Proofs", - "Zero-Knowledge Programming", - "Noir", - "EVM Blockchain", - "Smart Contracts", - "Recursion in Noir", - "Alice and Bob Guessing Game", - "Recursive Merkle Tree", - "Reusable Components", - "Optimizing Computational Resources", - "Improving Efficiency", - "Verification Key", - "Aggregation", - "Recursive zkSNARK schemes", - "PLONK", - "Proving and Verification Keys" - ] -sidebar_position: 1 -pagination_next: how_to/how-to-recursion ---- - -In programming, we tend to think of recursion as something calling itself. A classic example would be the calculation of the factorial of a number: - -```js -function factorial(n) { - if (n === 0 || n === 1) { - return 1; - } else { - return n * factorial(n - 1); - } -} -``` - -In this case, while `n` is not `1`, this function will keep calling itself until it hits the base case, bubbling up the result on the call stack: - -```md - Is `n` 1? <--------- - /\ / - / \ n = n -1 - / \ / - Yes No -------- -``` - -In Zero-Knowledge, recursion has some similarities. - -It is not a Noir function calling itself, but a proof being used as an input to another circuit. In short, you verify one proof *inside* another proof, returning the proof that both proofs are valid. - -This means that, given enough computational resources, you can prove the correctness of any arbitrary number of proofs in a single proof. This could be useful to design state channels (for which a common example would be [Bitcoin's Lightning Network](https://en.wikipedia.org/wiki/Lightning_Network)), to save on gas costs by settling one proof on-chain, or simply to make business logic less dependent on a consensus mechanism. - -## Examples - -Let us look at some of these examples - -### Alice and Bob - Guessing game - -Alice and Bob are friends, and they like guessing games. They want to play a guessing game online, but for that, they need a trusted third-party that knows both of their secrets and finishes the game once someone wins. - -So, they use zero-knowledge proofs. Alice tries to guess Bob's number, and Bob will generate a ZK proof stating whether she succeeded or failed. - -This ZK proof can go on a smart contract, revealing the winner and even giving prizes. However, this means every turn needs to be verified on-chain. This incurs some cost and waiting time that may simply make the game too expensive or time-consuming to be worth it. - -As a solution, Alice proposes the following: "what if Bob generates his proof, and instead of sending it on-chain, I verify it *within* my own proof before playing my own turn?". - -She can then generate a proof that she verified his proof, and so on. - -```md - Did you fail? <-------------------------- - / \ / - / \ n = n -1 - / \ / - Yes No / - | | / - | | / - | You win / - | / - | / -Generate proof of that / - + / - my own guess ---------------- -``` - -### Charlie - Recursive merkle tree - -Charlie is a concerned citizen, and wants to be sure his vote in an election is accounted for. He votes with a ZK proof, but he has no way of knowing that his ZK proof was included in the total vote count! - -If the vote collector puts all of the votes into a [Merkle tree](https://en.wikipedia.org/wiki/Merkle_tree), everyone can prove the verification of two proofs within one proof, as such: - -```md - abcd - __________|______________ - | | - ab cd - _____|_____ ______|______ - | | | | - alice bob charlie daniel -``` - -Doing this recursively allows us to arrive on a final proof `abcd` which if true, verifies the correctness of all the votes. - -### Daniel - Reusable components - -Daniel has a big circuit and a big headache. A part of his circuit is a setup phase that finishes with some assertions that need to be made. But that section alone takes most of the proving time, and is largely independent of the rest of the circuit. - -He might find it more efficient to generate a proof for that setup phase separately, and verify that proof recursively in the actual business logic section of his circuit. This will allow for parallelization of both proofs, which results in a considerable speedup. - -## What params do I need - -As you can see in the [recursion reference](noir/standard_library/recursion.mdx), a simple recursive proof requires: - -- The proof to verify -- The Verification Key of the circuit that generated the proof -- A hash of this verification key, as it's needed for some backends -- The public inputs for the proof - -:::info - -Recursive zkSNARK schemes do not necessarily "verify a proof" in the sense that you expect a true or false to be spit out by the verifier. Rather an aggregation object is built over the public inputs. - -So, taking the example of Alice and Bob and their guessing game: - -- Alice makes her guess. Her proof is *not* recursive: it doesn't verify any proof within it! It's just a standard `assert(x != y)` circuit -- Bob verifies Alice's proof and makes his own guess. In this circuit, he doesn't exactly *prove* the verification of Alice's proof. Instead, he *aggregates* his proof to Alice's proof. The actual verification is done when the full proof is verified, for example when using `nargo verify` or through the verifier smart contract. - -We can imagine recursive proofs a [relay race](https://en.wikipedia.org/wiki/Relay_race). The first runner doesn't have to receive the baton from anyone else, as he/she already starts with it. But when his/her turn is over, the next runner needs to receive it, run a bit more, and pass it along. Even though every runner could theoretically verify the baton mid-run (why not? 🏃🔍), only at the end of the race does the referee verify that the whole race is valid. - -::: - -## Some architecture - -As with everything in computer science, there's no one-size-fits all. But there are some patterns that could help understanding and implementing them. To give three examples: - -### Adding some logic to a proof verification - -This would be an approach for something like our guessing game, where proofs are sent back and forth and are verified by each opponent. This circuit would be divided in two sections: - -- A `recursive verification` section, which would be just the call to `std::verify_proof`, and that would be skipped on the first move (since there's no proof to verify) -- A `guessing` section, which is basically the logic part where the actual guessing happens - -In such a situation, and assuming Alice is first, she would skip the first part and try to guess Bob's number. Bob would then verify her proof on the first section of his run, and try to guess Alice's number on the second part, and so on. - -### Aggregating proofs - -In some one-way interaction situations, recursion would allow for aggregation of simple proofs that don't need to be immediately verified on-chain or elsewhere. - -To give a practical example, a barman wouldn't need to verify a "proof-of-age" on-chain every time he serves alcohol to a customer. Instead, the architecture would comprise two circuits: - -- A `main`, non-recursive circuit with some logic -- A `recursive` circuit meant to verify two proofs in one proof - -The customer's proofs would be intermediate, and made on their phones, and the barman could just verify them locally. He would then aggregate them into a final proof sent on-chain (or elsewhere) at the end of the day. - -### Recursively verifying different circuits - -Nothing prevents you from verifying different circuits in a recursive proof, for example: - -- A `circuit1` circuit -- A `circuit2` circuit -- A `recursive` circuit - -In this example, a regulator could verify that taxes were paid for a specific purchase by aggregating both a `payer` circuit (proving that a purchase was made and taxes were paid), and a `receipt` circuit (proving that the payment was received) - -## How fast is it - -At the time of writing, verifying recursive proofs is surprisingly fast. This is because most of the time is spent on generating the verification key that will be used to generate the next proof. So you are able to cache the verification key and reuse it later. - -Currently, Noir JS packages don't expose the functionality of loading proving and verification keys, but that feature exists in the underlying `bb.js` package. - -## How can I try it - -Learn more about using recursion in Nargo and NoirJS in the [how-to guide](../how_to/how-to-recursion.md) and see a full example in [noir-examples](https://github.com/noir-lang/noir-examples). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-writing-noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-writing-noir.md deleted file mode 100644 index 3ef6e014a2f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/explainers/explainer-writing-noir.md +++ /dev/null @@ -1,177 +0,0 @@ ---- -title: Writing Performant Noir -description: Understand new considerations when writing Noir -keywords: [Noir, programming, rust] -tags: [Optimization] -sidebar_position: 0 ---- - - -This article intends to set you up with key concepts essential for writing more viable applications that use zero knowledge proofs, namely around efficient circuits. - -## Context - 'Efficient' is subjective - -When writing a web application for a performant computer with high-speed internet connection, writing efficient code sometimes is seen as an afterthought only if needed. Large multiplications running at the innermost of nested loops may not even be on a dev's radar. -When writing firmware for a battery-powered microcontroller, you think of cpu cycles as rations to keep within a product's power budget. - -> Code is written to create applications that perform specific tasks within specific constraints - -And these constraints differ depending on where the compiled code is execute. - -### The Ethereum Virtual Machine (EVM) - -In scenarios where extremely low gas costs are required for an Ethereum application to be viable/competitive, Ethereum smart contract developers get into what is colloquially known as: "*gas golfing*". Finding the lowest execution cost of their compiled code (EVM bytecode) to achieve a specific task. - -The equivalent optimization task when writing zk circuits is affectionately referred to as "*gate golfing*", finding the lowest gate representation of the compiled Noir code. - -### Coding for circuits - a paradigm shift - -In zero knowledge cryptography, code is compiled to "circuits" consisting of arithmetic gates, and gate count is the significant cost. Depending on the proving system this is linearly proportionate to proving time, and so from a product point this should be kept as low as possible. - -Whilst writing efficient code for web apps and Solidity has a few key differences, writing efficient circuits have a different set of considerations. It is a bit of a paradigm shift, like writing code for GPUs for the first time... - -For example, drawing a circle at (0, 0) of radius `r`: -- For a single CPU thread, -``` -for theta in 0..2*pi { - let x = r * cos(theta); - let y = r * sin(theta); - draw(x, y); -} // note: would do 0 - pi/2 and draw +ve/-ve x and y. -``` - -- For GPUs (simultaneous parallel calls with x, y across image), -``` -if (x^2 + y^2 = r^2) { - draw(x, y); -} -``` - -([Related](https://www.youtube.com/watch?v=-P28LKWTzrI)) - -Whilst this CPU -> GPU does not translate to circuits exactly, it is intended to exemplify the difference in intuition when coding for different machine capabilities/constraints. - -### Context Takeaway - -For those coming from a primarily web app background, this article will explain what you need to consider when writing circuits. Furthermore, for those experienced writing efficient machine code, prepare to shift what you think is efficient 😬 - -## Translating from Rust - -For some applications using Noir, existing code might be a convenient starting point to then proceed to optimize the gate count of. - -:::note -Many valuable functions and algorithms have been written in more established languages (C/C++), and converted to modern ones (like Rust). -::: - -Fortunately for Noir developers, when needing a particular function a Rust implementation can be readily compiled into Noir with some key changes. While the compiler does a decent amount of optimizations, it won't be able to change code that has been optimized for clock-cycles into code optimized for arithmetic gates. - -A few things to do when converting Rust code to Noir: -- `println!` is not a macro, use `println` function (same for `assert_eq`) -- No early `return` in function. Use constrain via assertion instead -- No passing by reference. Remove `&` operator to pass by value (copy) -- No boolean operators (`&&`, `||`). Use bitwise operators (`&`, `|`) with boolean values -- No type `usize`. Use types `u8`, `u32`, `u64`, ... -- `main` return must be public, `pub` -- No `const`, use `global` -- Noir's LSP is your friend, so error message should be informative enough to resolve syntax issues. - -## Writing efficient Noir for performant products - -The following points help refine our understanding over time. - -:::note -A Noir program makes a statement that can be verified. -::: - -It compiles to a structure that represents the calculation, and can assert results within the calculation at any stage (via the `constrain` keyword). - -A Noir program compiles to an Abstract Circuit Intermediate Representation which is: - - Conceptually a tree structure - - Leaves (inputs) are the `Field` type - - Nodes contain arithmetic operations to combine them (gates) - - The root is the final result (return value) - -:::tip -The command `nargo info` shows the programs circuit size, and is useful to compare the value of changes made. -You can dig deeper and use the `--print-acir` param to take a closer look at individual ACIR opcodes, and the proving backend to see its gate count (eg for barretenberg, `bb gates -b ./target/program.json`). -::: - -### Use the `Field` type - -Since the native type of values in circuits are `Field`s, using them for variables in Noir means less gates converting them under the hood. -Some things to be mindful of when using a Field type for a regular integer value: -- A variable of type `Field` can be cast `as` an integer type (eg `u8`, `u64`) - - Note: this retains only the bits of the integer type. Eg a Field value of 260 as a `u8` becomes 4 -- For Field types arithmetic operations meaningfully overflow/underflow, yet for integer types they are checked according to their size -- Comparisons and bitwise operations do not exist for `Field`s, cast to an appropriately sized integer type when you need to - -:::tip -Where possible, use `Field` type for values. Using smaller value types, and bit-packing strategies, will result in MORE gates -::: - - -### Use Arithmetic over non-arithmetic operations - -Since circuits are made of arithmetic gates, the cost of arithmetic operations tends to be one gate. Whereas for procedural code, they represent several clock cycles. - -Inversely, non-arithmetic operators are achieved with multiple gates, vs 1 clock cycle for procedural code. - -| (cost\op) | arithmetic
(`*`, `+`) | bit-wise ops
(eg `<`, `\|`, `>>`) | -| - | - | - | -| **cycles** | 10+ | 1 | -| **gates** | 1 | 10+ | - -Bit-wise operations (e.g. bit shifts `<<` and `>>`), albeit commonly used in general programming and especially for clock cycle optimizations, are on the contrary expensive in gates when performed within circuits. - -Translate away from bit shifts when writing constrained functions for the best performance. - -On the flip side, feel free to use bit shifts in unconstrained functions and tests if necessary, as they are executed outside of circuits and does not induce performance hits. - -### Use static over dynamic values - -Another general theme that manifests in different ways is that static reads are represented with less gates than dynamic ones. - -Reading from read-only memory (ROM) adds less gates than random-access memory (RAM), 2 vs ~3.25 due to the additional bounds checks. Arrays of fixed length (albeit used at a lower capacity), will generate less gates than dynamic storage. - -Related to this, if an index used to access an array is not known at compile time (ie unknown until run time), then ROM will be converted to RAM, expanding the gate count. - -:::tip -Use arrays and indices that are known at compile time where possible. -Using `assert_constant(i);` before an index, `i`, is used in an array will give a compile error if `i` is NOT known at compile time. -::: - -### Leverage unconstrained execution - -Constrained verification can leverage unconstrained execution, this is especially useful for operations that are represented by many gates. -Use an [unconstrained function](../noir/concepts/unconstrained.md) to perform gate-heavy calculations, then verify and constrain the result. - -Eg division generates more gates than multiplication, so calculating the quotient in an unconstrained function then constraining the product for the quotient and divisor (+ any remainder) equals the dividend will be more efficient. - -Use ` if is_unconstrained() { /`, to conditionally execute code if being called in an unconstrained vs constrained way. - -## Advanced - -Unless you're well into the depth of gate optimization, this advanced section can be ignored. - -### Combine arithmetic operations - -A Noir program can be honed further by combining arithmetic operators in a way that makes the most of each constraint of the backend proving system. This is in scenarios where the backend might not be doing this perfectly. - -Eg Barretenberg backend (current default for Noir) is a width-4 PLONKish constraint system -$ w_1*w_2*q_m + w_1*q_1 + w_2*q_2 + w_3*q_3 + w_4*q_4 + q_c = 0 $ - -Here we see there is one occurrence of witness 1 and 2 ($w_1$, $w_2$) being multiplied together, with addition to witnesses 1-4 ($w_1$ .. $w_4$) multiplied by 4 corresponding circuit constants ($q_1$ .. $q_4$) (plus a final circuit constant, $q_c$). - -Use `nargo info --print-acir`, to inspect the ACIR opcodes (and the proving system for gates), and it may present opportunities to amend the order of operations and reduce the number of constraints. - -#### Variable as witness vs expression - -If you've come this far and really know what you're doing at the equation level, a temporary lever (that will become unnecessary/useless over time) is: `std::as_witness`. This informs the compiler to save a variable as a witness not an expression. - -The compiler will mostly be correct and optimal, but this may help some near term edge cases that are yet to optimize. -Note: When used incorrectly it will create **less** efficient circuits (higher gate count). - -## References -- Guillaume's ["`Cryptdoku`" talk](https://www.youtube.com/watch?v=MrQyzuogxgg) (Jun'23) -- Tips from Tom, Jake and Zac. -- [Idiomatic Noir](https://www.vlayer.xyz/blog/idiomatic-noir-part-1-collections) blog post diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/backend/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/backend/_category_.json deleted file mode 100644 index b82e92beb0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/backend/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "position": 1, - "label": "Install Proving Backend", - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/backend/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/backend/index.md deleted file mode 100644 index 7192d954877..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/backend/index.md +++ /dev/null @@ -1,31 +0,0 @@ ---- -title: Proving Backend Installation -description: Proving backends offer command line tools for proving and verifying Noir programs. This page describes how to install `bb` as an example. -keywords: [ - Proving - Backend - Barretenberg - bb - bbup - Installation - Terminal - Command - CLI - Version -] -pagination_next: getting_started/hello_noir/index ---- - -Proving backends each provide their own tools for working with Noir programs, providing functionality like proof generation, proof verification, and verifier smart contract generation. - -For the latest information on tooling provided by each proving backend, installation instructions, Noir version compatibility... you may refer to the proving backends' own documentation. - -You can find the full list of proving backends compatible with Noir in [Awesome Noir](https://github.com/noir-lang/awesome-noir/?tab=readme-ov-file#proving-backends). - -## Example: Installing `bb` - -`bb` is the CLI tool provided by the [Barretenberg proving backend](https://github.com/AztecProtocol/barretenberg) developed by Aztec Labs. - -You can find the instructions for installation in [`bb`'s documentation](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/cpp/src/barretenberg/bb/readme.md#installation). - -Once installed, we are ready to start working on [our first Noir program](../hello_noir/index.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/_category_.json deleted file mode 100644 index 976a2325de0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/index.md deleted file mode 100644 index 6760e54aad1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/index.md +++ /dev/null @@ -1,157 +0,0 @@ ---- -title: Creating a Project -description: - Learn how to create and verify your first Noir program using Nargo, a programming language for - zero-knowledge proofs. -keywords: - [ - Nargo, - Noir, - zero-knowledge proofs, - programming language, - create Noir program, - verify Noir program, - step-by-step guide, - ] -sidebar_position: 1 - ---- - -Now that we have installed Nargo and a proving backend, it is time to make our first hello world program! - -### 1. Create a new project directory - -Noir code can live anywhere on your computer. Let us create a _projects_ folder in the home -directory to house our first Noir program. - -Create the directory and change directory into it by running: - -```sh -mkdir ~/projects -cd ~/projects -``` - -## Nargo - -Nargo provides the ability to initiate and execute Noir projects. Read the [Nargo installation](../installation/index.md) section to learn more about Nargo and how to install it. - -### 2. Create a new Noir project - -Now that we are in the projects directory, create a new Nargo project by running: - -```sh -nargo new hello_world -``` - -`hello_world` can be any arbitrary project name, we are simply using `hello_world` for demonstration. - -In production, it is common practice to name the project folder, `circuits`, for clarity amongst other folders in the codebase (like: `contracts`, `scripts`, and `test`). - -A `hello_world` folder would be created. Similar to Rust, the folder houses _src/main.nr_ and -_Nargo.toml_ which contain the source code and environmental options of your Noir program -respectively. - -#### Intro to Noir Syntax - -Let us take a closer look at _main.nr_. The default _main.nr_ generated should look like this: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -The first line of the program specifies the program's inputs: - -```rust -x : Field, y : pub Field -``` - -Program inputs in Noir are private by default (e.g. `x`), but can be labeled public using the -keyword `pub` (e.g. `y`). To learn more about private and public values, check the -[Data Types](../../noir/concepts/data_types/index.md) section. - -The next line of the program specifies its body: - -```rust -assert(x != y); -``` - -The Noir syntax `assert` can be interpreted as something similar to constraints in other zk-contract languages. - -For more Noir syntax, check the [Language Concepts](../../noir/concepts/comments.md) chapter. - -### 3. Build in/output files - -Change directory into _hello_world_ and build in/output files for your Noir program by running: - -```sh -cd hello_world -nargo check -``` - -A _Prover.toml_ file will be generated in your project directory, to allow specifying input values to the program. - -### 4. Execute the Noir program - -Now that the project is set up, we can execute our Noir program. - -Fill in input values for execution in the _Prover.toml_ file. For example: - -```toml -x = "1" -y = "2" -``` - -Execute your Noir program: - -```sh -nargo execute witness-name -``` - -The witness corresponding to this execution will then be written to the file `./target/witness-name.gz`. - -The command also automatically compiles your Noir program if it was not already / was edited, which you may notice the compiled artifacts being written to the file `./target/hello_world.json`. - -## Proving Backend - -Proving backends provide the ability to generate and verify proofs of executing Noir programs, following Noir's tooling that compiles and executes the programs. Read the [proving backend installation](../backend/index.md) section to learn more about proving backends and how to install them. - -Barretenberg is used as an example here to demonstrate how proving and verifying could be implemented and used. Read the [`bb` installation](../backend/index.md#example-installing-bb) section for how to install Barretenberg's CLI tool; refer to [`bb`'s documentation](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/cpp/src/barretenberg/bb/readme.md) for full details about the tool. - -### 5. Prove an execution of the Noir program - -Using Barretenberg as an example, prove the valid execution of your Noir program running: - -```sh -bb prove -b ./target/hello_world.json -w ./target/witness-name.gz -o ./target/proof -``` - -The proof generated will then be written to the file `./target/proof`. - -:::tip -Since the params for `nargo` and `bb` often specify multiple filenames to read from or write to, remember to check each command is referring to the desired filenames. -Or for greater certainty, delete the target folder and go through each step again (compile, witness, prove, ...) to ensure files generated in past commands are being referenced in future ones. -::: - -### 6. Verify the execution proof - -Once a proof is generated, we can verify correct execution of our Noir program by verifying the proof file. - -Using Barretenberg as an example, compute the verification key for the Noir program by running: - -```sh -bb write_vk -b ./target/hello_world.json -o ./target/vk -``` - -And verify your proof by running: - -```sh -bb verify -k ./target/vk -p ./target/proof -``` - -If successful, the verification will complete in silence; if unsuccessful, the command will trigger logging of the corresponding error. - -Congratulations, you have now created and verified a proof for your very first Noir program! - -In the [next section](./project_breakdown.md), we will go into more detail on each step performed. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/project_breakdown.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/project_breakdown.md deleted file mode 100644 index 96e653f6c08..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/hello_noir/project_breakdown.md +++ /dev/null @@ -1,159 +0,0 @@ ---- -title: Project Breakdown -description: - Learn about the anatomy of a Nargo project, including the purpose of the Prover TOML - file, and how to prove and verify your program. -keywords: - [Nargo, Nargo project, Prover.toml, proof verification, private asset transfer] -sidebar_position: 2 ---- - -This section breaks down our hello world program from the previous section. - -## Anatomy of a Nargo Project - -Upon creating a new project with `nargo new` and building the in/output files with `nargo check` -commands, you would get a minimal Nargo project of the following structure: - - - src - - Prover.toml - - Nargo.toml - -The source directory _src_ holds the source code for your Noir program. By default only a _main.nr_ -file will be generated within it. - -### Prover.toml - -_Prover.toml_ is used for specifying the input values for executing and proving the program. You can specify `toml` files with different names by using the `--prover-name` or `-p` flags, see the [Prover](#provertoml) section below. Optionally you may specify expected output values for prove-time checking as well. - -### Nargo.toml - -_Nargo.toml_ contains the environmental options of your project. It contains a "package" section and a "dependencies" section. - -Example Nargo.toml: - -```toml -[package] -name = "noir_starter" -type = "bin" -authors = ["Alice"] -compiler_version = "0.9.0" -description = "Getting started with Noir" -entry = "circuit/main.nr" -license = "MIT" - -[dependencies] -ecrecover = {tag = "v0.9.0", git = "https://github.com/colinnielsen/ecrecover-noir.git"} -``` - -Nargo.toml for a [workspace](../../noir/modules_packages_crates/workspaces.md) will look a bit different. For example: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -#### Package section - -The package section defines a number of fields including: - -- `name` (**required**) - the name of the package -- `type` (**required**) - can be "bin", "lib", or "contract" to specify whether its a binary, library or Aztec contract -- `authors` (optional) - authors of the project -- `compiler_version` - specifies the version of the compiler to use. This is enforced by the compiler and follow's [Rust's versioning](https://doc.rust-lang.org/cargo/reference/manifest.html#the-version-field), so a `compiler_version = 0.18.0` will enforce Nargo version 0.18.0, `compiler_version = ^0.18.0` will enforce anything above 0.18.0 but below 0.19.0, etc. For more information, see how [Rust handles these operators](https://docs.rs/semver/latest/semver/enum.Op.html) -- `description` (optional) -- `entry` (optional) - a relative filepath to use as the entry point into your package (overrides the default of `src/lib.nr` or `src/main.nr`) -- `backend` (optional) -- `license` (optional) -- `expression_width` (optional) - Sets the default backend expression width. This field will override the default backend expression width specified by the Noir compiler (currently set to width 4). - -#### Dependencies section - -This is where you will specify any dependencies for your project. See the [Dependencies page](../../noir/modules_packages_crates/dependencies.md) for more info. - -`./proofs/` and `./contract/` directories will not be immediately visible until you create a proof or -verifier contract respectively. - -### main.nr - -The _main.nr_ file contains a `main` method, this method is the entry point into your Noir program. - -In our sample program, _main.nr_ looks like this: - -```rust -fn main(x : Field, y : Field) { - assert(x != y); -} -``` - -The parameters `x` and `y` can be seen as the API for the program and must be supplied by the prover. Since neither `x` nor `y` is marked as public, the verifier does not supply any inputs, when verifying the proof. - -The prover supplies the values for `x` and `y` in the _Prover.toml_ file. - -As for the program body, `assert` ensures that the condition to be satisfied (e.g. `x != y`) is constrained by the proof of the execution of said program (i.e. if the condition was not met, the verifier would reject the proof as an invalid proof). - -### Prover.toml - -The _Prover.toml_ file is a file which the prover uses to supply the inputs to the Noir program (both private and public). - -In our hello world program the _Prover.toml_ file looks like this: - -```toml -x = "1" -y = "2" -``` - -When the command `nargo execute` is executed, nargo will execute the Noir program using the inputs specified in `Prover.toml`, aborting if it finds that these do not satisfy the constraints defined by `main`. In this example, `x` and `y` must satisfy the inequality constraint `assert(x != y)`. - -If an output name is specified such as `nargo execute foo`, the witness generated by this execution will be written to `./target/foo.gz`. This can then be used to generate a proof of the execution. - -#### Arrays of Structs - -The following code shows how to pass an array of structs to a Noir program to generate a proof. - -```rust -// main.nr -struct Foo { - bar: Field, - baz: Field, -} - -fn main(foos: [Foo; 3]) -> pub Field { - foos[2].bar + foos[2].baz -} -``` - -Prover.toml: - -```toml -[[foos]] # foos[0] -bar = 0 -baz = 0 - -[[foos]] # foos[1] -bar = 0 -baz = 0 - -[[foos]] # foos[2] -bar = 1 -baz = 2 -``` - -#### Custom toml files - -You can specify a `toml` file with a different name to use for execution by using the `--prover-name` or `-p` flags. - -This command looks for proof inputs in the default **Prover.toml** and generates the witness and saves it at `./target/foo.gz`: - -```bash -nargo execute foo -``` - -This command looks for proof inputs in the custom **OtherProver.toml** and generates the witness and saves it at `./target/bar.gz`: - -```bash -nargo execute -p OtherProver bar -``` - -Now that you understand the concepts, you'll probably want some editor feedback while you are writing more complex code. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/_category_.json deleted file mode 100644 index 0c02fb5d4d7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "position": 0, - "label": "Install Nargo", - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/index.md deleted file mode 100644 index 53ea9c7891c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/index.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Nargo Installation -description: - nargo is a command line tool for interacting with Noir programs. This page is a quick guide on how to install Nargo through the most common and easy method, noirup -keywords: [ - Nargo - Noir - Rust - Cargo - Noirup - Installation - Terminal Commands - Version Check - Nightlies - Specific Versions - Branches - Noirup Repository -] -pagination_next: getting_started/hello_noir/index ---- - -`nargo` is a tool for working with Noir programs on the CLI, providing you with the ability to start new projects, compile, execute and test Noir programs from the terminal. - -The name is inspired by Rust's package manager `cargo`; and similar to Rust's `rustup`, Noir also has an easy installation script `noirup`. - -## Installing Noirup - -Open a terminal on your machine, and write: - -```bash -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Close the terminal, open another one, and run - -```bash -noirup -``` - -Done. That's it. You should have the latest version working. You can check with `nargo --version`. - -You can also install nightlies, specific versions -or branches. Check out the [noirup repository](https://github.com/noir-lang/noirup) for more -information. - -Now we're ready to start working on [our first Noir program!](../hello_noir/index.md) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/other_install_methods.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/other_install_methods.md deleted file mode 100644 index 3634723562b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/getting_started/installation/other_install_methods.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Alternative Installations -description: There are different ways to install Nargo, the one-stop shop and command-line tool for developing Noir programs. This guide explains how to specify which version to install when using noirup, and using WSL for windows. -keywords: [ - Installation - Nargo - Noirup - Binaries - Compiling from Source - WSL for Windows - macOS - Linux - Nix - Direnv - Uninstalling Nargo - ] -sidebar_position: 1 ---- - -## Encouraged Installation Method: Noirup - -Noirup is the endorsed method for installing Nargo, streamlining the process of fetching binaries or compiling from source. It supports a range of options to cater to your specific needs, from nightly builds and specific versions to compiling from various sources. - -### Installing Noirup - -First, ensure you have `noirup` installed: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -### Fetching Binaries - -With `noirup`, you can easily switch between different Nargo versions, including nightly builds: - -- **Nightly Version**: Install the latest nightly build. - - ```sh - noirup --version nightly - ``` - -- **Specific Version**: Install a specific version of Nargo. - ```sh - noirup --version - ``` - -### Compiling from Source - -`noirup` also enables compiling Nargo from various sources: - -- **From a Specific Branch**: Install from the latest commit on a branch. - - ```sh - noirup --branch - ``` - -- **From a Fork**: Install from the main branch of a fork. - - ```sh - noirup --repo - ``` - -- **From a Specific Branch in a Fork**: Install from a specific branch in a fork. - - ```sh - noirup --repo --branch - ``` - -- **From a Specific Pull Request**: Install from a specific PR. - - ```sh - noirup --pr - ``` - -- **From a Specific Commit**: Install from a specific commit. - - ```sh - noirup -C - ``` - -- **From Local Source**: Compile and install from a local directory. - ```sh - noirup --path ./path/to/local/source - ``` - -## Installation on Windows - -The default backend for Noir (Barretenberg) doesn't provide Windows binaries at this time. For that reason, Noir cannot be installed natively. However, it is available by using Windows Subsystem for Linux (WSL). - -Step 1: Follow the instructions [here](https://learn.microsoft.com/en-us/windows/wsl/install) to install and run WSL. - -step 2: Follow the [Noirup instructions](#encouraged-installation-method-noirup). - -## Uninstalling Nargo - -If you installed Nargo with `noirup`, you can uninstall Nargo by removing the files in `~/.nargo`, `~/nargo`, and `~/noir_cache`. This ensures that all installed binaries, configurations, and cache related to Nargo are fully removed from your system. - -```bash -rm -r ~/.nargo -rm -r ~/nargo -rm -r ~/noir_cache -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/_category_.json deleted file mode 100644 index 23b560f610b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/_category_.json deleted file mode 100644 index cc2cbb1c253..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugging", - "position": 5, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/debugging_with_the_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/debugging_with_the_repl.md deleted file mode 100644 index 1d64dae3f37..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/debugging_with_the_repl.md +++ /dev/null @@ -1,164 +0,0 @@ ---- -title: Using the REPL Debugger -description: - Step-by-step guide on how to debug your Noir circuits with the REPL Debugger. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -#### Pre-requisites - -In order to use the REPL debugger, first you need to install recent enough versions of Nargo and vscode-noir. - -## Debugging a simple circuit - -Let's debug a simple circuit: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -To start the REPL debugger, using a terminal, go to a Noir circuit's home directory. Then: - -`$ nargo debug` - -You should be seeing this in your terminal: - -``` -[main] Starting debugger -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:9 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> -``` - -The debugger displays the current Noir code location, and it is now waiting for us to drive it. - -Let's first take a look at the available commands. For that we'll use the `help` command. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -Some commands operate only for unconstrained functions, such as `memory` and `memset`. If you try to use them while execution is paused at an ACIR opcode, the debugger will simply inform you that you are not executing unconstrained code: - -``` -> memory -Unconstrained VM memory not available -> -``` - -Before continuing, we can take a look at the initial witness map: - -``` -> witness -_0 = 1 -_1 = 2 -> -``` - -Cool, since `x==1`, `y==2`, and we want to check that `x != y`, our circuit should succeed. At this point we could intervene and use the witness setter command to change one of the witnesses. Let's set `y=3`, then back to 2, so we don't affect the expected result: - -``` -> witness -_0 = 1 -_1 = 2 -> witness 1 3 -_1 = 3 -> witness -_0 = 1 -_1 = 3 -> witness 1 2 -_1 = 2 -> witness -_0 = 1 -_1 = 2 -> -``` - -Now we can inspect the current state of local variables. For that we use the `vars` command. - -``` -> vars -> -``` - -We currently have no vars in context, since we are at the entry point of the program. Let's use `next` to execute until the next point in the program. - -``` -> vars -> next -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:20 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> vars -x:Field = 0x01 -``` - -As a result of stepping, the variable `x`, whose initial value comes from the witness map, is now in context and returned by `vars`. - -``` -> next - 1 fn main(x : Field, y : pub Field) { - 2 -> assert(x != y); - 3 } -> vars -y:Field = 0x02 -x:Field = 0x01 -``` - -Stepping again we can finally see both variables and their values. And now we can see that the next assertion should succeed. - -Let's continue to the end: - -``` -> continue -(Continuing execution...) -Finished execution -> q -[main] Circuit witness successfully solved -``` - -Upon quitting the debugger after a solved circuit, the resulting circuit witness gets saved, equivalent to what would happen if we had run the same circuit with `nargo execute`. - -We just went through the basics of debugging using Noir REPL debugger. For a comprehensive reference, check out [the reference page](../../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/debugging_with_vs_code.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/debugging_with_vs_code.md deleted file mode 100644 index a5858c1a5eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/debugger/debugging_with_vs_code.md +++ /dev/null @@ -1,68 +0,0 @@ ---- -title: Using the VS Code Debugger -description: - Step by step guide on how to debug your Noir circuits with the VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -This guide will show you how to use VS Code with the vscode-noir extension to debug a Noir project. - -#### Pre-requisites - -- Nargo -- vscode-noir -- A Noir project with a `Nargo.toml`, `Prover.toml` and at least one Noir (`.nr`) containing an entry point function (typically `main`). - -## Running the debugger - -The easiest way to start debugging is to open the file you want to debug, and press `F5`. This will cause the debugger to launch, using your `Prover.toml` file as input. - -You should see something like this: - -![Debugger launched](@site/static/img/debugger/1-started.png) - -Let's inspect the state of the program. For that, we open VS Code's _Debug pane_. Look for this icon: - -![Debug pane icon](@site/static/img/debugger/2-icon.png) - -You will now see two categories of variables: Locals and Witness Map. - -![Debug pane expanded](@site/static/img/debugger/3-debug-pane.png) - -1. **Locals**: variables of your program. At this point in execution this section is empty, but as we step through the code it will get populated by `x`, `result`, `digest`, etc. - -2. **Witness map**: these are initially populated from your project's `Prover.toml` file. In this example, they will be used to populate `x` and `result` at the beginning of the `main` function. - -Most of the time you will probably be focusing mostly on locals, as they represent the high level state of your program. - -You might be interested in inspecting the witness map in case you are trying to solve a really low level issue in the compiler or runtime itself, so this concerns mostly advanced or niche users. - -Let's step through the program, by using the debugger buttons or their corresponding keyboard shortcuts. - -![Debugger buttons](@site/static/img/debugger/4-debugger-buttons.png) - -Now we can see in the variables pane that there's values for `digest`, `result` and `x`. - -![Inspecting locals](@site/static/img/debugger/5-assert.png) - -We can also inspect the values of variables by directly hovering on them on the code. - -![Hover locals](@site/static/img/debugger/6-hover.png) - -Let's set a break point at the `keccak256` function, so we can continue execution up to the point when it's first invoked without having to go one step at a time. - -We just need to click the to the right of the line number 18. Once the breakpoint appears, we can click the `continue` button or use its corresponding keyboard shortcut (`F5` by default). - -![Breakpoint](@site/static/img/debugger/7-break.png) - -Now we are debugging the `keccak256` function, notice the _Call Stack pane_ at the lower right. This lets us inspect the current call stack of our process. - -That covers most of the current debugger functionalities. Check out [the reference](../../reference/debugger/debugger_vscode.md) for more details on how to configure the debugger. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-oracles.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-oracles.md deleted file mode 100644 index 392dd8b452e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-oracles.md +++ /dev/null @@ -1,275 +0,0 @@ ---- -title: How to use Oracles -description: Learn how to use oracles in your Noir program with examples in both Nargo and NoirJS. This guide also covers writing a JSON RPC server and providing custom foreign call handlers for NoirJS. -keywords: - - Noir Programming - - Oracles - - Nargo - - NoirJS - - JSON RPC Server - - Foreign Call Handlers -sidebar_position: 1 ---- - -This guide shows you how to use oracles in your Noir program. For the sake of clarity, it assumes that: - -- You have read the [explainer on Oracles](../explainers/explainer-oracle.md) and are comfortable with the concept. -- You have a Noir program to add oracles to. You can create one using the [vite-hardhat starter](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) as a boilerplate. -- You understand the concept of a JSON-RPC server. Visit the [JSON-RPC website](https://www.jsonrpc.org/) if you need a refresher. -- You are comfortable with server-side JavaScript (e.g. Node.js, managing packages, etc.). - -## Rundown - -This guide has 3 major steps: - -1. How to modify our Noir program to make use of oracle calls as unconstrained functions -2. How to write a JSON RPC Server to resolve these oracle calls with Nargo -3. How to use them in Nargo and how to provide a custom resolver in NoirJS - -## Step 1 - Modify your Noir program - -An oracle is defined in a Noir program by defining two methods: - -- An unconstrained method - This tells the compiler that it is executing an [unconstrained functions](../noir/concepts//unconstrained.md). -- A decorated oracle method - This tells the compiler that this method is an RPC call. - -An example of an oracle that returns a `Field` would be: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(number: Field) -> Field { } - -unconstrained fn get_sqrt(number: Field) -> Field { - sqrt(number) -} -``` - -In this example, we're wrapping our oracle function in an unconstrained method, and decorating it with `oracle(getSqrt)`. We can then call the unconstrained function as we would call any other function: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); -} -``` - -In the next section, we will make this `getSqrt` (defined on the `sqrt` decorator) be a method of the RPC server Noir will use. - -:::danger - -As explained in the [Oracle Explainer](../explainers/explainer-oracle.md), this `main` function is unsafe unless you constrain its return value. For example: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); - assert(sqrt.pow_32(2) as u64 == input as u64); // <---- constrain the return of an oracle! -} -``` - -::: - -:::info - -Currently, oracles only work with single params or array params. For example: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt([Field; 2]) -> [Field; 2] { } -``` - -::: - -## Step 2 - Write an RPC server - -Brillig will call *one* RPC server. Most likely you will have to write your own, and you can do it in whatever language you prefer. In this guide, we will do it in Javascript. - -Let's use the above example of an oracle that consumes an array with two `Field` and returns their square roots: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(input: [Field; 2]) -> [Field; 2] { } - -unconstrained fn get_sqrt(input: [Field; 2]) -> [Field; 2] { - sqrt(input) -} - -fn main(input: [Field; 2]) { - let sqrt = get_sqrt(input); - assert(sqrt[0].pow_32(2) as u64 == input[0] as u64); - assert(sqrt[1].pow_32(2) as u64 == input[1] as u64); -} - -#[test] -fn test() { - let input = [4, 16]; - main(input); -} -``` - -:::info - -Why square root? - -In general, computing square roots is computationally more expensive than multiplications, which takes a toll when speaking about ZK applications. In this case, instead of calculating the square root in Noir, we are using our oracle to offload that computation to be made in plain. In our circuit we can simply multiply the two values. - -::: - -Now, we should write the correspondent RPC server, starting with the [default JSON-RPC 2.0 boilerplate](https://www.npmjs.com/package/json-rpc-2.0#example): - -```js -import { JSONRPCServer } from "json-rpc-2.0"; -import express from "express"; -import bodyParser from "body-parser"; - -const app = express(); -app.use(bodyParser.json()); - -const server = new JSONRPCServer(); -app.post("/", (req, res) => { - const jsonRPCRequest = req.body; - server.receive(jsonRPCRequest).then((jsonRPCResponse) => { - if (jsonRPCResponse) { - res.json(jsonRPCResponse); - } else { - res.sendStatus(204); - } - }); -}); - -app.listen(5555); -``` - -Now, we will add our `getSqrt` method, as expected by the `#[oracle(getSqrt)]` decorator in our Noir code. It maps through the params array and returns their square roots: - -```js -server.addMethod("resolve_foreign_call", async (params) => { - if (params[0].function !== "getSqrt") { - throw Error("Unexpected foreign call") - }; - const values = params[0].inputs[0].map((field) => { - return `${Math.sqrt(parseInt(field, 16))}`; - }); - return { values: [values] }; -}); -``` - -If you're using Typescript, the following types may be helpful in understanding the expected return value and making sure they're easy to follow: - -```js -export type ForeignCallSingle = string; - -export type ForeignCallArray = string[]; - -export type ForeignCallResult = { - values: (ForeignCallSingle | ForeignCallArray)[]; -}; -``` - -:::info Multidimensional Arrays - -If the Oracle function is returning an array containing other arrays, such as `[['1','2],['3','4']]`, you need to provide the values in JSON as flattened values. In the previous example, it would be `['1', '2', '3', '4']`. In the Noir program, the Oracle signature can use a nested type, the flattened values will be automatically converted to the nested type. - -::: - -## Step 3 - Usage with Nargo - -Using the [`nargo` CLI tool](../getting_started/installation/index.md), you can use oracles in the `nargo test` and `nargo execute` commands by passing a value to `--oracle-resolver`. For example: - -```bash -nargo test --oracle-resolver http://localhost:5555 -``` - -This tells `nargo` to use your RPC Server URL whenever it finds an oracle decorator. - -## Step 4 - Usage with NoirJS - -In a JS environment, an RPC server is not strictly necessary, as you may want to resolve your oracles without needing any JSON call at all. NoirJS simply expects that you pass a callback function when you generate proofs, and that callback function can be anything. - -For example, if your Noir program expects the host machine to provide CPU pseudo-randomness, you could simply pass it as the `foreignCallHandler`. You don't strictly need to create an RPC server to serve pseudo-randomness, as you may as well get it directly in your app: - -```js -const foreignCallHandler = (name, inputs) => crypto.randomBytes(16) // etc - -await noir.execute(inputs, foreignCallHandler) -``` - -As one can see, in NoirJS, the [`foreignCallHandler`](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) function simply means "a callback function that returns a value of type [`ForeignCallOutput`](../reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md). It doesn't have to be an RPC call like in the case for Nargo. - -:::tip - -Does this mean you don't have to write an RPC server like in [Step #2](#step-2---write-an-rpc-server)? - -You don't technically have to, but then how would you run `nargo test`? To use both `Nargo` and `NoirJS` in your development flow, you will have to write a JSON RPC server. - -::: - -In this case, let's make `foreignCallHandler` call the JSON RPC Server we created in [Step #2](#step-2---write-an-rpc-server), by making it a JSON RPC Client. - -For example, using the same `getSqrt` program in [Step #1](#step-1---modify-your-noir-program) (comments in the code): - -```js -import { JSONRPCClient } from "json-rpc-2.0"; - -// declaring the JSONRPCClient -const client = new JSONRPCClient((jsonRPCRequest) => { -// hitting the same JSON RPC Server we coded above - return fetch("http://localhost:5555", { - method: "POST", - headers: { - "content-type": "application/json", - }, - body: JSON.stringify(jsonRPCRequest), - }).then((response) => { - if (response.status === 200) { - return response - .json() - .then((jsonRPCResponse) => client.receive(jsonRPCResponse)); - } else if (jsonRPCRequest.id !== undefined) { - return Promise.reject(new Error(response.statusText)); - } - }); -}); - -// declaring a function that takes the name of the foreign call (getSqrt) and the inputs -const foreignCallHandler = async (name, input) => { - const inputs = input[0].map((i) => i.toString("hex")) - // notice that the "inputs" parameter contains *all* the inputs - // in this case we to make the RPC request with the first parameter "numbers", which would be input[0] - const oracleReturn = await client.request("resolve_foreign_call", [ - { - function: name, - inputs: [inputs] - }, - ]); - return [oracleReturn.values[0]]; -}; - -// the rest of your NoirJS code -const input = { input: [4, 16] }; -const { witness } = await noir.execute(input, foreignCallHandler); -``` - -:::tip - -If you're in a NoirJS environment running your RPC server together with a frontend app, you'll probably hit a familiar problem in full-stack development: requests being blocked by [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) policy. For development only, you can simply install and use the [`cors` npm package](https://www.npmjs.com/package/cors) to get around the problem: - -```bash -yarn add cors -``` - -and use it as a middleware: - -```js -import cors from "cors"; - -const app = express(); -app.use(cors()) -``` - -::: - -## Conclusion - -Hopefully by the end of this guide, you should be able to: - -- Write your own logic around Oracles and how to write a JSON RPC server to make them work with your Nargo commands. -- Provide custom foreign call handlers for NoirJS. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-recursion.md deleted file mode 100644 index c8c4dc9f5b4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-recursion.md +++ /dev/null @@ -1,180 +0,0 @@ ---- -title: How to use recursion on NoirJS -description: Learn how to implement recursion with NoirJS, a powerful tool for creating smart contracts on the EVM blockchain. This guide assumes familiarity with NoirJS, solidity verifiers, and the Barretenberg proving backend. Discover how to generate both final and intermediate proofs using `noir_js` and `backend_barretenberg`. -keywords: - [ - "NoirJS", - "EVM blockchain", - "smart contracts", - "recursion", - "solidity verifiers", - "Barretenberg backend", - "noir_js", - "backend_barretenberg", - "intermediate proofs", - "final proofs", - "nargo compile", - "json import", - "recursive circuit", - "recursive app" - ] -sidebar_position: 1 ---- - -This guide shows you how to use recursive proofs in your NoirJS app. For the sake of clarity, it is assumed that: - -- You already have a NoirJS app. If you don't, please visit the [NoirJS tutorial](../tutorials/noirjs_app.md) and the [reference](../reference/NoirJS/noir_js/index.md). -- You are familiar with what are recursive proofs and you have read the [recursion explainer](../explainers/explainer-recursion.md) -- You already built a recursive circuit following [the reference](../noir/standard_library/recursion.mdx), and understand how it works. - -It is also assumed that you're not using `noir_wasm` for compilation, and instead you've used [`nargo compile`](../reference/nargo_commands.md) to generate the `json` you're now importing into your project. However, the guide should work just the same if you're using `noir_wasm`. - -:::info - -As you've read in the [explainer](../explainers/explainer-recursion.md), a recursive proof is an intermediate proof. This means that it doesn't necessarily generate the final step that makes it verifiable in a smart contract. However, it is easy to verify within another circuit. - -While "standard" usage of NoirJS packages abstracts final proofs, it currently lacks the necessary interface to abstract away intermediate proofs. This means that these proofs need to be created by using the backend directly. - -In short: - -- `noir_js` generates *only* final proofs -- `backend_barretenberg` generates both types of proofs - -::: - -In a standard recursive app, you're also dealing with at least two circuits. For the purpose of this guide, we will assume the following: - -- `main`: a circuit of type `assert(x != y)`, where `main` is marked with a `#[recursive]` attribute. This attribute states that the backend should generate proofs that are friendly for verification within another circuit. -- `recursive`: a circuit that verifies `main` - -For a full example of how recursive proofs work, please refer to the [noir-examples](https://github.com/noir-lang/noir-examples) repository. We will *not* be using it as a reference for this guide. - -## Step 1: Setup - -In a common NoirJS app, you need to instantiate a backend with something like `const backend = new Backend(circuit)`. Then you feed it to the `noir_js` interface. - -For recursion, this doesn't happen, and the only need for `noir_js` is only to `execute` a circuit and get its witness and return value. Everything else is not interfaced, so it needs to happen on the `backend` object. - -It is also recommended that you instantiate the backend with as many threads as possible, to allow for maximum concurrency: - -```js -const backend = new Backend(circuit, { threads: 8 }) -``` - -:::tip -You can use the [`os.cpus()`](https://nodejs.org/api/os.html#oscpus) object in `nodejs` or [`navigator.hardwareConcurrency`](https://developer.mozilla.org/en-US/docs/Web/API/Navigator/hardwareConcurrency) on the browser to make the most out of those glorious cpu cores -::: - -## Step 2: Generating the witness and the proof for `main` - -After instantiating the backend, you should also instantiate `noir_js`. We will use it to execute the circuit and get the witness. - -```js -const noir = new Noir(circuit) -const { witness } = noir.execute(input) -``` - -With this witness, you are now able to generate the intermediate proof for the main circuit: - -```js -const { proof, publicInputs } = await backend.generateProof(witness) -``` - -:::warning - -Always keep in mind what is actually happening on your development process, otherwise you'll quickly become confused about what circuit we are actually running and why! - -In this case, you can imagine that Alice (running the `main` circuit) is proving something to Bob (running the `recursive` circuit), and Bob is verifying her proof within his proof. - -With this in mind, it becomes clear that our intermediate proof is the one *meant to be verified within another circuit*, so it must be Alice's. Actually, the only final proof in this theoretical scenario would be the last one, sent on-chain. - -::: - -## Step 3 - Verification and proof artifacts - -Optionally, you are able to verify the intermediate proof: - -```js -const verified = await backend.verifyProof({ proof, publicInputs }) -``` - -This can be useful to make sure our intermediate proof was correctly generated. But the real goal is to do it within another circuit. For that, we need to generate recursive proof artifacts that will be passed to the circuit that is verifying the proof we just generated. Instead of passing the proof and verification key as a byte array, we pass them as fields which makes it cheaper to verify in a circuit: - -```js -const { proofAsFields, vkAsFields, vkHash } = await backend.generateRecursiveProofArtifacts( { publicInputs, proof }, publicInputsCount) -``` - -This call takes the public inputs and the proof, but also the public inputs count. While this is easily retrievable by simply counting the `publicInputs` length, the backend interface doesn't currently abstract it away. - -:::info - -The `proofAsFields` has a constant size `[Field; 93]` and verification keys in Barretenberg are always `[Field; 114]`. - -::: - -:::warning - -One common mistake is to forget *who* makes this call. - -In a situation where Alice is generating the `main` proof, if she generates the proof artifacts and sends them to Bob, which gladly takes them as true, this would mean Alice could prove anything! - -Instead, Bob needs to make sure *he* extracts the proof artifacts, using his own instance of the `main` circuit backend. This way, Alice has to provide a valid proof for the correct `main` circuit. - -::: - -## Step 4 - Recursive proof generation - -With the artifacts, generating a recursive proof is no different from a normal proof. You simply use the `backend` (with the recursive circuit) to generate it: - -```js -const recursiveInputs = { - verification_key: vkAsFields, // array of length 114 - proof: proofAsFields, // array of length 93 + size of public inputs - publicInputs: [mainInput.y], // using the example above, where `y` is the only public input - key_hash: vkHash, -} - -const { witness, returnValue } = noir.execute(recursiveInputs) // we're executing the recursive circuit now! -const { proof, publicInputs } = backend.generateProof(witness) -const verified = backend.verifyProof({ proof, publicInputs }) -``` - -You can obviously chain this proof into another proof. In fact, if you're using recursive proofs, you're probably interested of using them this way! - -:::tip - -Managing circuits and "who does what" can be confusing. To make sure your naming is consistent, you can keep them in an object. For example: - -```js -const circuits = { - main: mainJSON, - recursive: recursiveJSON -} -const backends = { - main: new BarretenbergBackend(circuits.main), - recursive: new BarretenbergBackend(circuits.recursive) -} -const noir_programs = { - main: new Noir(circuits.main), - recursive: new Noir(circuits.recursive) -} -``` - -This allows you to neatly call exactly the method you want without conflicting names: - -```js -// Alice runs this 👇 -const { witness: mainWitness } = await noir_programs.main.execute(input) -const proof = await backends.main.generateProof(mainWitness) - -// Bob runs this 👇 -const verified = await backends.main.verifyProof(proof) -const { proofAsFields, vkAsFields, vkHash } = await backends.main.generateRecursiveProofArtifacts( - proof, - numPublicInputs, -); -const { witness: recursiveWitness } = await noir_programs.recursive.execute(recursiveInputs) -const recursiveProof = await backends.recursive.generateProof(recursiveWitness); -``` - -::: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-solidity-verifier.md deleted file mode 100644 index 3bb96c66795..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/how-to-solidity-verifier.md +++ /dev/null @@ -1,257 +0,0 @@ ---- -title: Generate a Solidity Verifier -description: - Learn how to run the verifier as a smart contract on the blockchain. Compile a Solidity verifier - contract for your Noir program and deploy it on any EVM blockchain acting as a verifier smart - contract. Read more to find out -keywords: - [ - solidity verifier, - smart contract, - blockchain, - compiler, - plonk_vk.sol, - EVM blockchain, - verifying Noir programs, - proving backend, - Barretenberg, - ] -sidebar_position: 0 -pagination_next: tutorials/noirjs_app ---- - -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. - -This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. - -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: - -- You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network -- You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/installation/index.md) with Nargo and the example Hello Noir circuit -- You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. - -## Rundown - -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: - -1. How to generate a solidity smart contract -2. How to compile the smart contract in the RemixIDE -3. How to deploy it to a testnet - -## Step 1 - Generate a contract - -This is by far the most straightforward step. Just run: - -```sh -nargo compile -``` - -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: - -```sh -# Here we pass the path to the newly generated Noir artifact. -bb write_vk -b ./target/.json -bb contract -``` - -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. - -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. - -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: - -## Step 2 - Compiling - -We will mostly skip the details of RemixIDE, as the UI can change from version to version. For now, we can just open -Remix and create a blank workspace. - -![Create Workspace](@site/static/img/how-tos/solidity_verifier_1.png) - -We will create a new file to contain the contract Nargo generated, and copy-paste its content. - -:::warning - -You'll likely see a warning advising you to not trust pasted code. While it is an important warning, it is irrelevant in the context of this guide and can be ignored. We will not be deploying anywhere near a mainnet. - -::: - -To compile our the verifier, we can navigate to the compilation tab: - -![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) - -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: - -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) - -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: - -![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) - -## Step 3 - Deploying - -At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. - -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": - -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) - -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. - -:::note - -Why "UltraVerifier"? - -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. - -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: - -## Step 4 - Verifying - -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: - -```solidity -function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) -``` - -When using the default example in the [Hello Noir](../getting_started/hello_noir/index.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. - -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/hello_noir/index.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the - -```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') - -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS - -echo "Proof:" -echo "0x$HEX_PROOF" -``` - -Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. - -A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): - -```solidity -function castVote(bytes calldata proof, uint proposalId, uint vote, bytes32 nullifierHash) public returns (bool) { - // ... - bytes32[] memory publicInputs = new bytes32[](4); - publicInputs[0] = merkleRoot; - publicInputs[1] = bytes32(proposalId); - publicInputs[2] = bytes32(vote); - publicInputs[3] = nullifierHash; - require(verifier.verify(proof, publicInputs), "Invalid proof"); -``` - -:::info[Return Values] - -A circuit doesn't have the concept of a return value. Return values are just syntactic sugar in Noir. - -Under the hood, the return value is passed as an input to the circuit and is checked at the end of the circuit program. - -For example, if you have Noir program like this: - -```rust -fn main( - // Public inputs - pubkey_x: pub Field, - pubkey_y: pub Field, - // Private inputs - priv_key: Field, -) -> pub Field -``` - -the `verify` function will expect the public inputs array (second function parameter) to be of length 3, the two inputs and the return value. - -Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. - -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. - -::: - -:::tip[Structs] - -You can pass structs to the verifier contract. They will be flattened so that the array of inputs is 1-dimensional array. - -For example, consider the following program: - -```rust -struct Type1 { - val1: Field, - val2: Field, -} - -struct Nested { - t1: Type1, - is_true: bool, -} - -fn main(x: pub Field, nested: pub Nested, y: pub Field) { - //... -} -``` - -The order of these inputs would be flattened to: `[x, nested.t1.val1, nested.t1.val2, nested.is_true, y]` - -::: - -The other function you can call is our entrypoint `verify` function, as defined above. - -:::tip - -It's worth noticing that the `verify` function is actually a `view` function. A `view` function does not alter the blockchain state, so it doesn't need to be distributed (i.e. it will run only on the executing node), and therefore doesn't cost any gas. - -This can be particularly useful in some situations. If Alice generated a proof and wants Bob to verify its correctness, Bob doesn't need to run Nargo, NoirJS, or any Noir specific infrastructure. He can simply make a call to the blockchain with the proof and verify it is correct without paying any gas. - -It would be incorrect to say that a Noir proof verification costs any gas at all. However, most of the time the result of `verify` is used to modify state (for example, to update a balance, a game state, etc). In that case the whole network needs to execute it, which does incur gas costs (calldata and execution, but not storage). - -::: - -## A Note on EVM chains - -Noir proof verification requires the ecMul, ecAdd and ecPairing precompiles. Not all EVM chains support EC Pairings, notably some of the ZK-EVMs. This means that you won't be able to use the verifier contract in all of them. You can find an incomplete list of which EVM chains support these precompiles [here](https://www.evmdiff.com/features?feature=precompiles). - -For example, chains like `zkSync ERA` and `Polygon zkEVM` do not currently support these precompiles, so proof verification via Solidity verifier contracts won't work. Here's a quick list of EVM chains that have been tested and are known to work: - -- Optimism -- Arbitrum -- Polygon PoS -- Scroll -- Celo -- BSC -- Blast L2 -- Avalanche C-Chain -- Mode -- Linea -- Moonbeam - -If you test any other chains, please open a PR on this page to update the list. See [this doc](https://github.com/noir-lang/noir-starter/tree/main/with-foundry#testing-on-chain) for more info about testing verifier contracts on different EVM chains. - -## What's next - -Now that you know how to call a Noir Solidity Verifier on a smart contract using Remix, you should be comfortable with using it with some programmatic frameworks, such as [hardhat](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) and [foundry](https://github.com/noir-lang/noir-starter/tree/main/with-foundry). - -You can find other tools, examples, boilerplates and libraries in the [awesome-noir](https://github.com/noir-lang/awesome-noir) repository. - -You should also be ready to write and deploy your first NoirJS app and start generating proofs on websites, phones, and NodeJS environments! Head on to the [NoirJS tutorial](../tutorials/noirjs_app.md) to learn how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/merkle-proof.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/merkle-proof.mdx deleted file mode 100644 index 0a128adb2de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/merkle-proof.mdx +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Prove Merkle Tree Membership -description: - Learn how to use merkle membership proof in Noir to prove that a given leaf is a member of a - merkle tree with a specified root, at a given index. -keywords: - [merkle proof, merkle membership proof, Noir, rust, hash function, Pedersen, sha256, merkle tree] -sidebar_position: 4 ---- - -Let's walk through an example of a merkle membership proof in Noir that proves that a given leaf is -in a merkle tree. - -```rust - -fn main(message : [Field; 62], index : Field, hashpath : [Field; 40], root : Field) { - let leaf = std::hash::hash_to_field(message.as_slice()); - let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); - assert(merkle_root == root); -} - -``` - -The message is hashed using `hash_to_field`. The specific hash function that is being used is chosen -by the backend. The only requirement is that this hash function can heuristically be used as a -random oracle. If only collision resistance is needed, then one can call `std::hash::pedersen_hash` -instead. - -```rust -let leaf = std::hash::hash_to_field(message.as_slice()); -``` - -The leaf is then passed to a compute_merkle_root function with the root, index and hashpath. The returned root can then be asserted to be the same as the provided root. - -```rust -let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); -assert (merkle_root == root); -``` - -> **Note:** It is possible to re-implement the merkle tree implementation without standard library. -> However, for most usecases, it is enough. In general, the standard library will always opt to be -> as conservative as possible, while striking a balance with efficiency. - -An example, the merkle membership proof, only requires a hash function that has collision -resistance, hence a hash function like Pedersen is allowed, which in most cases is more efficient -than the even more conservative sha256. - -[View an example on the starter repo](https://github.com/noir-lang/noir-examples/blob/3ea09545cabfa464124ec2f3ea8e60c608abe6df/stealthdrop/circuits/src/main.nr#L20) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/using-devcontainers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/using-devcontainers.mdx deleted file mode 100644 index 727ec6ca667..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/how_to/using-devcontainers.mdx +++ /dev/null @@ -1,110 +0,0 @@ ---- -title: Developer Containers and Codespaces -description: "Learn how to set up a devcontainer in your GitHub repository for a seamless coding experience with Codespaces. Follow our easy 8-step guide to create your own Noir environment without installing Nargo locally." -keywords: ["Devcontainer", "Codespaces", "GitHub", "Noir Environment", "Docker Image", "Development Environment", "Remote Coding", "GitHub Codespaces", "Noir Programming", "Nargo", "VSCode Extensions", "Noirup"] -sidebar_position: 1 ---- - -Adding a developer container configuration file to your Noir project is one of the easiest way to unlock coding in browser. - -## What's a devcontainer after all? - -A [Developer Container](https://containers.dev/) (devcontainer for short) is a Docker image that comes preloaded with tools, extensions, and other tools you need to quickly get started or continue a project, without having to install Nargo locally. Think of it as a development environment in a box. - -There are many advantages to this: - -- It's platform and architecture agnostic -- You don't need to have an IDE installed, or Nargo, or use a terminal at all -- It's safer for using on a public machine or public network - -One of the best ways of using devcontainers is... not using your machine at all, for maximum control, performance, and ease of use. -Enter Codespaces. - -## Codespaces - -If a devcontainer is just a Docker image, then what stops you from provisioning a `p3dn.24xlarge` AWS EC2 instance with 92 vCPUs and 768 GiB RAM and using it to prove your 10-gate SNARK proof? - -Nothing! Except perhaps the 30-40$ per hour it will cost you. - -The problem is that provisioning takes time, and I bet you don't want to see the AWS console every time you want to code something real quick. - -Fortunately, there's an easy and free way to get a decent remote machine ready and loaded in less than 2 minutes: Codespaces. [Codespaces is a Github feature](https://github.com/features/codespaces) that allows you to code in a remote machine by using devcontainers, and it's pretty cool: - -- You can start coding Noir in less than a minute -- It uses the resources of a remote machine, so you can code on your grandma's phone if needed be -- It makes it easy to share work with your frens -- It's fully reusable, you can stop and restart whenever you need to - -:::info - -Don't take out your wallet just yet. Free GitHub accounts get about [15-60 hours of coding](https://github.com/features/codespaces) for free per month, depending on the size of your provisioned machine. - -::: - -## Tell me it's _actually_ easy - -It is! - -Github comes with a default codespace and you can use it to code your own devcontainer. That's exactly what we will be doing in this guide. - - - -8 simple steps: - -#### 1. Create a new repository on GitHub. - -#### 2. Click "Start coding with Codespaces". This will use the default image. - -#### 3. Create a folder called `.devcontainer` in the root of your repository. - -#### 4. Create a Dockerfile in that folder, and paste the following code: - -```docker -FROM --platform=linux/amd64 node:lts-bookworm-slim -SHELL ["/bin/bash", "-c"] -RUN apt update && apt install -y curl bash git tar gzip libc++-dev -RUN curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -ENV PATH="/root/.nargo/bin:$PATH" -RUN noirup -ENTRYPOINT ["nargo"] -``` -#### 5. Create a file called `devcontainer.json` in the same folder, and paste the following code: - -```json -{ - "name": "Noir on Codespaces", - "build": { - "context": ".", - "dockerfile": "Dockerfile" - }, - "customizations": { - "vscode": { - "extensions": ["noir-lang.vscode-noir"] - } - } -} -``` -#### 6. Commit and push your changes - -This will pull the new image and build it, so it could take a minute or so - -#### 8. Done! -Just wait for the build to finish, and there's your easy Noir environment. - - -Refer to [noir-starter](https://github.com/noir-lang/noir-starter/) as an example of how devcontainers can be used together with codespaces. - - - -## How do I use it? - -Using the codespace is obviously much easier than setting it up. -Just navigate to your repository and click "Code" -> "Open with Codespaces". It should take a few seconds to load, and you're ready to go. - -:::info - -If you really like the experience, you can add a badge to your readme, links to existing codespaces, and more. -Check out the [official docs](https://docs.github.com/en/codespaces/setting-up-your-project-for-codespaces/setting-up-your-repository/facilitating-quick-creation-and-resumption-of-codespaces) for more info. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/index.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/index.mdx deleted file mode 100644 index a6bd306f91d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/index.mdx +++ /dev/null @@ -1,67 +0,0 @@ ---- -title: Noir Lang -hide_title: true -description: - Learn about the public alpha release of Noir, a domain specific language heavily influenced by Rust that compiles to - an intermediate language which can be compiled to an arithmetic circuit or a rank-1 constraint system. -keywords: - [Noir, - Domain Specific Language, - Rust, - Intermediate Language, - Arithmetic Circuit, - Rank-1 Constraint System, - Ethereum Developers, - Protocol Developers, - Blockchain Developers, - Proving System, - Smart Contract Language] -sidebar_position: 0 ---- - -import Tabs from '@theme/Tabs'; -import TabItem from '@theme/TabItem'; - -Noir Logo - -Noir is an open-source Domain-Specific Language for safe and seamless construction of privacy-preserving Zero-Knowledge programs, requiring no previous knowledge on the underlying mathematics or cryptography. - -ZK programs are programs that can generate short proofs of statements without revealing all inputs to the statements. You can read more about Zero-Knowledge Proofs [here](https://dev.to/spalladino/a-beginners-intro-to-coding-zero-knowledge-proofs-c56). - -## What's new about Noir? - -Noir works differently from most ZK languages by taking a two-pronged path. First, it compiles the program to an adaptable intermediate language known as ACIR. From there, depending on a given project's needs, ACIR can be further compiled into an arithmetic circuit for integration with the proving backend. - -:::info - -Noir is backend agnostic, which means it makes no assumptions on which proving backend powers the ZK proof. Being the language that powers [Aztec Contracts](https://docs.aztec.network/developers/contracts/main), it defaults to Aztec's Barretenberg proving backend. - -However, the ACIR output can be transformed to be compatible with other PLONK-based backends, or into a [rank-1 constraint system](https://www.rareskills.io/post/rank-1-constraint-system) suitable for backends such as Arkwork's Marlin. - -::: - -## Who is Noir for? - -Noir can be used both in complex cloud-based backends and in user's smartphones, requiring no knowledge on the underlying math or cryptography. From authorization systems that keep a password in the user's device, to complex on-chain verification of recursive proofs, Noir is designed to abstract away complexity without any significant overhead. Here are some examples of situations where Noir can be used: - - - - Noir Logo - - Aztec Contracts leverage Noir to allow for the storage and execution of private information. Writing an Aztec Contract is as easy as writing Noir, and Aztec developers can easily interact with the network storage and execution through the [Aztec.nr](https://docs.aztec.network/developers/contracts/main) library. - - - Soliditry Verifier Example - Noir can auto-generate Solidity verifier contracts that verify Noir proofs. This allows for non-interactive verification of proofs containing private information in an immutable system. This feature powers a multitude of use-case scenarios, from P2P chess tournaments, to [Aztec Layer-2 Blockchain](https://docs.aztec.network/) - - - Aztec Labs developed NoirJS, an easy interface to generate and verify Noir proofs in a Javascript environment. This allows for Noir to be used in webpages, mobile apps, games, and any other environment supporting JS execution in a standalone manner. - - - - -## Libraries - -Noir is meant to be easy to extend by simply importing Noir libraries just like in Rust. -The [awesome-noir repo](https://github.com/noir-lang/awesome-noir#libraries) is a collection of libraries developed by the Noir community. -Writing a new library is easy and makes code be composable and easy to reuse. See the section on [dependencies](noir/modules_packages_crates/dependencies.md) for more information. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/migration_notes.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/migration_notes.md deleted file mode 100644 index 6bd740024e5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/migration_notes.md +++ /dev/null @@ -1,105 +0,0 @@ ---- -title: Migration notes -description: Read about migration notes from previous versions, which could solve problems while updating -keywords: [Noir, notes, migration, updating, upgrading] ---- - -Noir is in full-speed development. Things break fast, wild, and often. This page attempts to leave some notes on errors you might encounter when upgrading and how to resolve them until proper patches are built. - -### `backend encountered an error: libc++.so.1` - -Depending on your OS, you may encounter the following error when running `nargo prove` for the first time: - -```text -The backend encountered an error: "/home/codespace/.nargo/backends/acvm-backend-barretenberg/backend_binary: error while loading shared libraries: libc++.so.1: cannot open shared object file: No such file or directory\n" -``` - -Install the `libc++-dev` library with: - -```bash -sudo apt install libc++-dev -``` - -## ≥0.19 - -### Enforcing `compiler_version` - -From this version on, the compiler will check for the `compiler_version` field in `Nargo.toml`, and will error if it doesn't match the current Nargo version in use. - -To update, please make sure this field in `Nargo.toml` matches the output of `nargo --version`. - -## ≥0.14 - -The index of the [for loops](noir/concepts/control_flow.md#loops) is now of type `u64` instead of `Field`. An example refactor would be: - -```rust -for i in 0..10 { - let i = i as Field; -} -``` - -## ≥v0.11.0 and Nargo backend - -From this version onwards, Nargo starts managing backends through the `nargo backend` command. Upgrading to the versions per usual steps might lead to: - -### `backend encountered an error` - -This is likely due to the existing locally installed version of proving backend (e.g. barretenberg) is incompatible with the version of Nargo in use. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo prove -``` - -with your Noir program. - -This will trigger the download and installation of the latest version of barretenberg compatible with your Nargo in use. - -### `backend encountered an error: illegal instruction` - -On certain Intel-based systems, an `illegal instruction` error may arise due to incompatibility of barretenberg with certain CPU instructions. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo backend install acvm-backend-barretenberg https://github.com/noir-lang/barretenberg-js-binary/raw/master/run-bb.tar.gz -``` - -This downloads and installs a specific bb.js based version of barretenberg binary from GitHub. - -The gzipped file is running [this bash script](https://github.com/noir-lang/barretenberg-js-binary/blob/master/run-bb-js.sh), where we need to gzip it as the Nargo currently expect the backend to be zipped up. - -Then run: - -``` -DESIRED_BINARY_VERSION=0.8.1 nargo info -``` - -This overrides the bb native binary with a bb.js node application instead, which should be compatible with most if not all hardware. This does come with the drawback of being generally slower than native binary. - -0.8.1 indicates bb.js version 0.8.1, so if you change that it will update to a different version or the default version in the script if none was supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/_category_.json deleted file mode 100644 index 7da08f8a8c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Concepts", - "position": 0, - "collapsible": true, - "collapsed": true -} \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/assert.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/assert.md deleted file mode 100644 index 2132de42072..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/assert.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Assert Function -description: - Learn about the `assert` and `static_assert` functions in Noir, which can be used to explicitly - constrain the predicate or comparison expression that follows to be true, and what happens if - the expression is false at runtime or compile-time, respectively. -keywords: [Noir programming language, assert statement, predicate expression, comparison expression] -sidebar_position: 4 ---- - -Noir includes a special `assert` function which will explicitly constrain the predicate/comparison -expression that follows to be true. If this expression is false at runtime, the program will fail to -be proven. Example: - -```rust -fn main(x : Field, y : Field) { - assert(x == y); -} -``` - -> Assertions only work for predicate operations, such as `==`. If there's any ambiguity on the operation, the program will fail to compile. For example, it is unclear if `assert(x + y)` would check for `x + y == 0` or simply would return `true`. - -You can optionally provide a message to be logged when the assertion fails: - -```rust -assert(x == y, "x and y are not equal"); -``` - -Aside string literals, the optional message can be a format string or any other type supported as input for Noir's [print](../standard_library/logging.md) functions. This feature lets you incorporate runtime variables into your failed assertion logs: - -```rust -assert(x == y, f"Expected x == y, but got {x} == {y}"); -``` - -Using a variable as an assertion message directly: - -```rust -struct myStruct { - myField: Field -} - -let s = myStruct { myField: y }; -assert(s.myField == x, s); -``` - -There is also a special `static_assert` function that behaves like `assert`, -but that runs at compile-time. - -```rust -fn main(xs: [Field; 3]) { - let x = 2 + 2; - let y = 4; - static_assert(x == y, "expected 2 + 2 to equal 4"); - - // This passes since the length of `xs` is known at compile-time - static_assert(xs.len() == 3, "expected the input to have 3 elements"); -} -``` - -This function fails when passed a dynamic (run-time) argument: - -```rust -fn main(x : Field, y : Field) { - // this fails because `x` is not known at compile-time - static_assert(x == 2, "expected x to be known at compile-time and equal to 2"); - - let mut example_slice = &[]; - if y == 4 { - example_slice = example_slice.push_back(0); - } - - // This fails because the length of `example_slice` is not known at - // compile-time - let error_message = "expected an empty slice, known at compile-time"; - static_assert(example_slice.len() == 0, error_message); -} -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/comments.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/comments.md deleted file mode 100644 index b51a85f5c94..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/comments.md +++ /dev/null @@ -1,33 +0,0 @@ ---- -title: Comments -description: - Learn how to write comments in Noir programming language. A comment is a line of code that is - ignored by the compiler, but it can be read by programmers. Single-line and multi-line comments - are supported in Noir. -keywords: [Noir programming language, comments, single-line comments, multi-line comments] -sidebar_position: 10 ---- - -A comment is a line in your codebase which the compiler ignores, however it can be read by -programmers. - -Here is a single line comment: - -```rust -// This is a comment and is ignored -``` - -`//` is used to tell the compiler to ignore the rest of the line. - -Noir also supports multi-line block comments. Start a block comment with `/*` and end the block with `*/`. - -Noir does not natively support doc comments. You may be able to use [Rust doc comments](https://doc.rust-lang.org/reference/comments.html) in your code to leverage some Rust documentation build tools with Noir code. - -```rust -/* - This is a block comment describing a complex function. -*/ -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/comptime.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/comptime.md deleted file mode 100644 index 6f91fde9bd8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/comptime.md +++ /dev/null @@ -1,424 +0,0 @@ ---- -title: Compile-time Code & Metaprogramming -description: Learn how to use metaprogramming in Noir to create macros or derive your own traits -keywords: [Noir, comptime, compile-time, metaprogramming, macros, quote, unquote] -sidebar_position: 15 ---- - -# Overview - -Metaprogramming in Noir is comprised of three parts: -1. `comptime` code -2. Quoting and unquoting -3. The metaprogramming API in `std::meta` - -Each of these are explained in more detail in the next sections but the wide picture is that -`comptime` allows us to write code which runs at compile-time. In this `comptime` code we -can quote and unquote snippets of the program, manipulate them, and insert them in other -parts of the program. Comptime functions which do this are said to be macros. Additionally, -there's a compile-time API of built-in types and functions provided by the compiler which allows -for greater analysis and modification of programs. - ---- - -# Comptime - -`comptime` is a new keyword in Noir which marks an item as executing or existing at compile-time. It can be used in several ways: - -- `comptime fn` to define functions which execute exclusively during compile-time. -- `comptime global` to define a global variable which is evaluated at compile-time. - - Unlike runtime globals, `comptime global`s can be mutable. -- `comptime { ... }` to execute a block of statements during compile-time. -- `comptime let` to define a variable whose value is evaluated at compile-time. -- `comptime for` to run a for loop at compile-time. Syntax sugar for `comptime { for .. }`. - -## Scoping - -Note that while in a `comptime` context, any runtime variables _local to the current function_ are never visible. - -## Evaluating - -Evaluation rules of `comptime` follows the normal unconstrained evaluation rules for other Noir code. There are a few things to note though: - -- Certain built-in functions may not be available, although more may be added over time. -- Evaluation order of global items is currently unspecified. For example, given the following two functions we can't guarantee -which `println` will execute first. The ordering of the two printouts will be arbitrary, but should be stable across multiple compilations with the same `nargo` version as long as the program is also unchanged. - -```rust -fn one() { - comptime { println("one"); } -} - -fn two() { - comptime { println("two"); } -} -``` - -- Since evaluation order is unspecified, care should be taken when using mutable globals so that they do not rely on a particular ordering. -For example, using globals to generate unique ids should be fine but relying on certain ids always being produced (especially after edits to the program) should be avoided. -- Although most ordering of globals is unspecified, two are: - - Dependencies of a crate will always be evaluated before the dependent crate. - - Any annotations on a function will be run before the function itself is resolved. This is to allow the annotation to modify the function if necessary. Note that if the - function itself was called at compile-time previously, it will already be resolved and cannot be modified. To prevent accidentally calling functions you wish to modify - at compile-time, it may be helpful to sort your `comptime` annotation functions into a different crate along with any dependencies they require. - -## Lowering - -When a `comptime` value is used in runtime code it must be lowered into a runtime value. This means replacing the expression with the literal that it evaluated to. For example, the code: - -```rust -struct Foo { array: [Field; 2], len: u32 } - -fn main() { - println(comptime { - let mut foo = std::mem::zeroed::(); - foo.array[0] = 4; - foo.len = 1; - foo - }); -} -``` - -will be converted to the following after `comptime` expressions are evaluated: - -```rust -struct Foo { array: [Field; 2], len: u32 } - -fn main() { - println(Foo { array: [4, 0], len: 1 }); -} -``` - -Not all types of values can be lowered. For example, `Type`s and `TypeDefinition`s (among other types) cannot be lowered at all. - -```rust -fn main() { - // There's nothing we could inline here to create a Type value at runtime - // let _ = get_type!(); -} - -comptime fn get_type() -> Type { ... } -``` - ---- - -# (Quasi) Quote - -Macros in Noir are `comptime` functions which return code as a value which is inserted into the call site when it is lowered there. -A code value in this case is of type `Quoted` and can be created by a `quote { ... }` expression. -More specifically, the code value `quote` creates is a token stream - a representation of source code as a series of words, numbers, string literals, or operators. -For example, the expression `quote { Hi "there reader"! }` would quote three tokens: the word "hi", the string "there reader", and an exclamation mark. -You'll note that snippets that would otherwise be invalid syntax can still be quoted. - -When a `Quoted` value is used in runtime code, it is lowered into a `quote { ... }` expression. Since this expression is only valid -in compile-time code however, we'd get an error if we tried this. Instead, we can use macro insertion to insert each token into the -program at that point, and parse it as an expression. To do this, we have to add a `!` after the function name returning the `Quoted` value. -If the value was created locally and there is no function returning it, `std::meta::unquote!(_)` can be used instead. -Calling such a function at compile-time without `!` will just return the `Quoted` value to be further manipulated. For example: - -```rust title="quote-example" showLineNumbers -comptime fn quote_one() -> Quoted { - quote { 1 } - } - - #[test] - fn returning_versus_macro_insertion() { - comptime - { - // let _a: Quoted = quote { 1 }; - let _a: Quoted = quote_one(); - - // let _b: i32 = 1; - let _b: i32 = quote_one!(); - } - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L120-L136 - - -For those familiar with quoting from other languages (primarily lisps), Noir's `quote` is actually a _quasiquote_. -This means we can escape the quoting by using the unquote operator to splice values in the middle of quoted code. - -# Unquote - -The unquote operator `$` is usable within a `quote` expression. -It takes a variable as an argument, evaluates the variable, and splices the resulting value into the quoted token stream at that point. For example, - -```rust -comptime { - let x = 1 + 2; - let y = quote { $x + 4 }; -} -``` - -The value of `y` above will be the token stream containing `3`, `+`, and `4`. We can also use this to combine `Quoted` values into larger token streams: - -```rust -comptime { - let x = quote { 1 + 2 }; - let y = quote { $x + 4 }; -} -``` - -The value of `y` above is now the token stream containing five tokens: `1 + 2 + 4`. - -Note that to unquote something, a variable name _must_ follow the `$` operator in a token stream. -If it is an expression (even a parenthesized one), it will do nothing. Most likely a parse error will be given when the macro is later unquoted. - -Unquoting can also be avoided by escaping the `$` with a backslash: - -``` -comptime { - let x = quote { 1 + 2 }; - - // y contains the four tokens: `$x + 4` - let y = quote { \$x + 4 }; -} -``` - ---- - -# Annotations - -Annotations provide a way to run a `comptime` function on an item in the program. -When you use an annotation, the function with the same name will be called with that item as an argument: - -```rust -#[my_struct_annotation] -struct Foo {} - -comptime fn my_struct_annotation(s: StructDefinition) { - println("Called my_struct_annotation!"); -} - -#[my_function_annotation] -fn foo() {} - -comptime fn my_function_annotation(f: FunctionDefinition) { - println("Called my_function_annotation!"); -} -``` - -Anything returned from one of these functions will be inserted at top-level along with the original item. -Note that expressions are not valid at top-level so you'll get an error trying to return `3` or similar just as if you tried to write a program containing `3; struct Foo {}`. -You can insert other top-level items such as trait impls, structs, or functions this way though. -For example, this is the mechanism used to insert additional trait implementations into the program when deriving a trait impl from a struct: - -```rust title="derive-field-count-example" showLineNumbers -trait FieldCount { - fn field_count() -> u32; - } - - #[derive_field_count] - struct Bar { x: Field, y: [Field; 2] } - - comptime fn derive_field_count(s: StructDefinition) -> Quoted { - let typ = s.as_type(); - let field_count = s.fields().len(); - quote { - impl FieldCount for $typ { - fn field_count() -> u32 { - $field_count - } - } - } - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L138-L157 - - -## Calling annotations with additional arguments - -Arguments may optionally be given to annotations. -When this is done, these additional arguments are passed to the annotation function after the item argument. - -```rust title="annotation-arguments-example" showLineNumbers -#[assert_field_is_type(quote { i32 }.as_type())] - struct MyStruct { my_field: i32 } - - comptime fn assert_field_is_type(s: StructDefinition, typ: Type) { - // Assert the first field in `s` has type `typ` - let fields = s.fields(); - assert_eq(fields[0].1, typ); - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L159-L168 - - -We can also take any number of arguments by adding the `varargs` annotation: - -```rust title="annotation-varargs-example" showLineNumbers -#[assert_three_args(1, 2, 3)] - struct MyOtherStruct { my_other_field: u32 } - - #[varargs] - comptime fn assert_three_args(_s: StructDefinition, args: [Field]) { - assert_eq(args.len(), 3); - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L170-L178 - - ---- - -# Comptime API - -Although `comptime`, `quote`, and unquoting provide a flexible base for writing macros, -Noir's true metaprogramming ability comes from being able to interact with the compiler through a compile-time API. -This API can be accessed through built-in functions in `std::meta` as well as on methods of several `comptime` types. - -The following is an incomplete list of some `comptime` types along with some useful methods on them. - -- `Quoted`: A token stream -- `Type`: The type of a Noir type - - `fn implements(self, constraint: TraitConstraint) -> bool` - - Returns true if `self` implements the given trait constraint -- `Expr`: A syntactically valid expression. Can be used to recur on a program's parse tree to inspect how it is structured. - - Methods: - - `fn as_function_call(self) -> Option<(Expr, [Expr])>` - - If this is a function call expression, return `(function, arguments)` - - `fn as_block(self) -> Option<[Expr]>` - - If this is a block, return each statement in the block -- `FunctionDefinition`: A function definition - - Methods: - - `fn parameters(self) -> [(Quoted, Type)]` - - Returns a slice of `(name, type)` pairs for each parameter -- `StructDefinition`: A struct definition - - Methods: - - `fn as_type(self) -> Type` - - Returns this `StructDefinition` as a `Type`. Any generics are kept as-is - - `fn generics(self) -> [Quoted]` - - Return the name of each generic on this struct - - `fn fields(self) -> [(Quoted, Type)]` - - Return the name and type of each field -- `TraitConstraint`: A trait constraint such as `From` -- `TypedExpr`: A type-checked expression. -- `UnresolvedType`: A syntactic notation that refers to a Noir type that hasn't been resolved yet - -There are many more functions available by exploring the `std::meta` module and its submodules. -Using these methods is the key to writing powerful metaprogramming libraries. - ---- - -# Example: Derive - -Using all of the above, we can write a `derive` macro that behaves similarly to Rust's but is not built into the language. -From the user's perspective it will look like this: - -```rust -// Example usage -#[derive(Default, Eq, Ord)] -struct MyStruct { my_field: u32 } -``` - -To implement `derive` we'll have to create a `comptime` function that accepts -a variable amount of traits. - -```rust title="derive_example" showLineNumbers -// These are needed for the unconstrained hashmap we're using to store derive functions -use crate::collections::umap::UHashMap; -use crate::hash::BuildHasherDefault; -use crate::hash::poseidon2::Poseidon2Hasher; - -// A derive function is one that given a struct definition can -// create us a quoted trait impl from it. -type DeriveFunction = fn(StructDefinition) -> Quoted; - -// We'll keep a global HANDLERS map to keep track of the derive handler for each trait -comptime mut global HANDLERS: UHashMap> = UHashMap::default(); - -// Given a struct and a slice of traits to derive, create trait impls for each. -// This function is as simple as iterating over the slice, checking if we have a trait -// handler registered for the given trait, calling it, and appending the result. -#[varargs] -pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted { - let mut result = quote {}; - - for trait_to_derive in traits { - let handler = unsafe { - HANDLERS.get(trait_to_derive) - }; - assert(handler.is_some(), f"No derive function registered for `{trait_to_derive}`"); - - let trait_impl = handler.unwrap()(s); - result = quote { $result $trait_impl }; - } - - result -} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L30-L64 - - -Registering a derive function could be done as follows: - -```rust title="derive_via" showLineNumbers -// To register a handler for a trait, just add it to our handlers map -pub comptime fn derive_via(t: TraitDefinition, f: DeriveFunction) { - HANDLERS.insert(t, f); -} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L66-L73 - - -```rust title="big-derive-usage-example" showLineNumbers -// Finally, to register a handler we call the above function as an annotation - // with our handler function. - #[derive_via(derive_do_nothing)] - trait DoNothing { - fn do_nothing(self); - } - - comptime fn derive_do_nothing(s: StructDefinition) -> Quoted { - // This is simplified since we don't handle generics or where clauses! - // In a real example we'd likely also need to introduce each of - // `s.generics()` as well as a trait constraint for each generic - // to ensure they also implement the trait. - let typ = s.as_type(); - quote { - impl DoNothing for $typ { - fn do_nothing(self) { - // Traits can't tell us what to do - println("something"); - } - } - } - } - - // Since `DoNothing` is a simple trait which: - // 1. Only has one method - // 2. Does not have any generics on the trait itself - // We can use `std::meta::make_trait_impl` to help us out. - // This helper function will generate our impl for us along with any - // necessary where clauses and still provides a flexible interface - // for us to work on each field on the struct. - comptime fn derive_do_nothing_alt(s: StructDefinition) -> Quoted { - let trait_name = quote { DoNothing }; - let method_signature = quote { fn do_nothing(self) }; - - // Call `do_nothing` recursively on each field in the struct - let for_each_field = |field_name| quote { self.$field_name.do_nothing(); }; - - // Some traits like Eq want to join each field expression with something like `&`. - // We don't need that here - let join_fields_with = quote {}; - - // The body function is a spot to insert any extra setup/teardown needed. - // We'll insert our println here. Since we recur on each field, we should see - // one println for the struct itself, followed by a println for every field (recursively). - let body = |body| quote { - println("something"); - $body - }; - crate::meta::make_trait_impl( - s, - trait_name, - method_signature, - for_each_field, - join_fields_with, - body - ) - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L180-L238 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/control_flow.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/control_flow.md deleted file mode 100644 index 045d3c3a5f5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/control_flow.md +++ /dev/null @@ -1,77 +0,0 @@ ---- -title: Control Flow -description: - Learn how to use loops and if expressions in the Noir programming language. Discover the syntax - and examples for for loops and if-else statements. -keywords: [Noir programming language, loops, for loop, if-else statements, Rust syntax] -sidebar_position: 2 ---- - -## If Expressions - -Noir supports `if-else` statements. The syntax is most similar to Rust's where it is not required -for the statement's conditional to be surrounded by parentheses. - -```rust -let a = 0; -let mut x: u32 = 0; - -if a == 0 { - if a != 0 { - x = 6; - } else { - x = 2; - } -} else { - x = 5; - assert(x == 5); -} -assert(x == 2); -``` - -## Loops - -Noir has one kind of loop: the `for` loop. `for` loops allow you to repeat a block of code multiple -times. - -The following block of code between the braces is run 10 times. - -```rust -for i in 0..10 { - // do something -} -``` - -The index for loops is of type `u64`. - -### Break and Continue - -In unconstrained code, `break` and `continue` are also allowed in `for` loops. These are only allowed -in unconstrained code since normal constrained code requires that Noir knows exactly how many iterations -a loop may have. `break` and `continue` can be used like so: - -```rust -for i in 0 .. 10 { - println("Iteration start") - - if i == 2 { - continue; - } - - if i == 5 { - break; - } - - println(i); -} -println("Loop end") -``` - -When used, `break` will end the current loop early and jump to the statement after the for loop. In the example -above, the `break` will stop the loop and jump to the `println("Loop end")`. - -`continue` will stop the current iteration of the loop, and jump to the start of the next iteration. In the example -above, `continue` will jump to `println("Iteration start")` when used. Note that the loop continues as normal after this. -The iteration variable `i` is still increased by one as normal when `continue` is used. - -`break` and `continue` cannot currently be used to jump out of more than a single loop at a time. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_bus.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_bus.mdx deleted file mode 100644 index e55e58622ce..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_bus.mdx +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: Data Bus -sidebar_position: 13 ---- -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -The data bus is an optimization that the backend can use to make recursion more efficient. -In order to use it, you must define some inputs of the program entry points (usually the `main()` -function) with the `call_data` modifier, and the return values with the `return_data` modifier. -These modifiers are incompatible with `pub` and `mut` modifiers. - -## Example - -```rust -fn main(mut x: u32, y: call_data u32, z: call_data [u32;4] ) -> return_data u32 { - let a = z[x]; - a+y -} -``` - -As a result, both call_data and return_data will be treated as private inputs and encapsulated into a read-only array each, for the backend to process. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/arrays.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/arrays.md deleted file mode 100644 index d0d8eb70aa6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/arrays.md +++ /dev/null @@ -1,275 +0,0 @@ ---- -title: Arrays -description: - Dive into the Array data type in Noir. Grasp its methods, practical examples, and best practices for efficiently using Arrays in your Noir code. -keywords: - [ - noir, - array type, - methods, - examples, - indexing, - ] -sidebar_position: 4 ---- - -An array is one way of grouping together values into one compound type. Array types can be inferred -or explicitly specified via the syntax `[; ]`: - -```rust -fn main(x : Field, y : Field) { - let my_arr = [x, y]; - let your_arr: [Field; 2] = [x, y]; -} -``` - -Here, both `my_arr` and `your_arr` are instantiated as an array containing two `Field` elements. - -Array elements can be accessed using indexing: - -```rust -fn main() { - let a = [1, 2, 3, 4, 5]; - - let first = a[0]; - let second = a[1]; -} -``` - -All elements in an array must be of the same type (i.e. homogeneous). That is, an array cannot group -a `Field` value and a `u8` value together for example. - -You can write mutable arrays, like: - -```rust -fn main() { - let mut arr = [1, 2, 3, 4, 5]; - assert(arr[0] == 1); - - arr[0] = 42; - assert(arr[0] == 42); -} -``` - -You can instantiate a new array of a fixed size with the same value repeated for each element. The following example instantiates an array of length 32 where each element is of type Field and has the value 0. - -```rust -let array: [Field; 32] = [0; 32]; -``` - -Like in Rust, arrays in Noir are a fixed size. However, if you wish to convert an array to a [slice](./slices.mdx), you can just call `as_slice` on your array: - -```rust -let array: [Field; 32] = [0; 32]; -let sl = array.as_slice() -``` - -You can define multidimensional arrays: - -```rust -let array : [[Field; 2]; 2]; -let element = array[0][0]; -``` - -However, multidimensional slices are not supported. For example, the following code will error at compile time: - -```rust -let slice : [[Field]] = &[]; -``` - -## Types - -You can create arrays of primitive types or structs. There is not yet support for nested arrays -(arrays of arrays) or arrays of structs that contain arrays. - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for arrays. -Each of these functions are located within the generic impl `impl [T; N] {`. -So anywhere `self` appears, it refers to the variable `self: [T; N]`. - -### len - -Returns the length of an array - -```rust -fn len(self) -> Field -``` - -example - -```rust -fn main() { - let array = [42, 42]; - assert(array.len() == 2); -} -``` - -### sort - -Returns a new sorted array. The original array remains untouched. Notice that this function will -only work for arrays of fields or integers, not for any arbitrary type. This is because the sorting -logic it uses internally is optimized specifically for these values. If you need a sort function to -sort any type, you should use the function `sort_via` described below. - -```rust -fn sort(self) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32]; - let sorted = arr.sort(); - assert(sorted == [32, 42]); -} -``` - -### sort_via - -Sorts the array with a custom comparison function. The ordering function must return true if the first argument should be sorted to be before the second argument or is equal to the second argument. - -Using this method with an operator like `<` that does not return `true` for equal values will result in an assertion failure for arrays with equal elements. - -```rust -fn sort_via(self, ordering: fn(T, T) -> bool) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32] - let sorted_ascending = arr.sort_via(|a, b| a <= b); - assert(sorted_ascending == [32, 42]); // verifies - - let sorted_descending = arr.sort_via(|a, b| a >= b); - assert(sorted_descending == [32, 42]); // does not verify -} -``` - -### map - -Applies a function to each element of the array, returning a new array containing the mapped elements. - -```rust -fn map(self, f: fn(T) -> U) -> [U; N] -``` - -example - -```rust -let a = [1, 2, 3]; -let b = a.map(|a| a * 2); // b is now [2, 4, 6] -``` - -### fold - -Applies a function to each element of the array, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn(U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the array, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = [1]; -let a2 = [1, 2]; -let a3 = [1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let arr = [2, 2, 2, 2, 2]; - let folded = arr.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn(T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let reduced = arr.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let all = arr.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 5]; - let any = arr.any(|a| a == 5); - assert(any); -} - -``` - -### as_str_unchecked - -Converts a byte array of type `[u8; N]` to a string. Note that this performs no UTF-8 validation - -the given array is interpreted as-is as a string. - -```rust -impl [u8; N] { - pub fn as_str_unchecked(self) -> str -} -``` - -example: - -```rust -fn main() { - let hi = [104, 105].as_str_unchecked(); - assert_eq(hi, "hi"); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/booleans.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/booleans.md deleted file mode 100644 index 2507af710e7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/booleans.md +++ /dev/null @@ -1,28 +0,0 @@ ---- -title: Booleans -description: - Delve into the Boolean data type in Noir. Understand its methods, practical examples, and best practices for using Booleans in your Noir programs. -keywords: - [ - noir, - boolean type, - methods, - examples, - logical operations, - ] -sidebar_position: 2 ---- - - -The `bool` type in Noir has two possible values: `true` and `false`: - -```rust -fn main() { - let t = true; - let f: bool = false; -} -``` - -The boolean type is most commonly used in conditionals like `if` expressions and `assert` -statements. More about conditionals is covered in the [Control Flow](../control_flow.md) and -[Assert Function](../assert.md) sections. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/fields.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/fields.md deleted file mode 100644 index 23367a32451..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/fields.md +++ /dev/null @@ -1,240 +0,0 @@ ---- -title: Fields -description: - Dive deep into the Field data type in Noir. Understand its methods, practical examples, and best practices to effectively use Fields in your Noir programs. -keywords: - [ - noir, - field type, - methods, - examples, - best practices, - ] -sidebar_position: 0 ---- - -The field type corresponds to the native field type of the proving backend. - -The size of a Noir field depends on the elliptic curve's finite field for the proving backend -adopted. For example, a field would be a 254-bit integer when paired with the default backend that -spans the Grumpkin curve. - -Fields support integer arithmetic and are often used as the default numeric type in Noir: - -```rust -fn main(x : Field, y : Field) { - let z = x + y; -} -``` - -`x`, `y` and `z` are all private fields in this example. Using the `let` keyword we defined a new -private value `z` constrained to be equal to `x + y`. - -If proving efficiency is of priority, fields should be used as a default for solving problems. -Smaller integer types (e.g. `u64`) incur extra range constraints. - -## Methods - -After declaring a Field, you can use these common methods on it: - -### to_le_bits - -Transforms the field into an array of bits, Little Endian. - -```rust title="to_le_bits" showLineNumbers -pub fn to_le_bits(self: Self) -> [u1; N] {} -``` -> Source code: noir_stdlib/src/field/mod.nr#L32-L34 - - -example: - -```rust title="to_le_bits_example" showLineNumbers -fn test_to_le_bits() { - let field = 2; - let bits: [u1; 8] = field.to_le_bits(); - assert_eq(bits, [0, 1, 0, 0, 0, 0, 0, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L196-L202 - - - -### to_be_bits - -Transforms the field into an array of bits, Big Endian. - -```rust title="to_be_bits" showLineNumbers -pub fn to_be_bits(self: Self) -> [u1; N] {} -``` -> Source code: noir_stdlib/src/field/mod.nr#L48-L50 - - -example: - -```rust title="to_be_bits_example" showLineNumbers -fn test_to_be_bits() { - let field = 2; - let bits: [u1; 8] = field.to_be_bits(); - assert_eq(bits, [0, 0, 0, 0, 0, 0, 1, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L187-L193 - - - -### to_le_bytes - -Transforms into an array of bytes, Little Endian - -```rust title="to_le_bytes" showLineNumbers -pub fn to_le_bytes(self: Self) -> [u8; N] { - self.to_le_radix(256) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L63-L67 - - -example: - -```rust title="to_le_bytes_example" showLineNumbers -fn test_to_le_bytes() { - let field = 2; - let bits: [u8; 8] = field.to_le_bytes(); - assert_eq(bits, [2, 0, 0, 0, 0, 0, 0, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L214-L220 - - -### to_be_bytes - -Transforms into an array of bytes, Big Endian - -```rust title="to_be_bytes" showLineNumbers -pub fn to_be_bytes(self: Self) -> [u8; N] { - self.to_be_radix(256) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L80-L84 - - -example: - -```rust title="to_be_bytes_example" showLineNumbers -fn test_to_be_bytes() { - let field = 2; - let bits: [u8; 8] = field.to_be_bytes(); - assert_eq(bits, [0, 0, 0, 0, 0, 0, 0, 2]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L205-L211 - - - -### to_le_radix - -Decomposes into an array over the specified base, Little Endian - -```rust title="to_le_radix" showLineNumbers -pub fn to_le_radix(self: Self, radix: u32) -> [u8; N] { - crate::assert_constant(radix); - self.__to_le_radix(radix) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L86-L91 - - - -example: - -```rust title="to_le_radix_example" showLineNumbers -fn test_to_le_radix() { - let field = 2; - let bits: [u8; 8] = field.to_le_radix(256); - assert_eq(bits, [2, 0, 0, 0, 0, 0, 0, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L232-L238 - - - -### to_be_radix - -Decomposes into an array over the specified base, Big Endian - -```rust title="to_be_radix" showLineNumbers -pub fn to_be_radix(self: Self, radix: u32) -> [u8; N] { - crate::assert_constant(radix); - self.__to_be_radix(radix) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L93-L98 - - -example: - -```rust title="to_be_radix_example" showLineNumbers -fn test_to_be_radix() { - let field = 2; - let bits: [u8; 8] = field.to_be_radix(256); - assert_eq(bits, [0, 0, 0, 0, 0, 0, 0, 2]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L223-L229 - - - -### pow_32 - -Returns the value to the power of the specified exponent - -```rust -fn pow_32(self, exponent: Field) -> Field -``` - -example: - -```rust -fn main() { - let field = 2 - let pow = field.pow_32(4); - assert(pow == 16); -} -``` - -### assert_max_bit_size - -Adds a constraint to specify that the field can be represented with `bit_size` number of bits - -```rust title="assert_max_bit_size" showLineNumbers -pub fn assert_max_bit_size(self, bit_size: u32) { -``` -> Source code: noir_stdlib/src/field/mod.nr#L9-L11 - - -example: - -```rust -fn main() { - let field = 2 - field.assert_max_bit_size(32); -} -``` - -### sgn0 - -Parity of (prime) Field element, i.e. sgn0(x mod p) = 0 if x ∈ \{0, ..., p-1\} is even, otherwise sgn0(x mod p) = 1. - -```rust -fn sgn0(self) -> u1 -``` - - -### lt - -Returns true if the field is less than the other field - -```rust -pub fn lt(self, another: Field) -> bool -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/function_types.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/function_types.md deleted file mode 100644 index f6121af17e2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/function_types.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Function types -sidebar_position: 10 ---- - -Noir supports higher-order functions. The syntax for a function type is as follows: - -```rust -fn(arg1_type, arg2_type, ...) -> return_type -``` - -Example: - -```rust -fn assert_returns_100(f: fn() -> Field) { // f takes no args and returns a Field - assert(f() == 100); -} - -fn main() { - assert_returns_100(|| 100); // ok - assert_returns_100(|| 150); // fails -} -``` - -A function type also has an optional capture environment - this is necessary to support closures. -See [Lambdas](../lambdas.md) for more details. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/index.md deleted file mode 100644 index 3eadb2dc8a4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/index.md +++ /dev/null @@ -1,118 +0,0 @@ ---- -title: Data Types -description: - Get a clear understanding of the two categories of Noir data types - primitive types and compound - types. Learn about their characteristics, differences, and how to use them in your Noir - programming. -keywords: - [ - noir, - data types, - primitive types, - compound types, - private types, - public types, - ] ---- - -Every value in Noir has a type, which determines which operations are valid for it. - -All values in Noir are fundamentally composed of `Field` elements. For a more approachable -developing experience, abstractions are added on top to introduce different data types in Noir. - -Noir has two category of data types: primitive types (e.g. `Field`, integers, `bool`) and compound -types that group primitive types (e.g. arrays, tuples, structs). Each value can either be private or -public. - -## Private & Public Types - -A **private value** is known only to the Prover, while a **public value** is known by both the -Prover and Verifier. Mark values as `private` when the value should only be known to the prover. All -primitive types (including individual fields of compound types) in Noir are private by default, and -can be marked public when certain values are intended to be revealed to the Verifier. - -> **Note:** For public values defined in Noir programs paired with smart contract verifiers, once -> the proofs are verified on-chain the values can be considered known to everyone that has access to -> that blockchain. - -Public data types are treated no differently to private types apart from the fact that their values -will be revealed in proofs generated. Simply changing the value of a public type will not change the -circuit (where the same goes for changing values of private types as well). - -_Private values_ are also referred to as _witnesses_ sometimes. - -> **Note:** The terms private and public when applied to a type (e.g. `pub Field`) have a different -> meaning than when applied to a function (e.g. `pub fn foo() {}`). -> -> The former is a visibility modifier for the Prover to interpret if a value should be made known to -> the Verifier, while the latter is a visibility modifier for the compiler to interpret if a -> function should be made accessible to external Noir programs like in other languages. - -### pub Modifier - -All data types in Noir are private by default. Types are explicitly declared as public using the -`pub` modifier: - -```rust -fn main(x : Field, y : pub Field) -> pub Field { - x + y -} -``` - -In this example, `x` is **private** while `y` and `x + y` (the return value) are **public**. Note -that visibility is handled **per variable**, so it is perfectly valid to have one input that is -private and another that is public. - -> **Note:** Public types can only be declared through parameters on `main`. - -## Type Aliases - -A type alias is a new name for an existing type. Type aliases are declared with the keyword `type`: - -```rust -type Id = u8; - -fn main() { - let id: Id = 1; - let zero: u8 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can also be used with [generics](../generics.md): - -```rust -type Id = Size; - -fn main() { - let id: Id = 1; - let zero: u32 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can even refer to other aliases. An error will be issued if they form a cycle: - -```rust -// Ok! -type A = B; -type B = Field; - -type Bad1 = Bad2; - -// error: Dependency cycle found -type Bad2 = Bad1; -// ^^^^^^^^^^^ 'Bad2' recursively depends on itself: Bad2 -> Bad1 -> Bad2 -``` - -## Wildcard Type -Noir can usually infer the type of the variable from the context, so specifying the type of a variable is only required when it cannot be inferred. However, specifying a complex type can be tedious, especially when it has multiple generic arguments. Often some of the generic types can be inferred from the context, and Noir only needs a hint to properly infer the other types. We can partially specify a variable's type by using `_` as a marker, indicating where we still want the compiler to infer the type. - -```rust -let a: [_; 4] = foo(b); -``` - - -### BigInt - -You can achieve BigInt functionality using the [Noir BigInt](https://github.com/shuklaayush/noir-bigint) library. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/integers.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/integers.md deleted file mode 100644 index a1d59bf3166..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/integers.md +++ /dev/null @@ -1,156 +0,0 @@ ---- -title: Integers -description: Explore the Integer data type in Noir. Learn about its methods, see real-world examples, and grasp how to efficiently use Integers in your Noir code. -keywords: [noir, integer types, methods, examples, arithmetic] -sidebar_position: 1 ---- - -An integer type is a range constrained field type. -The Noir frontend supports both unsigned and signed integer types. -The allowed sizes are 1, 8, 16, 32 and 64 bits. - -:::info - -When an integer is defined in Noir without a specific type, it will default to `Field`. - -The one exception is for loop indices which default to `u64` since comparisons on `Field`s are not possible. - -::: - -## Unsigned Integers - -An unsigned integer type is specified first with the letter `u` (indicating its unsigned nature) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: u8 = 1; - let y: u8 = 1; - let z = x + y; - assert (z == 2); -} -``` - -The bit size determines the maximum value the integer type can store. For example, a `u8` variable can store a value in the range of 0 to 255 (i.e. $\\2^{8}-1\\$). - -## Signed Integers - -A signed integer type is specified first with the letter `i` (which stands for integer) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: i8 = -1; - let y: i8 = -1; - let z = x + y; - assert (z == -2); -} -``` - -The bit size determines the maximum and minimum range of value the integer type can store. For example, an `i8` variable can store a value in the range of -128 to 127 (i.e. $\\-2^{7}\\$ to $\\2^{7}-1\\$). - -## 128 bits Unsigned Integers - -The built-in structure `U128` allows you to use 128-bit unsigned integers almost like a native integer type. However, there are some differences to keep in mind: -- You cannot cast between a native integer and `U128` -- There is a higher performance cost when using `U128`, compared to a native type. - -Conversion between unsigned integer types and U128 are done through the use of `from_integer` and `to_integer` functions. `from_integer` also accepts the `Field` type as input. - -```rust -fn main() { - let x = U128::from_integer(23); - let y = U128::from_hex("0x7"); - let z = x + y; - assert(z.to_integer() == 30); -} -``` - -`U128` is implemented with two 64 bits limbs, representing the low and high bits, which explains the performance cost. You should expect `U128` to be twice more costly for addition and four times more costly for multiplication. -You can construct a U128 from its limbs: -```rust -fn main(x: u64, y: u64) { - let x = U128::from_u64s_be(x,y); - assert(z.hi == x as Field); - assert(z.lo == y as Field); -} -``` - -Note that the limbs are stored as Field elements in order to avoid unnecessary conversions. -Apart from this, most operations will work as usual: - -```rust -fn main(x: U128, y: U128) { - // multiplication - let c = x * y; - // addition and subtraction - let c = c - x + y; - // division - let c = x / y; - // bit operation; - let c = x & y | y; - // bit shift - let c = x << y; - // comparisons; - let c = x < y; - let c = x == y; -} -``` - -## Overflows - -Computations that exceed the type boundaries will result in overflow errors. This happens with both signed and unsigned integers. For example, attempting to prove: - -```rust -fn main(x: u8, y: u8) { - let z = x + y; -} -``` - -With: - -```toml -x = "255" -y = "1" -``` - -Would result in: - -``` -$ nargo execute -error: Assertion failed: 'attempt to add with overflow' -┌─ ~/src/main.nr:9:13 -│ -│ let z = x + y; -│ ----- -│ -= Call stack: - ... -``` - -A similar error would happen with signed integers: - -```rust -fn main() { - let x: i8 = -118; - let y: i8 = -11; - let z = x + y; -} -``` - -### Wrapping methods - -Although integer overflow is expected to error, some use-cases rely on wrapping. For these use-cases, the standard library provides `wrapping` variants of certain common operations: - -```rust -fn wrapping_add(x: T, y: T) -> T; -fn wrapping_sub(x: T, y: T) -> T; -fn wrapping_mul(x: T, y: T) -> T; -``` - -Example of how it is used: - -```rust - -fn main(x: u8, y: u8) -> pub u8 { - std::wrapping_add(x, y) -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/references.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/references.md deleted file mode 100644 index a5293d11cfb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/references.md +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: References -sidebar_position: 9 ---- - -Noir supports first-class references. References are a bit like pointers: they point to a specific address that can be followed to access the data stored at that address. You can use Rust-like syntax to use pointers in Noir: the `&` operator references the variable, the `*` operator dereferences it. - -Example: - -```rust -fn main() { - let mut x = 2; - - // you can reference x as &mut and pass it to multiplyBy2 - multiplyBy2(&mut x); -} - -// you can access &mut here -fn multiplyBy2(x: &mut Field) { - // and dereference it with * - *x = *x * 2; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/slices.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/slices.mdx deleted file mode 100644 index a0c87c29259..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/slices.mdx +++ /dev/null @@ -1,358 +0,0 @@ ---- -title: Slices -description: Explore the Slice data type in Noir. Understand its methods, see real-world examples, and learn how to effectively use Slices in your Noir programs. -keywords: [noir, slice type, methods, examples, subarrays] -sidebar_position: 5 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A slice is a dynamically-sized view into a sequence of elements. They can be resized at runtime, but because they don't own the data, they cannot be returned from a circuit. You can treat slices as arrays without a constrained size. - -```rust -fn main() -> pub u32 { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -To write a slice literal, use a preceding ampersand as in: `&[0; 2]` or -`&[1, 2, 3]`. - -It is important to note that slices are not references to arrays. In Noir, -`&[..]` is more similar to an immutable, growable vector. - -View the corresponding test file [here][test-file]. - -[test-file]: https://github.com/noir-lang/noir/blob/f387ec1475129732f72ba294877efdf6857135ac/crates/nargo_cli/tests/test_data_ssa_refactor/slices/src/main.nr - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for slices: - -### push_back - -Pushes a new element to the end of the slice, returning a new slice with a length one greater than the original unmodified slice. - -```rust -fn push_back(_self: [T], _elem: T) -> [T] -``` - -example: - -```rust -fn main() -> pub Field { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -View the corresponding test file [here][test-file]. - -### push_front - -Returns a new array with the specified element inserted at index 0. The existing elements indexes are incremented by 1. - -```rust -fn push_front(_self: Self, _elem: T) -> Self -``` - -Example: - -```rust -let mut new_slice: [Field] = &[]; -new_slice = new_slice.push_front(20); -assert(new_slice[0] == 20); // returns true -``` - -View the corresponding test file [here][test-file]. - -### pop_front - -Returns a tuple of two items, the first element of the array and the rest of the array. - -```rust -fn pop_front(_self: Self) -> (T, Self) -``` - -Example: - -```rust -let (first_elem, rest_of_slice) = slice.pop_front(); -``` - -View the corresponding test file [here][test-file]. - -### pop_back - -Returns a tuple of two items, the beginning of the array with the last element omitted and the last element. - -```rust -fn pop_back(_self: Self) -> (Self, T) -``` - -Example: - -```rust -let (popped_slice, last_elem) = slice.pop_back(); -``` - -View the corresponding test file [here][test-file]. - -### append - -Loops over a slice and adds it to the end of another. - -```rust -fn append(mut self, other: Self) -> Self -``` - -Example: - -```rust -let append = &[1, 2].append(&[3, 4, 5]); -``` - -### insert - -Inserts an element at a specified index and shifts all following elements by 1. - -```rust -fn insert(_self: Self, _index: Field, _elem: T) -> Self -``` - -Example: - -```rust -new_slice = rest_of_slice.insert(2, 100); -assert(new_slice[2] == 100); -``` - -View the corresponding test file [here][test-file]. - -### remove - -Remove an element at a specified index, shifting all elements after it to the left, returning the altered slice and the removed element. - -```rust -fn remove(_self: Self, _index: Field) -> (Self, T) -``` - -Example: - -```rust -let (remove_slice, removed_elem) = slice.remove(3); -``` - -### len - -Returns the length of a slice - -```rust -fn len(self) -> Field -``` - -Example: - -```rust -fn main() { - let slice = &[42, 42]; - assert(slice.len() == 2); -} -``` - -### as_array - -Converts this slice into an array. - -Make sure to specify the size of the resulting array. -Panics if the resulting array length is different than the slice's length. - -```rust -fn as_array(self) -> [T; N] -``` - -Example: - -```rust -fn main() { - let slice = &[5, 6]; - - // Always specify the length of the resulting array! - let array: [Field; 2] = slice.as_array(); - - assert(array[0] == slice[0]); - assert(array[1] == slice[1]); -} -``` - -### map - -Applies a function to each element of the slice, returning a new slice containing the mapped elements. - -```rust -fn map(self, f: fn[Env](T) -> U) -> [U] -``` - -example - -```rust -let a = &[1, 2, 3]; -let b = a.map(|a| a * 2); // b is now &[2, 4, 6] -``` - -### fold - -Applies a function to each element of the slice, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn[Env](U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the slice, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = &[1]; -let a2 = &[1, 2]; -let a3 = &[1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let folded = slice.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn[Env](T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let reduced = slice.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### filter - -Returns a new slice containing only elements for which the given predicate returns true. - -```rust -fn filter(self, f: fn[Env](T) -> bool) -> Self -``` - -example: - -```rust -fn main() { - let slice = &[1, 2, 3, 4, 5]; - let odds = slice.filter(|x| x % 2 == 1); - assert_eq(odds, &[1, 3, 5]); -} -``` - -### join - -Flatten each element in the slice into one value, separated by `separator`. - -Note that although slices implement `Append`, `join` cannot be used on slice -elements since nested slices are prohibited. - -```rust -fn join(self, separator: T) -> T where T: Append -``` - -example: - -```rust -struct Accumulator { - total: Field, -} - -// "Append" two accumulators by adding them -impl Append for Accumulator { - fn empty() -> Self { - Self { total: 0 } - } - - fn append(self, other: Self) -> Self { - Self { total: self.total + other.total } - } -} - -fn main() { - let slice = &[1, 2, 3, 4, 5].map(|total| Accumulator { total }); - - let result = slice.join(Accumulator::empty()); - assert_eq(result, Accumulator { total: 15 }); - - // We can use a non-empty separator to insert additional elements to sum: - let separator = Accumulator { total: 10 }; - let result = slice.join(separator); - assert_eq(result, Accumulator { total: 55 }); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let all = slice.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 5]; - let any = slice.any(|a| a == 5); - assert(any); -} - -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/structs.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/structs.md deleted file mode 100644 index dbf68c99813..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/structs.md +++ /dev/null @@ -1,70 +0,0 @@ ---- -title: Structs -description: - Explore the Struct data type in Noir. Learn about its methods, see real-world examples, and grasp how to effectively define and use Structs in your Noir programs. -keywords: - [ - noir, - struct type, - methods, - examples, - data structures, - ] -sidebar_position: 8 ---- - -A struct also allows for grouping multiple values of different types. Unlike tuples, we can also -name each field. - -> **Note:** The usage of _field_ here refers to each element of the struct and is unrelated to the -> field type of Noir. - -Defining a struct requires giving it a name and listing each field within as `: ` pairs: - -```rust -struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` - -An instance of a struct can then be created with actual values in `: ` pairs in any -order. Struct fields are accessible using their given names: - -```rust -fn main() { - let legs = 4; - - let dog = Animal { - eyes: 2, - hands: 0, - legs, - }; - - let zero = dog.hands; -} -``` - -Structs can also be destructured in a pattern, binding each field to a new variable: - -```rust -fn main() { - let Animal { hands, legs: feet, eyes } = get_octopus(); - - let ten = hands + feet + eyes as u8; -} - -fn get_octopus() -> Animal { - let octopus = Animal { - hands: 0, - legs: 8, - eyes: 2, - }; - - octopus -} -``` - -The new variables can be bound with names different from the original struct field names, as -showcased in the `legs --> feet` binding in the example above. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/tuples.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/tuples.md deleted file mode 100644 index 2ec5c9c4113..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/tuples.md +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Tuples -description: - Dive into the Tuple data type in Noir. Understand its methods, practical examples, and best practices for efficiently using Tuples in your Noir code. -keywords: - [ - noir, - tuple type, - methods, - examples, - multi-value containers, - ] -sidebar_position: 7 ---- - -A tuple collects multiple values like an array, but with the added ability to collect values of -different types: - -```rust -fn main() { - let tup: (u8, u64, Field) = (255, 500, 1000); -} -``` - -One way to access tuple elements is via destructuring using pattern matching: - -```rust -fn main() { - let tup = (1, 2); - - let (one, two) = tup; - - let three = one + two; -} -``` - -Another way to access tuple elements is via direct member access, using a period (`.`) followed by -the index of the element we want to access. Index `0` corresponds to the first tuple element, `1` to -the second and so on: - -```rust -fn main() { - let tup = (5, 6, 7, 8); - - let five = tup.0; - let eight = tup.3; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/functions.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/functions.md deleted file mode 100644 index f656cdfd97a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/functions.md +++ /dev/null @@ -1,226 +0,0 @@ ---- -title: Functions -description: - Learn how to declare functions and methods in Noir, a programming language with Rust semantics. - This guide covers parameter declaration, return types, call expressions, and more. -keywords: [Noir, Rust, functions, methods, parameter declaration, return types, call expressions] -sidebar_position: 1 ---- - -Functions in Noir follow the same semantics of Rust, though Noir does not support early returns. - -To declare a function the `fn` keyword is used. - -```rust -fn foo() {} -``` - -By default, functions are visible only within the package they are defined. To make them visible outside of that package (for example, as part of a [library](../modules_packages_crates/crates_and_packages.md#libraries)), you should mark them as `pub`: - -```rust -pub fn foo() {} -``` - -You can also restrict the visibility of the function to only the crate it was defined in, by specifying `pub(crate)`: - -```rust -pub(crate) fn foo() {} //foo can only be called within its crate -``` - -All parameters in a function must have a type and all types are known at compile time. The parameter -is pre-pended with a colon and the parameter type. Multiple parameters are separated using a comma. - -```rust -fn foo(x : Field, y : Field){} -``` - -The return type of a function can be stated by using the `->` arrow notation. The function below -states that the foo function must return a `Field`. If the function returns no value, then the arrow -is omitted. - -```rust -fn foo(x : Field, y : Field) -> Field { - x + y -} -``` - -Note that a `return` keyword is unneeded in this case - the last expression in a function's body is -returned. - -## Main function - -If you're writing a binary, the `main` function is the starting point of your program. You can pass all types of expressions to it, as long as they have a fixed size at compile time: - -```rust -fn main(x : Field) // this is fine: passing a Field -fn main(x : [Field; 2]) // this is also fine: passing a Field with known size at compile-time -fn main(x : (Field, bool)) // 👌: passing a (Field, bool) tuple means size 2 -fn main(x : str<5>) // this is fine, as long as you pass a string of size 5 - -fn main(x : Vec) // can't compile, has variable size -fn main(x : [Field]) // can't compile, has variable size -fn main(....// i think you got it by now -``` - -Keep in mind [tests](../../tooling/testing.md) don't differentiate between `main` and any other function. The following snippet passes tests, but won't compile or prove: - -```rust -fn main(x : [Field]) { - assert(x[0] == 1); -} - -#[test] -fn test_one() { - main(&[1, 2]); -} -``` - -```bash -$ nargo test -[testing] Running 1 test functions -[testing] Testing test_one... ok -[testing] All tests passed - -$ nargo check -The application panicked (crashed). -Message: Cannot have variable sized arrays as a parameter to main -``` - -## Call Expressions - -Calling a function in Noir is executed by using the function name and passing in the necessary -arguments. - -Below we show how to call the `foo` function from the `main` function using a call expression: - -```rust -fn main(x : Field, y : Field) { - let z = foo(x); -} - -fn foo(x : Field) -> Field { - x + x -} -``` - -## Methods - -You can define methods in Noir on any struct type in scope. - -```rust -struct MyStruct { - foo: Field, - bar: Field, -} - -impl MyStruct { - fn new(foo: Field) -> MyStruct { - MyStruct { - foo, - bar: 2, - } - } - - fn sum(self) -> Field { - self.foo + self.bar - } -} - -fn main() { - let s = MyStruct::new(40); - assert(s.sum() == 42); -} -``` - -Methods are just syntactic sugar for functions, so if we wanted to we could also call `sum` as -follows: - -```rust -assert(MyStruct::sum(s) == 42); -``` - -It is also possible to specialize which method is chosen depending on the [generic](./generics.md) type that is used. In this example, the `foo` function returns different values depending on its type: - -```rust -struct Foo {} - -impl Foo { - fn foo(self) -> Field { 1 } -} - -impl Foo { - fn foo(self) -> Field { 2 } -} - -fn main() { - let f1: Foo = Foo{}; - let f2: Foo = Foo{}; - assert(f1.foo() + f2.foo() == 3); -} -``` - -Also note that impls with the same method name defined in them cannot overlap. For example, if we already have `foo` defined for `Foo` and `Foo` like we do above, we cannot also define `foo` in an `impl Foo` since it would be ambiguous which version of `foo` to choose. - -```rust -// Including this impl in the same project as the above snippet would -// cause an overlapping impls error -impl Foo { - fn foo(self) -> Field { 3 } -} -``` - -## Lambdas - -Lambdas are anonymous functions. They follow the syntax of Rust - `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -See [Lambdas](./lambdas.md) for more details. - -## Attributes - -Attributes are metadata that can be applied to a function, using the following syntax: `#[attribute(value)]`. - -Supported attributes include: - -- **builtin**: the function is implemented by the compiler, for efficiency purposes. -- **deprecated**: mark the function as _deprecated_. Calling the function will generate a warning: `warning: use of deprecated function` -- **field**: Used to enable conditional compilation of code depending on the field size. See below for more details -- **oracle**: mark the function as _oracle_; meaning it is an external unconstrained function, implemented in noir_js. See [Unconstrained](./unconstrained.md) and [NoirJS](../../reference/NoirJS/noir_js/index.md) for more details. -- **test**: mark the function as unit tests. See [Tests](../../tooling/testing.md) for more details - -### Field Attribute - -The field attribute defines which field the function is compatible for. The function is conditionally compiled, under the condition that the field attribute matches the Noir native field. -The field can be defined implicitly, by using the name of the elliptic curve usually associated to it - for instance bn254, bls12_381 - or explicitly by using the field (prime) order, in decimal or hexadecimal form. -As a result, it is possible to define multiple versions of a function with each version specialized for a different field attribute. This can be useful when a function requires different parameters depending on the underlying elliptic curve. - -Example: we define the function `foo()` three times below. Once for the default Noir bn254 curve, once for the field $\mathbb F_{23}$, which will normally never be used by Noir, and once again for the bls12_381 curve. - -```rust -#[field(bn254)] -fn foo() -> u32 { - 1 -} - -#[field(23)] -fn foo() -> u32 { - 2 -} - -// This commented code would not compile as foo would be defined twice because it is the same field as bn254 -// #[field(21888242871839275222246405745257275088548364400416034343698204186575808495617)] -// fn foo() -> u32 { -// 2 -// } - -#[field(bls12_381)] -fn foo() -> u32 { - 3 -} -``` - -If the field name is not known to Noir, it will discard the function. Field names are case insensitive. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/generics.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/generics.md deleted file mode 100644 index 1f313cef58e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/generics.md +++ /dev/null @@ -1,258 +0,0 @@ ---- -title: Generics -description: Learn how to use Generics in Noir -keywords: [Noir, Rust, generics, functions, structs] -sidebar_position: 7 ---- - -Generics allow you to use the same functions with multiple different concrete data types. You can -read more about the concept of generics in the Rust documentation -[here](https://doc.rust-lang.org/book/ch10-01-syntax.html). - -Here is a trivial example showing the identity function that supports any type. In Rust, it is -common to refer to the most general type as `T`. We follow the same convention in Noir. - -```rust -fn id(x: T) -> T { - x -} -``` - -## In Structs - -Generics are useful for specifying types in structs. For example, we can specify that a field in a -struct will be of a certain generic type. In this case `value` is of type `T`. - -```rust -struct RepeatedValue { - value: T, - count: Field, -} - -impl RepeatedValue { - fn print(self) { - for _i in 0 .. self.count { - println(self.value); - } - } -} - -fn main() { - let repeated = RepeatedValue { value: "Hello!", count: 2 }; - repeated.print(); -} -``` - -The `print` function will print `Hello!` an arbitrary number of times, twice in this case. - -## Numeric Generics - -If we want to be generic over array lengths (which are type-level integers), we can use numeric -generics. Using these looks similar to using regular generics, but introducing them into scope -requires declaring them with `let MyGenericName: IntegerType`. This can be done anywhere a normal -generic is declared. Instead of types, these generics resolve to integers at compile-time. -Here's an example of a struct that is generic over the size of the array it contains internally: - -```rust -struct BigInt { - limbs: [u32; N], -} - -impl BigInt { - // `N` is in scope of all methods in the impl - fn first(first: BigInt, second: BigInt) -> Self { - assert(first.limbs != second.limbs); - first - - fn second(first: BigInt, second: Self) -> Self { - assert(first.limbs != second.limbs); - second - } -} -``` - -## Calling functions on generic parameters - -Since a generic type `T` can represent any type, how can we call functions on the underlying type? -In other words, how can we go from "any type `T`" to "any type `T` that has certain methods available?" - -This is what [traits](../concepts/traits.md) are for in Noir. Here's an example of a function generic over -any type `T` that implements the `Eq` trait for equality: - -```rust -fn first_element_is_equal(array1: [T; N], array2: [T; N]) -> bool - where T: Eq -{ - if (array1.len() == 0) | (array2.len() == 0) { - true - } else { - array1[0] == array2[0] - } -} - -fn main() { - assert(first_element_is_equal([1, 2, 3], [1, 5, 6])); - - // We can use first_element_is_equal for arrays of any type - // as long as we have an Eq impl for the types we pass in - let array = [MyStruct::new(), MyStruct::new()]; - assert(array_eq(array, array, MyStruct::eq)); -} - -impl Eq for MyStruct { - fn eq(self, other: MyStruct) -> bool { - self.foo == other.foo - } -} -``` - -You can find more details on traits and trait implementations on the [traits page](../concepts/traits.md). - -## Manually Specifying Generics with the Turbofish Operator - -There are times when the compiler cannot reasonably infer what type should be used for a generic, or when the developer themselves may want to manually distinguish generic type parameters. This is where the `::<>` turbofish operator comes into play. - -The `::<>` operator can follow a variable or path and can be used to manually specify generic arguments within the angle brackets. -The name "turbofish" comes from that `::<>` looks like a little fish. - -Examples: -```rust -fn main() { - let mut slice = []; - slice = slice.push_back(1); - slice = slice.push_back(2); - // Without turbofish a type annotation would be needed on the left hand side - let array = slice.as_array::<2>(); -} -``` -```rust -fn double() -> u32 { - N * 2 -} -fn example() { - assert(double::<9>() == 18); - assert(double::<7 + 8>() == 30); -} -``` -```rust -trait MyTrait { - fn ten() -> Self; -} - -impl MyTrait for Field { - fn ten() -> Self { 10 } -} - -struct Foo { - inner: T -} - -impl Foo { - fn generic_method(_self: Self) -> U where U: MyTrait { - U::ten() - } -} - -fn example() { - let foo: Foo = Foo { inner: 1 }; - // Using a type other than `Field` here (e.g. u32) would fail as - // there is no matching impl for `u32: MyTrait`. - // - // Substituting the `10` on the left hand side of this assert - // with `10 as u32` would also fail with a type mismatch as we - // are expecting a `Field` from the right hand side. - assert(10 as u32 == foo.generic_method::()); -} -``` - -## Arithmetic Generics - -In addition to numeric generics, Noir also allows a limited form of arithmetic on generics. -When you have a numeric generic such as `N`, you can use the following operators on it in a -type position: `+`, `-`, `*`, `/`, and `%`. - -Note that type checking arithmetic generics is a best effort guess from the compiler and there -are many cases of types that are equal that the compiler may not see as such. For example, -we know that `T * (N + M)` should be equal to `T*N + T*M` but the compiler does not currently -apply the distributive law and thus sees these as different types. - -Even with this limitation though, the compiler can handle common cases decently well: - -```rust -trait Serialize { - fn serialize(self) -> [Field; N]; -} - -impl Serialize<1> for Field { - fn serialize(self) -> [Field; 1] { - [self] - } -} - -impl Serialize for [T; N] - where T: Serialize { .. } - -impl Serialize for (T, U) - where T: Serialize, U: Serialize { .. } - -fn main() { - let data = (1, [2, 3, 4]); - assert_eq(data.serialize().len(), 4); -} -``` - -Note that if there is any over or underflow the types will fail to unify: - -```rust title="underflow-example" showLineNumbers -fn pop(array: [Field; N]) -> [Field; N - 1] { - let mut result: [Field; N - 1] = std::mem::zeroed(); - for i in 0..N - 1 { - result[i] = array[i]; - } - result -} - -fn main() { - // error: Could not determine array length `(0 - 1)` - pop([]); -} -``` -> Source code: test_programs/compile_failure/arithmetic_generics_underflow/src/main.nr#L1-L14 - - -This also applies if there is underflow in an intermediate calculation: - -```rust title="intermediate-underflow-example" showLineNumbers -fn main() { - // From main it looks like there's nothing sketchy going on - seems_fine([]); -} - -// Since `seems_fine` says it can receive and return any length N -fn seems_fine(array: [Field; N]) -> [Field; N] { - // But inside `seems_fine` we pop from the array which - // requires the length to be greater than zero. - - // error: Could not determine array length `(0 - 1)` - push_zero(pop(array)) -} - -fn pop(array: [Field; N]) -> [Field; N - 1] { - let mut result: [Field; N - 1] = std::mem::zeroed(); - for i in 0..N - 1 { - result[i] = array[i]; - } - result -} - -fn push_zero(array: [Field; N]) -> [Field; N + 1] { - let mut result: [Field; N + 1] = std::mem::zeroed(); - for i in 0..N { - result[i] = array[i]; - } - // index N is already zeroed - result -} -``` -> Source code: test_programs/compile_failure/arithmetic_generics_intermediate_underflow/src/main.nr#L1-L32 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/globals.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/globals.md deleted file mode 100644 index 063a3d89248..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/globals.md +++ /dev/null @@ -1,72 +0,0 @@ ---- -title: Global Variables -description: - Learn about global variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, globals, global variables, constants] -sidebar_position: 8 ---- - -## Globals - - -Noir supports global variables. The global's type can be inferred by the compiler entirely: - -```rust -global N = 5; // Same as `global N: Field = 5` - -global TUPLE = (3, 2); - -fn main() { - assert(N == 5); - assert(N == TUPLE.0 + TUPLE.1); -} -``` - -:::info - -Globals can be defined as any expression, so long as they don't depend on themselves - otherwise there would be a dependency cycle! For example: - -```rust -global T = foo(T); // dependency error -``` - -::: - - -If they are initialized to a literal integer, globals can be used to specify an array's length: - -```rust -global N: Field = 2; - -fn main(y : [Field; N]) { - assert(y[0] == y[1]) -} -``` - -A global from another module can be imported or referenced externally like any other name: - -```rust -global N = 20; - -fn main() { - assert(my_submodule::N != N); -} - -mod my_submodule { - global N: Field = 10; -} -``` - -When a global is used, Noir replaces the name with its definition on each occurrence. -This means globals defined using function calls will repeat the call each time they're used: - -```rust -global RESULT = foo(); - -fn foo() -> [Field; 100] { ... } -``` - -This is usually fine since Noir will generally optimize any function call that does not -refer to a program input into a constant. It should be kept in mind however, if the called -function performs side-effects like `println`, as these will still occur on each use. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/lambdas.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/lambdas.md deleted file mode 100644 index be3c7e0b5ca..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/lambdas.md +++ /dev/null @@ -1,81 +0,0 @@ ---- -title: Lambdas -description: Learn how to use anonymous functions in Noir programming language. -keywords: [Noir programming language, lambda, closure, function, anonymous function] -sidebar_position: 9 ---- - -## Introduction - -Lambdas are anonymous functions. The syntax is `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -A block can be used as the body of a lambda, allowing you to declare local variables inside it: - -```rust -let cool = || { - let x = 100; - let y = 100; - x + y -} - -assert(cool() == 200); -``` - -## Closures - -Inside the body of a lambda, you can use variables defined in the enclosing function. Such lambdas are called **closures**. In this example `x` is defined inside `main` and is accessed from within the lambda: - -```rust -fn main() { - let x = 100; - let closure = || x + 150; - assert(closure() == 250); -} -``` - -## Passing closures to higher-order functions - -It may catch you by surprise that the following code fails to compile: - -```rust -fn foo(f: fn () -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // error :( -} -``` - -The reason is that the closure's capture environment affects its type - we have a closure that captures two Fields and `foo` -expects a regular function as an argument - those are incompatible. -:::note - -Variables contained within the `||` are the closure's parameters, and the expression that follows it is the closure's body. The capture environment is comprised of any variables used in the closure's body that are not parameters. - -E.g. in |x| x + y, y would be a captured variable, but x would not be, since it is a parameter of the closure. - -::: -The syntax for the type of a closure is `fn[env](args) -> ret_type`, where `env` is the capture environment of the closure - -in this example that's `(Field, Field)`. - -The best solution in our case is to make `foo` generic over the environment type of its parameter, so that it can be called -with closures with any environment, as well as with regular functions: - -```rust -fn foo(f: fn[Env]() -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // compiles fine - assert(foo(|| 60) == 60); // compiles fine -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/mutability.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/mutability.md deleted file mode 100644 index fdeef6a87c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/mutability.md +++ /dev/null @@ -1,121 +0,0 @@ ---- -title: Mutability -description: - Learn about mutable variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, mutability in noir, mutable variables] -sidebar_position: 8 ---- - -Variables in noir can be declared mutable via the `mut` keyword. Mutable variables can be reassigned -to via an assignment expression. - -```rust -let x = 2; -x = 3; // error: x must be mutable to be assigned to - -let mut y = 3; -let y = 4; // OK -``` - -The `mut` modifier can also apply to patterns: - -```rust -let (a, mut b) = (1, 2); -a = 11; // error: a must be mutable to be assigned to -b = 12; // OK - -let mut (c, d) = (3, 4); -c = 13; // OK -d = 14; // OK - -// etc. -let MyStruct { x: mut y } = MyStruct { x: a }; -// y is now in scope -``` - -Note that mutability in noir is local and everything is passed by value, so if a called function -mutates its parameters then the parent function will keep the old value of the parameters. - -```rust -fn main() -> pub Field { - let x = 3; - helper(x); - x // x is still 3 -} - -fn helper(mut x: i32) { - x = 4; -} -``` - -## Non-local mutability - -Non-local mutability can be achieved through the mutable reference type `&mut T`: - -```rust -fn set_to_zero(x: &mut Field) { - *x = 0; -} - -fn main() { - let mut y = 42; - set_to_zero(&mut y); - assert(*y == 0); -} -``` - -When creating a mutable reference, the original variable being referred to (`y` in this -example) must also be mutable. Since mutable references are a reference type, they must -be explicitly dereferenced via `*` to retrieve the underlying value. Note that this yields -a copy of the value, so mutating this copy will not change the original value behind the -reference: - -```rust -fn main() { - let mut x = 1; - let x_ref = &mut x; - - let mut y = *x_ref; - let y_ref = &mut y; - - x = 2; - *x_ref = 3; - - y = 4; - *y_ref = 5; - - assert(x == 3); - assert(*x_ref == 3); - assert(y == 5); - assert(*y_ref == 5); -} -``` - -Note that types in Noir are actually deeply immutable so the copy that occurs when -dereferencing is only a conceptual copy - no additional constraints will occur. - -Mutable references can also be stored within structs. Note that there is also -no lifetime parameter on these unlike rust. This is because the allocated memory -always lasts the entire program - as if it were an array of one element. - -```rust -struct Foo { - x: &mut Field -} - -impl Foo { - fn incr(mut self) { - *self.x += 1; - } -} - -fn main() { - let foo = Foo { x: &mut 0 }; - foo.incr(); - assert(*foo.x == 1); -} -``` - -In general, you should avoid non-local & shared mutability unless it is needed. Sticking -to only local mutability will improve readability and potentially improve compiler optimizations as well. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/ops.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/ops.md deleted file mode 100644 index c35c36c38a9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/ops.md +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: Logical Operations -description: - Learn about the supported arithmetic and logical operations in the Noir programming language. - Discover how to perform operations on private input types, integers, and booleans. -keywords: - [ - Noir programming language, - supported operations, - arithmetic operations, - logical operations, - predicate operators, - bitwise operations, - short-circuiting, - backend, - ] -sidebar_position: 3 ---- - -# Operations - -## Table of Supported Operations - -| Operation | Description | Requirements | -| :-------- | :------------------------------------------------------------: | -------------------------------------: | -| + | Adds two private input types together | Types must be private input | -| - | Subtracts two private input types together | Types must be private input | -| \* | Multiplies two private input types together | Types must be private input | -| / | Divides two private input types together | Types must be private input | -| ^ | XOR two private input types together | Types must be integer | -| & | AND two private input types together | Types must be integer | -| \| | OR two private input types together | Types must be integer | -| \<\< | Left shift an integer by another integer amount | Types must be integer, shift must be u8 | -| >> | Right shift an integer by another integer amount | Types must be integer, shift must be u8 | -| ! | Bitwise not of a value | Type must be integer or boolean | -| \< | returns a bool if one value is less than the other | Upper bound must have a known bit size | -| \<= | returns a bool if one value is less than or equal to the other | Upper bound must have a known bit size | -| > | returns a bool if one value is more than the other | Upper bound must have a known bit size | -| >= | returns a bool if one value is more than or equal to the other | Upper bound must have a known bit size | -| == | returns a bool if one value is equal to the other | Both types must not be constants | -| != | returns a bool if one value is not equal to the other | Both types must not be constants | - -### Predicate Operators - -`<,<=, !=, == , >, >=` are known as predicate/comparison operations because they compare two values. -This differs from the operations such as `+` where the operands are used in _computation_. - -### Bitwise Operations Example - -```rust -fn main(x : Field) { - let y = x as u32; - let z = y & y; -} -``` - -`z` is implicitly constrained to be the result of `y & y`. The `&` operand is used to denote bitwise -`&`. - -> `x & x` would not compile as `x` is a `Field` and not an integer type. - -### Logical Operators - -Noir has no support for the logical operators `||` and `&&`. This is because encoding the -short-circuiting that these operators require can be inefficient for Noir's backend. Instead you can -use the bitwise operators `|` and `&` which operate identically for booleans, just without the -short-circuiting. - -```rust -let my_val = 5; - -let mut flag = 1; -if (my_val > 6) | (my_val == 0) { - flag = 0; -} -assert(flag == 1); - -if (my_val != 10) & (my_val < 50) { - flag = 0; -} -assert(flag == 0); -``` - -### Shorthand operators - -Noir shorthand operators for most of the above operators, namely `+=, -=, *=, /=, %=, &=, |=, ^=, <<=`, and `>>=`. These allow for more concise syntax. For example: - -```rust -let mut i = 0; -i = i + 1; -``` - -could be written as: - -```rust -let mut i = 0; -i += 1; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/oracles.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/oracles.mdx deleted file mode 100644 index 77a2ac1550a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/oracles.mdx +++ /dev/null @@ -1,29 +0,0 @@ ---- -title: Oracles -description: Dive into how Noir supports Oracles via RPC calls, and learn how to declare an Oracle in Noir with our comprehensive guide. -keywords: - - Noir - - Oracles - - RPC Calls - - Unconstrained Functions - - Programming - - Blockchain -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -Noir has support for Oracles via RPC calls. This means Noir will make an RPC call and use the return value for proof generation. - -Since Oracles are not resolved by Noir, they are [`unconstrained` functions](./unconstrained.md) - -You can declare an Oracle through the `#[oracle()]` flag. Example: - -```rust -#[oracle(get_number_sequence)] -unconstrained fn get_number_sequence(_size: Field) -> [Field] {} -``` - -The timeout for when using an external RPC oracle resolver can be set with the `NARGO_FOREIGN_CALL_TIMEOUT` environment variable. This timeout is in units of milliseconds. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/shadowing.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/shadowing.md deleted file mode 100644 index 5ce6130d201..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/shadowing.md +++ /dev/null @@ -1,44 +0,0 @@ ---- -title: Shadowing -sidebar_position: 12 ---- - -Noir allows for inheriting variables' values and re-declaring them with the same name similar to Rust, known as shadowing. - -For example, the following function is valid in Noir: - -```rust -fn main() { - let x = 5; - - { - let x = x * 2; - assert (x == 10); - } - - assert (x == 5); -} -``` - -In this example, a variable x is first defined with the value 5. - -The local scope that follows shadows the original x, i.e. creates a local mutable x based on the value of the original x. It is given a value of 2 times the original x. - -When we return to the main scope, x once again refers to just the original x, which stays at the value of 5. - -## Temporal mutability - -One way that shadowing is useful, in addition to ergonomics across scopes, is for temporarily mutating variables. - -```rust -fn main() { - let age = 30; - // age = age + 5; // Would error as `age` is immutable by default. - - let mut age = age + 5; // Temporarily mutates `age` with a new value. - - let age = age; // Locks `age`'s mutability again. - - assert (age == 35); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/traits.md deleted file mode 100644 index 597c62c737c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/traits.md +++ /dev/null @@ -1,465 +0,0 @@ ---- -title: Traits -description: - Traits in Noir can be used to abstract out a common interface for functions across - several data types. -keywords: [noir programming language, traits, interfaces, generic, protocol] -sidebar_position: 14 ---- - -## Overview - -Traits in Noir are a useful abstraction similar to interfaces or protocols in other languages. Each trait defines -the interface of several methods contained within the trait. Types can then implement this trait by providing -implementations for these methods. For example in the program: - -```rust -struct Rectangle { - width: Field, - height: Field, -} - -impl Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -fn log_area(r: Rectangle) { - println(r.area()); -} -``` - -We have a function `log_area` to log the area of a `Rectangle`. Now how should we change the program if we want this -function to work on `Triangle`s as well?: - -```rust -struct Triangle { - width: Field, - height: Field, -} - -impl Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Making `log_area` generic over all types `T` would be invalid since not all types have an `area` method. Instead, we can -introduce a new `Area` trait and make `log_area` generic over all types `T` that implement `Area`: - -```rust -trait Area { - fn area(self) -> Field; -} - -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -We also need to explicitly implement `Area` for `Rectangle` and `Triangle`. We can do that by changing their existing -impls slightly. Note that the parameter types and return type of each of our `area` methods must match those defined -by the `Area` trait. - -```rust -impl Area for Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -impl Area for Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Now we have a working program that is generic over any type of Shape that is used! Others can even use this program -as a library with their own types - such as `Circle` - as long as they also implement `Area` for these types. - -## Where Clauses - -As seen in `log_area` above, when we want to create a function or method that is generic over any type that implements -a trait, we can add a where clause to the generic function. - -```rust -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -It is also possible to apply multiple trait constraints on the same variable at once by combining traits with the `+` -operator. Similarly, we can have multiple trait constraints by separating each with a comma: - -```rust -fn foo(elements: [T], thing: U) where - T: Default + Add + Eq, - U: Bar, -{ - let mut sum = T::default(); - - for element in elements { - sum += element; - } - - if sum == T::default() { - thing.bar(); - } -} -``` - -## Generic Implementations - -You can add generics to a trait implementation by adding the generic list after the `impl` keyword: - -```rust -trait Second { - fn second(self) -> Field; -} - -impl Second for (T, Field) { - fn second(self) -> Field { - self.1 - } -} -``` - -You can also implement a trait for every type this way: - -```rust -trait Debug { - fn debug(self); -} - -impl Debug for T { - fn debug(self) { - println(self); - } -} - -fn main() { - 1.debug(); -} -``` - -### Generic Trait Implementations With Where Clauses - -Where clauses can be placed on trait implementations themselves to restrict generics in a similar way. -For example, while `impl Foo for T` implements the trait `Foo` for every type, `impl Foo for T where T: Bar` -will implement `Foo` only for types that also implement `Bar`. This is often used for implementing generic types. -For example, here is the implementation for array equality: - -```rust -impl Eq for [T; N] where T: Eq { - // Test if two arrays have the same elements. - // Because both arrays must have length N, we know their lengths already match. - fn eq(self, other: Self) -> bool { - let mut result = true; - - for i in 0 .. self.len() { - // The T: Eq constraint is needed to call == on the array elements here - result &= self[i] == other[i]; - } - - result - } -} -``` - -Where clauses can also be placed on struct implementations. -For example, here is a method utilizing a generic type that implements the equality trait. - -```rust -struct Foo { - a: u32, - b: T, -} - -impl Foo where T: Eq { - fn eq(self, other: Self) -> bool { - (self.a == other.a) & self.b.eq(other.b) - } -} -``` - -## Generic Traits - -Traits themselves can also be generic by placing the generic arguments after the trait name. These generics are in -scope of every item within the trait. - -```rust -trait Into { - // Convert `self` to type `T` - fn into(self) -> T; -} -``` - -When implementing generic traits the generic arguments of the trait must be specified. This is also true anytime -when referencing a generic trait (e.g. in a `where` clause). - -```rust -struct MyStruct { - array: [Field; 2], -} - -impl Into<[Field; 2]> for MyStruct { - fn into(self) -> [Field; 2] { - self.array - } -} - -fn as_array(x: T) -> [Field; 2] - where T: Into<[Field; 2]> -{ - x.into() -} - -fn main() { - let array = [1, 2]; - let my_struct = MyStruct { array }; - - assert_eq(as_array(my_struct), array); -} -``` - -### Associated Types and Constants - -Traits also support associated types and constraints which can be thought of as additional generics that are referred to by name. - -Here's an example of a trait with an associated type `Foo` and a constant `Bar`: - -```rust -trait MyTrait { - type Foo; - - let Bar: u32; -} -``` - -Now when we're implementing `MyTrait` we also have to provide values for `Foo` and `Bar`: - -```rust -impl MyTrait for Field { - type Foo = i32; - - let Bar: u32 = 11; -} -``` - -Since associated constants can also be used in a type position, its values are limited to only other -expression kinds allowed in numeric generics. - -Note that currently all associated types and constants must be explicitly specified in a trait constraint. -If we leave out any, we'll get an error that we're missing one: - -```rust -// Error! Constraint is missing associated constant for `Bar` -fn foo(x: T) where T: MyTrait { - ... -} -``` - -Because all associated types and constants must be explicitly specified, they are essentially named generics, -although this is set to change in the future. Future versions of Noir will allow users to elide associated types -in trait constraints similar to Rust. When this is done, you may still refer to their value with the `::AssociatedType` -syntax: - -```rust -// Only valid in future versions of Noir: -fn foo(x: T) where T: MyTrait { - let _: ::Foo = ...; -} -``` - -The type as trait syntax is possible in Noir today but is less useful when each type must be explicitly specified anyway: - -```rust -fn foo(x: T) where T: MyTrait { - // Works, but could just use F directly - let _: >::Foo = ...; - - let _: F = ...; -} -``` - -## Trait Methods With No `self` - -A trait can contain any number of methods, each of which have access to the `Self` type which represents each type -that eventually implements the trait. Similarly, the `self` variable is available as well but is not required to be used. -For example, we can define a trait to create a default value for a type. This trait will need to return the `Self` type -but doesn't need to take any parameters: - -```rust -trait Default { - fn default() -> Self; -} -``` - -Implementing this trait can be done similarly to any other trait: - -```rust -impl Default for Field { - fn default() -> Field { - 0 - } -} - -struct MyType {} - -impl Default for MyType { - fn default() -> Field { - MyType {} - } -} -``` - -However, since there is no `self` parameter, we cannot call it via the method call syntax `object.method()`. -Instead, we'll need to refer to the function directly. This can be done either by referring to the -specific impl `MyType::default()` or referring to the trait itself `Default::default()`. In the later -case, type inference determines the impl that is selected. - -```rust -let my_struct = MyStruct::default(); - -let x: Field = Default::default(); -let result = x + Default::default(); -``` - -:::warning - -```rust -let _ = Default::default(); -``` - -If type inference cannot select which impl to use because of an ambiguous `Self` type, an impl will be -arbitrarily selected. This occurs most often when the result of a trait function call with no parameters -is unused. To avoid this, when calling a trait function with no `self` or `Self` parameters or return type, -always refer to it via the implementation type's namespace - e.g. `MyType::default()`. -This is set to change to an error in future Noir versions. - -::: - -## Default Method Implementations - -A trait can also have default implementations of its methods by giving a body to the desired functions. -Note that this body must be valid for all types that may implement the trait. As a result, the only -valid operations on `self` will be operations valid for any type or other operations on the trait itself. - -```rust -trait Numeric { - fn add(self, other: Self) -> Self; - - // Default implementation of double is (self + self) - fn double(self) -> Self { - self.add(self) - } -} -``` - -When implementing a trait with default functions, a type may choose to implement only the required functions: - -```rust -impl Numeric for Field { - fn add(self, other: Field) -> Field { - self + other - } -} -``` - -Or it may implement the optional methods as well: - -```rust -impl Numeric for u32 { - fn add(self, other: u32) -> u32 { - self + other - } - - fn double(self) -> u32 { - self * 2 - } -} -``` - -## Impl Specialization - -When implementing traits for a generic type it is possible to implement the trait for only a certain combination -of generics. This can be either as an optimization or because those specific generics are required to implement the trait. - -```rust -trait Sub { - fn sub(self, other: Self) -> Self; -} - -struct NonZero { - value: T, -} - -impl Sub for NonZero { - fn sub(self, other: Self) -> Self { - let value = self.value - other.value; - assert(value != 0); - NonZero { value } - } -} -``` - -## Overlapping Implementations - -Overlapping implementations are disallowed by Noir to ensure Noir's decision on which impl to select is never ambiguous. -This means if a trait `Foo` is already implemented -by a type `Bar` for all `T`, then we cannot also have a separate impl for `Bar` (or any other -type argument). Similarly, if there is an impl for all `T` such as `impl Debug for T`, we cannot create -any more impls to `Debug` for other types since it would be ambiguous which impl to choose for any given -method call. - -```rust -trait Trait {} - -// Previous impl defined here -impl Trait for (A, B) {} - -// error: Impl for type `(Field, Field)` overlaps with existing impl -impl Trait for (Field, Field) {} -``` - -## Trait Coherence - -Another restriction on trait implementations is coherence. This restriction ensures other crates cannot create -impls that may overlap with other impls, even if several unrelated crates are used as dependencies in the same -program. - -The coherence restriction is: to implement a trait, either the trait itself or the object type must be declared -in the crate the impl is in. - -In practice this often comes up when using types provided by libraries. If a library provides a type `Foo` that does -not implement a trait in the standard library such as `Default`, you may not `impl Default for Foo` in your own crate. -While restrictive, this prevents later issues or silent changes in the program if the `Foo` library later added its -own impl for `Default`. If you are a user of the `Foo` library in this scenario and need a trait not implemented by the -library your choices are to either submit a patch to the library or use the newtype pattern. - -### The Newtype Pattern - -The newtype pattern gets around the coherence restriction by creating a new wrapper type around the library type -that we cannot create `impl`s for. Since the new wrapper type is defined in our current crate, we can create -impls for any trait we need on it. - -```rust -struct Wrapper { - foo: some_library::Foo, -} - -impl Default for Wrapper { - fn default() -> Wrapper { - Wrapper { - foo: some_library::Foo::new(), - } - } -} -``` - -Since we have an impl for our own type, the behavior of this code will not change even if `some_library` is updated -to provide its own `impl Default for Foo`. The downside of this pattern is that it requires extra wrapping and -unwrapping of values when converting to and from the `Wrapper` and `Foo` types. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/_category_.json deleted file mode 100644 index 1debcfe7675..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Modules, Packages and Crates", - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/crates_and_packages.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/crates_and_packages.md deleted file mode 100644 index 95ee9f52ab2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/crates_and_packages.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Crates and Packages -description: Learn how to use Crates and Packages in your Noir project -keywords: [Nargo, dependencies, package management, crates, package] -sidebar_position: 0 ---- - -## Crates - -A crate is the smallest amount of code that the Noir compiler considers at a time. -Crates can contain modules, and the modules may be defined in other files that get compiled with the crate, as we’ll see in the coming sections. - -### Crate Types - -A Noir crate can come in several forms: binaries, libraries or contracts. - -#### Binaries - -_Binary crates_ are programs which you can compile to an ACIR circuit which you can then create proofs against. Each must have a function called `main` that defines the ACIR circuit which is to be proved. - -#### Libraries - -_Library crates_ don't have a `main` function and they don't compile down to ACIR. Instead they define functionality intended to be shared with multiple projects, and eventually included in a binary crate. - -#### Contracts - -Contract crates are similar to binary crates in that they compile to ACIR which you can create proofs against. They are different in that they do not have a single `main` function, but are a collection of functions to be deployed to the [Aztec network](https://aztec.network). You can learn more about the technical details of Aztec in the [monorepo](https://github.com/AztecProtocol/aztec-packages) or contract [examples](https://github.com/AztecProtocol/aztec-packages/tree/master/noir-projects/noir-contracts/contracts). - -### Crate Root - -Every crate has a root, which is the source file that the compiler starts, this is also known as the root module. The Noir compiler does not enforce any conditions on the name of the file which is the crate root, however if you are compiling via Nargo the crate root must be called `lib.nr` or `main.nr` for library or binary crates respectively. - -## Packages - -A Nargo _package_ is a collection of one of more crates that provides a set of functionality. A package must include a Nargo.toml file. - -A package _must_ contain either a library or a binary crate, but not both. - -### Differences from Cargo Packages - -One notable difference between Rust's Cargo and Noir's Nargo is that while Cargo allows a package to contain an unlimited number of binary crates and a single library crate, Nargo currently only allows a package to contain a single crate. - -In future this restriction may be lifted to allow a Nargo package to contain both a binary and library crate or multiple binary crates. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/dependencies.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/dependencies.md deleted file mode 100644 index 24e02de08fe..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/dependencies.md +++ /dev/null @@ -1,124 +0,0 @@ ---- -title: Dependencies -description: - Learn how to specify and manage dependencies in Nargo, allowing you to upload packages to GitHub - and use them easily in your project. -keywords: [Nargo, dependencies, GitHub, package management, versioning] -sidebar_position: 1 ---- - -Nargo allows you to upload packages to GitHub and use them as dependencies. - -## Specifying a dependency - -Specifying a dependency requires a tag to a specific commit and the git url to the url containing -the package. - -Currently, there are no requirements on the tag contents. If requirements are added, it would follow -semver 2.0 guidelines. - -> Note: Without a `tag` , there would be no versioning and dependencies would change each time you -> compile your project. - -For example, to add the [ecrecover-noir library](https://github.com/colinnielsen/ecrecover-noir) to your project, add it to `Nargo.toml`: - -```toml -# Nargo.toml - -[dependencies] -ecrecover = {tag = "v0.8.0", git = "https://github.com/colinnielsen/ecrecover-noir"} -``` - -If the module is in a subdirectory, you can define a subdirectory in your git repository, for example: - -```toml -# Nargo.toml - -[dependencies] -easy_private_token_contract = {tag ="v0.1.0-alpha62", git = "https://github.com/AztecProtocol/aztec-packages", directory = "noir-contracts/contracts/easy_private_token_contract"} -``` - -## Specifying a local dependency - -You can also specify dependencies that are local to your machine. - -For example, this file structure has a library and binary crate - -```tree -├── binary_crate -│   ├── Nargo.toml -│   └── src -│   └── main.nr -└── lib_a - ├── Nargo.toml - └── src - └── lib.nr -``` - -Inside of the binary crate, you can specify: - -```toml -# Nargo.toml - -[dependencies] -lib_a = { path = "../lib_a" } -``` - -## Importing dependencies - -You can import a dependency to a Noir file using the following syntax. For example, to import the -ecrecover-noir library and local lib_a referenced above: - -```rust -use ecrecover; -use lib_a; -``` - -You can also import only the specific parts of dependency that you want to use, like so: - -```rust -use std::hash::sha256; -use std::scalar_mul::fixed_base_embedded_curve; -``` - -Lastly, as demonstrated in the -[elliptic curve example](../standard_library/cryptographic_primitives/ec_primitives.md#examples), you -can import multiple items in the same line by enclosing them in curly braces: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; -``` - -We don't have a way to consume libraries from inside a [workspace](./workspaces.md) as external dependencies right now. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -## Dependencies of Dependencies - -Note that when you import a dependency, you also get access to all of the dependencies of that package. - -For example, the [phy_vector](https://github.com/resurgencelabs/phy_vector) library imports an [fraction](https://github.com/resurgencelabs/fraction) library. If you're importing the phy_vector library, then you can access the functions in fractions library like so: - -```rust -use phy_vector; - -fn main(x : Field, y : pub Field) { - //... - let f = phy_vector::fraction::toFraction(true, 2, 1); - //... -} -``` - -## Available Libraries - -Noir does not currently have an official package manager. You can find a list of available Noir libraries in the [awesome-noir repo here](https://github.com/noir-lang/awesome-noir#libraries). - -Some libraries that are available today include: - -- [Standard Library](https://github.com/noir-lang/noir/tree/master/noir_stdlib) - the Noir Standard Library -- [Ethereum Storage Proof Verification](https://github.com/aragonzkresearch/noir-trie-proofs) - a library that contains the primitives necessary for RLP decoding (in the form of look-up table construction) and Ethereum state and storage proof verification (or verification of any trie proof involving 32-byte long keys) -- [BigInt](https://github.com/shuklaayush/noir-bigint) - a library that provides a custom BigUint56 data type, allowing for computations on large unsigned integers -- [ECrecover](https://github.com/colinnielsen/ecrecover-noir/tree/main) - a library to verify an ECDSA signature and return the source Ethereum address -- [Sparse Merkle Tree Verifier](https://github.com/vocdoni/smtverifier-noir/tree/main) - a library for verification of sparse Merkle trees -- [Signed Int](https://github.com/resurgencelabs/signed_int) - a library for accessing a custom Signed Integer data type, allowing access to negative numbers on Noir -- [Fraction](https://github.com/resurgencelabs/fraction) - a library for accessing fractional number data type in Noir, allowing results that aren't whole numbers diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/modules.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/modules.md deleted file mode 100644 index d21b009be3b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/modules.md +++ /dev/null @@ -1,211 +0,0 @@ ---- -title: Modules -description: - Learn how to organize your files using modules in Noir, following the same convention as Rust's - module system. Examples included. -keywords: [Noir, Rust, modules, organizing files, sub-modules] -sidebar_position: 2 ---- - -Noir's module system follows the same convention as the _newer_ version of Rust's module system. - -## Purpose of Modules - -Modules are used to organize files. Without modules all of your code would need to live in a single -file. In Noir, the compiler does not automatically scan all of your files to detect modules. This -must be done explicitly by the developer. - -## Examples - -### Importing a module in the crate root - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo.nr` - -```rust -fn from_foo() {} -``` - -In the above snippet, the crate root is the `src/main.nr` file. The compiler sees the module -declaration `mod foo` which prompts it to look for a foo.nr file. - -Visually this module hierarchy looks like the following : - -``` -crate - ├── main - │ - └── foo - └── from_foo - -``` - -The module filename may also be the name of the module as a directory with the contents in a -file named `mod.nr` within that directory. The above example can alternatively be expressed like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -fn from_foo() {} -``` - -Note that it's an error to have both files `src/foo.nr` and `src/foo/mod.nr` in the filesystem. - -### Importing a module throughout the tree - -All modules are accessible from the `crate::` namespace. - -``` -crate - ├── bar - ├── foo - └── main - -``` - -In the above snippet, if `bar` would like to use functions in `foo`, it can do so by `use crate::foo::function_name`. - -### Sub-modules - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -fn from_bar() {} -``` - -In the above snippet, we have added an extra module to the module tree; `bar`. `bar` is a submodule -of `foo` hence we declare bar in `foo.nr` with `mod bar`. Since `foo` is not the crate root, the -compiler looks for the file associated with the `bar` module in `src/foo/bar.nr` - -Visually the module hierarchy looks as follows: - -``` -crate - ├── main - │ - └── foo - ├── from_foo - └── bar - └── from_bar -``` - -Similar to importing a module in the crate root, modules can be placed in a `mod.nr` file, like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar/mod.nr` - -```rust -fn from_bar() {} -``` - -### Referencing a parent module - -Given a submodule, you can refer to its parent module using the `super` keyword. - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; - -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -// Same as bar::from_foo -use super::from_foo; - -fn from_bar() { - from_foo(); // invokes super::from_foo(), which is bar::from_foo() - super::from_foo(); // also invokes bar::from_foo() -} -``` - -### `use` visibility - -`use` declarations are private to the containing module, by default. However, like functions, -they can be marked as `pub` or `pub(crate)`. Such a use declaration serves to _re-export_ a name. -A public `use` declaration can therefore redirect some public name to a different target definition: -even a definition with a private canonical path, inside a different module. - -An example of re-exporting: - -```rust -mod some_module { - pub use foo::{bar, baz}; - mod foo { - pub fn bar() {} - pub fn baz() {} - } -} - -fn main() { - some_module::bar(); - some_module::baz(); -} -``` - -In this example, the module `some_module` re-exports two public names defined in `foo`. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/workspaces.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/workspaces.md deleted file mode 100644 index 513497f12bf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/modules_packages_crates/workspaces.md +++ /dev/null @@ -1,42 +0,0 @@ ---- -title: Workspaces -sidebar_position: 3 ---- - -Workspaces are a feature of nargo that allow you to manage multiple related Noir packages in a single repository. A workspace is essentially a group of related projects that share common build output directories and configurations. - -Each Noir project (with it's own Nargo.toml file) can be thought of as a package. Each package is expected to contain exactly one "named circuit", being the "name" defined in Nargo.toml with the program logic defined in `./src/main.nr`. - -For a project with the following structure: - -```tree -├── crates -│ ├── a -│ │ ├── Nargo.toml -│ │ └── Prover.toml -│ │ └── src -│ │ └── main.nr -│ └── b -│ ├── Nargo.toml -│ └── Prover.toml -│ └── src -│ └── main.nr -│ -└── Nargo.toml -``` - -You can define a workspace in Nargo.toml like so: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -`members` indicates which packages are included in the workspace. As such, all member packages of a workspace will be processed when the `--workspace` flag is used with various commands or if a `default-member` is not specified. - -`default-member` indicates which package various commands process by default. - -Libraries can be defined in a workspace. Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/_category_.json deleted file mode 100644 index af04c0933fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Standard Library", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/bigint.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/bigint.md deleted file mode 100644 index bcbb6b0d252..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/bigint.md +++ /dev/null @@ -1,127 +0,0 @@ ---- -title: Big Integers -description: How to use big integers from Noir standard library -keywords: - [ - Big Integer, - Noir programming language, - Noir libraries, - ] ---- - -The BigInt module in the standard library exposes some class of integers which do not fit (well) into a Noir native field. It implements modulo arithmetic, modulo a 'big' prime number. - -:::note - -The module can currently be considered as `Field`s with fixed modulo sizes used by a set of elliptic curves, in addition to just the native curve. [More work](https://github.com/noir-lang/noir/issues/510) is needed to achieve arbitrarily sized big integers. - -:::note - -`nargo` can be built with `--profile release-pedantic` to enable extra overflow checks which may affect `BigInt` results in some cases. -Consider the [`noir-bignum`](https://github.com/noir-lang/noir-bignum) library for an optimized alternative approach. - -::: - -Currently 6 classes of integers (i.e 'big' prime numbers) are available in the module, namely: - -- BN254 Fq: Bn254Fq -- BN254 Fr: Bn254Fr -- Secp256k1 Fq: Secpk1Fq -- Secp256k1 Fr: Secpk1Fr -- Secp256r1 Fr: Secpr1Fr -- Secp256r1 Fq: Secpr1Fq - -Where XXX Fq and XXX Fr denote respectively the order of the base and scalar field of the (usual) elliptic curve XXX. -For instance the big integer 'Secpk1Fq' in the standard library refers to integers modulo $2^{256}-2^{32}-977$. - -Feel free to explore the source code for the other primes: - -```rust title="big_int_definition" showLineNumbers -struct BigInt { - pointer: u32, - modulus: u32, -} -``` -> Source code: noir_stdlib/src/bigint.nr#L14-L19 - - -## Example usage - -A common use-case is when constructing a big integer from its bytes representation, and performing arithmetic operations on it: - -```rust title="big_int_example" showLineNumbers -fn big_int_example(x: u8, y: u8) { - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - let b = Secpk1Fq::from_le_bytes(&[y, x, 9]); - let c = (a + b) * b / a; - let d = c.to_le_bytes(); - println(d[0]); -} -``` -> Source code: test_programs/execution_success/bigint/src/main.nr#L72-L80 - - -## Methods - -The available operations for each big integer are: - -### from_le_bytes - -Construct a big integer from its little-endian bytes representation. Example: - -```rust - // Construct a big integer from a slice of bytes - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - // Construct a big integer from an array of 32 bytes - let a = Secpk1Fq::from_le_bytes_32([1;32]); - ``` - -Sure, here's the formatted version of the remaining methods: - -### to_le_bytes - -Return the little-endian bytes representation of a big integer. Example: - -```rust -let bytes = a.to_le_bytes(); -``` - -### add - -Add two big integers. Example: - -```rust -let sum = a + b; -``` - -### sub - -Subtract two big integers. Example: - -```rust -let difference = a - b; -``` - -### mul - -Multiply two big integers. Example: - -```rust -let product = a * b; -``` - -### div - -Divide two big integers. Note that division is field division and not euclidean division. Example: - -```rust -let quotient = a / b; -``` - -### eq - -Compare two big integers. Example: - -```rust -let are_equal = a == b; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/black_box_fns.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/black_box_fns.md deleted file mode 100644 index d6079ab182c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/black_box_fns.md +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Black Box Functions -description: Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. -keywords: [noir, black box functions] ---- - -Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. This makes certain zk-snark unfriendly computations cheaper than if they were implemented in Noir. - -The ACVM spec defines a set of blackbox functions which backends will be expected to implement. This allows backends to use optimized implementations of these constraints if they have them, however they may also fallback to less efficient naive implementations if not. - -## Function list - -Here is a list of the current black box functions: - -- [AES128](./cryptographic_primitives/ciphers.mdx#aes128) -- [SHA256](./cryptographic_primitives/hashes.mdx#sha256) -- [Schnorr signature verification](./cryptographic_primitives/schnorr.mdx) -- [Blake2s](./cryptographic_primitives/hashes.mdx#blake2s) -- [Blake3](./cryptographic_primitives/hashes.mdx#blake3) -- [Pedersen Hash](./cryptographic_primitives/hashes.mdx#pedersen_hash) -- [Pedersen Commitment](./cryptographic_primitives/hashes.mdx#pedersen_commitment) -- [ECDSA signature verification](./cryptographic_primitives/ecdsa_sig_verification.mdx) -- [Embedded curve operations (MSM, addition, ...)](./cryptographic_primitives/embedded_curve_ops.mdx) -- AND -- XOR -- RANGE -- [Keccak256](./cryptographic_primitives/hashes.mdx#keccak256) -- [Recursive proof verification](./recursion.mdx) - -Most black box functions are included as part of the Noir standard library, however `AND`, `XOR` and `RANGE` are used as part of the Noir language syntax. For instance, using the bitwise operator `&` will invoke the `AND` black box function. - -You can view the black box functions defined in the ACVM code [here](https://github.com/noir-lang/noir/blob/master/acvm-repo/acir/src/circuit/black_box_functions.rs). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/bn254.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/bn254.md deleted file mode 100644 index 3294f005dbb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/bn254.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Bn254 Field Library ---- - -Noir provides a module in standard library with some optimized functions for bn254 Fr in `std::field::bn254`. - -## decompose - -```rust -fn decompose(x: Field) -> (Field, Field) {} -``` - -Decomposes a single field into two fields, low and high. The low field contains the lower 16 bytes of the input field and the high field contains the upper 16 bytes of the input field. Both field results are range checked to 128 bits. - - -## assert_gt - -```rust -fn assert_gt(a: Field, b: Field) {} -``` - -Asserts that a > b. This will generate less constraints than using `assert(gt(a, b))`. - -## assert_lt - -```rust -fn assert_lt(a: Field, b: Field) {} -``` - -Asserts that a < b. This will generate less constraints than using `assert(lt(a, b))`. - -## gt - -```rust -fn gt(a: Field, b: Field) -> bool {} -``` - -Returns true if a > b. - -## lt - -```rust -fn lt(a: Field, b: Field) -> bool {} -``` - -Returns true if a < b. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/boundedvec.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/boundedvec.md deleted file mode 100644 index 7463ca092f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/boundedvec.md +++ /dev/null @@ -1,419 +0,0 @@ ---- -title: Bounded Vectors -keywords: [noir, vector, bounded vector, slice] -sidebar_position: 1 ---- - -A `BoundedVec` is a growable storage similar to a `Vec` except that it -is bounded with a maximum possible length. Unlike `Vec`, `BoundedVec` is not implemented -via slices and thus is not subject to the same restrictions slices are (notably, nested -slices - and thus nested vectors as well - are disallowed). - -Since a BoundedVec is backed by a normal array under the hood, growing the BoundedVec by -pushing an additional element is also more efficient - the length only needs to be increased -by one. - -For these reasons `BoundedVec` should generally be preferred over `Vec` when there -is a reasonable maximum bound that can be placed on the vector. - -Example: - -```rust -let mut vector: BoundedVec = BoundedVec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -assert(vector.max_len() == 10); -``` - -## Methods - -### new - -```rust -pub fn new() -> Self -``` - -Creates a new, empty vector of length zero. - -Since this container is backed by an array internally, it still needs an initial value -to give each element. To resolve this, each element is zeroed internally. This value -is guaranteed to be inaccessible unless `get_unchecked` is used. - -Example: - -```rust -let empty_vector: BoundedVec = BoundedVec::new(); -assert(empty_vector.len() == 0); -``` - -Note that whenever calling `new` the maximum length of the vector should always be specified -via a type signature: - -```rust title="new_example" showLineNumbers -fn good() -> BoundedVec { - // Ok! MaxLen is specified with a type annotation - let v1: BoundedVec = BoundedVec::new(); - let v2 = BoundedVec::new(); - - // Ok! MaxLen is known from the type of `good`'s return value - v2 -} - -fn bad() { - // Error: Type annotation needed - // The compiler can't infer `MaxLen` from this code. - let mut v3 = BoundedVec::new(); - v3.push(5); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L11-L27 - - -This defaulting of `MaxLen` (and numeric generics in general) to zero may change in future noir versions -but for now make sure to use type annotations when using bounded vectors. Otherwise, you will receive a constraint failure at runtime when the vec is pushed to. - -### get - -```rust -pub fn get(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this -will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - let last = v.get(v.len() - 1); - assert(first != last); -} -``` - -### get_unchecked - -```rust -pub fn get_unchecked(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero, without -performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, -it is unsafe! Use at your own risk! - -Example: - -```rust title="get_unchecked_example" showLineNumbers -fn sum_of_first_three(v: BoundedVec) -> u32 { - // Always ensure the length is larger than the largest - // index passed to get_unchecked - assert(v.len() > 2); - let first = v.get_unchecked(0); - let second = v.get_unchecked(1); - let third = v.get_unchecked(2); - first + second + third -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L54-L64 - - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - assert(first != 42); - v.set(0, 42); - let new_first = v.get(0); - assert(new_first == 42); -} -``` - -### set_unchecked - -```rust -pub fn set_unchecked(&mut self: Self, index: u64, value: T) -> T { -``` - -Writes an element to the vector at the given index, starting from zero, without performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, it is unsafe! Use at your own risk! - -Example: - -```rust title="set_unchecked_example" showLineNumbers -fn set_unchecked_example() { - let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([1, 2]); - - // Here we're safely writing within the valid range of `vec` - // `vec` now has the value [42, 2] - vec.set_unchecked(0, 42); - - // We can then safely read this value back out of `vec`. - // Notice that we use the checked version of `get` which would prevent reading unsafe values. - assert_eq(vec.get(0), 42); - - // We've now written past the end of `vec`. - // As this index is still within the maximum potential length of `v`, - // it won't cause a constraint failure. - vec.set_unchecked(2, 42); - println(vec); - - // This will write past the end of the maximum potential length of `vec`, - // it will then trigger a constraint failure. - vec.set_unchecked(5, 42); - println(vec); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L67-L91 - - - -### push - -```rust -pub fn push(&mut self, elem: T) { -``` - -Pushes an element to the end of the vector. This increases the length -of the vector by one. - -Panics if the new length of the vector will be greater than the max length. - -Example: - -```rust title="bounded-vec-push-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - v.push(1); - v.push(2); - - // Panics with failed assertion "push out of bounds" - v.push(3); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L95-L103 - - -### pop - -```rust -pub fn pop(&mut self) -> T -``` - -Pops the element at the end of the vector. This will decrease the length -of the vector by one. - -Panics if the vector is empty. - -Example: - -```rust title="bounded-vec-pop-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.push(1); - v.push(2); - - let two = v.pop(); - let one = v.pop(); - - assert(two == 2); - assert(one == 1); - // error: cannot pop from an empty vector - // let _ = v.pop(); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L108-L120 - - -### len - -```rust -pub fn len(self) -> u64 { -``` - -Returns the current length of this vector - -Example: - -```rust title="bounded-vec-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - assert(v.len() == 0); - - v.push(100); - assert(v.len() == 1); - - v.push(200); - v.push(300); - v.push(400); - assert(v.len() == 4); - - let _ = v.pop(); - let _ = v.pop(); - assert(v.len() == 2); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L125-L140 - - -### max_len - -```rust -pub fn max_len(_self: BoundedVec) -> u64 { -``` - -Returns the maximum length of this vector. This is always -equal to the `MaxLen` parameter this vector was initialized with. - -Example: - -```rust title="bounded-vec-max-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.max_len() == 5); - v.push(10); - assert(v.max_len() == 5); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L145-L151 - - -### storage - -```rust -pub fn storage(self) -> [T; MaxLen] { -``` - -Returns the internal array within this vector. -Since arrays in Noir are immutable, mutating the returned storage array will not mutate -the storage held internally by this vector. - -Note that uninitialized elements may be zeroed out! - -Example: - -```rust title="bounded-vec-storage-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.storage() == [0, 0, 0, 0, 0]); - - v.push(57); - assert(v.storage() == [57, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L156-L163 - - -### extend_from_array - -```rust -pub fn extend_from_array(&mut self, array: [T; Len]) -``` - -Pushes each element from the given array to this vector. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-array-example" showLineNumbers -let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([2, 4]); - - assert(vec.len == 2); - assert(vec.get(0) == 2); - assert(vec.get(1) == 4); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L168-L175 - - -### extend_from_bounded_vec - -```rust -pub fn extend_from_bounded_vec(&mut self, vec: BoundedVec) -``` - -Pushes each element from the other vector to this vector. The length of -the other vector is left unchanged. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-bounded-vec-example" showLineNumbers -let mut v1: BoundedVec = BoundedVec::new(); - let mut v2: BoundedVec = BoundedVec::new(); - - v2.extend_from_array([1, 2, 3]); - v1.extend_from_bounded_vec(v2); - - assert(v1.storage() == [1, 2, 3, 0, 0]); - assert(v2.storage() == [1, 2, 3, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L180-L189 - - -### from_array - -```rust -pub fn from_array(array: [T; Len]) -> Self -``` - -Creates a new vector, populating it with values derived from an array input. -The maximum length of the vector is determined based on the type signature. - -Example: -```rust -let bounded_vec: BoundedVec = BoundedVec::from_array([1, 2, 3]) -``` - -### map - -```rust -pub fn map(self, f: fn[Env](T) -> U) -> BoundedVec -``` - -Creates a new vector of equal size by calling a closure on each element in this vector. - -Example: - -```rust title="bounded-vec-map-example" showLineNumbers -let vec: BoundedVec = BoundedVec::from_array([1, 2, 3, 4]); - let result = vec.map(|value| value * 2); -``` -> Source code: noir_stdlib/src/collections/bounded_vec.nr#L493-L496 - - -### any - -```rust -pub fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -Returns true if the given predicate returns true for any element -in this vector. - -Example: - -```rust title="bounded-vec-any-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.extend_from_array([2, 4, 6]); - - let all_even = !v.any(|elem: u32| elem % 2 != 0); - assert(all_even); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L256-L262 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/hashmap.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/hashmap.md deleted file mode 100644 index 39f1ae9b32c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/hashmap.md +++ /dev/null @@ -1,594 +0,0 @@ ---- -title: HashMap -keywords: [noir, map, hash, hashmap] -sidebar_position: 1 ---- - -`HashMap` is used to efficiently store and look up key-value pairs. - -`HashMap` is a bounded type which can store anywhere from zero to `MaxLen` total elements. -Note that due to hash collisions, the actual maximum number of elements stored by any particular -hashmap is likely lower than `MaxLen`. This is true even with cryptographic hash functions since -every hash value will be performed modulo `MaxLen`. - -Example: - -```rust -// Create a mapping from Fields to u32s with a maximum length of 12 -// using a poseidon2 hasher -use std::hash::poseidon2::Poseidon2Hasher; -let mut map: HashMap> = HashMap::default(); - -map.insert(1, 2); -map.insert(3, 4); - -let two = map.get(1).unwrap(); -``` - -## Methods - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default { - /// Constructs an empty HashMap. - /// - /// Example: - /// - /// ```noir - /// let hashmap: HashMap> = HashMap::default(); - /// assert(hashmap.is_empty()); - /// ``` - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L694-L708 - - -Creates a fresh, empty HashMap. - -When using this function, always make sure to specify the maximum size of the hash map. - -This is the same `default` from the `Default` implementation given further below. It is -repeated here for convenience since it is the recommended way to create a hashmap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L204 - - -Because `HashMap` has so many generic arguments that are likely to be the same throughout -your program, it may be helpful to create a type alias: - -```rust title="type_alias" showLineNumbers -type MyMap = HashMap>; -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L195-L197 - - -### with_hasher - -```rust title="with_hasher" showLineNumbers -pub fn with_hasher(_build_hasher: B) -> Self - where - B: BuildHasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L103-L107 - - -Creates a hashmap with an existing `BuildHasher`. This can be used to ensure multiple -hashmaps are created with the same hasher instance. - -Example: - -```rust title="with_hasher_example" showLineNumbers -let my_hasher: BuildHasherDefault = Default::default(); - let hashmap: HashMap> = HashMap::with_hasher(my_hasher); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L206-L210 - - -### get - -```rust title="get" showLineNumbers -pub fn get( - self, - key: K - ) -> Option - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L470-L479 - - -Retrieves a value from the hashmap, returning `Option::none()` if it was not found. - -Example: - -```rust title="get_example" showLineNumbers -fn get_example(map: HashMap>) { - let x = map.get(12); - - if x.is_some() { - assert(x.unwrap() == 42); - } -} -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L298-L306 - - -### insert - -```rust title="insert" showLineNumbers -pub fn insert( - &mut self, - key: K, - value: V - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L514-L524 - - -Inserts a new key-value pair into the map. If the key was already in the map, its -previous value will be overridden with the newly provided one. - -Example: - -```rust title="insert_example" showLineNumbers -let mut map: HashMap> = HashMap::default(); - map.insert(12, 42); - assert(map.len() == 1); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L212-L216 - - -### remove - -```rust title="remove" showLineNumbers -pub fn remove( - &mut self, - key: K - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L573-L582 - - -Removes the given key-value pair from the map. If the key was not already present -in the map, this does nothing. - -Example: - -```rust title="remove_example" showLineNumbers -map.remove(12); - assert(map.is_empty()); - - // If a key was not present in the map, remove does nothing - map.remove(12); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L220-L227 - - -### is_empty - -```rust title="is_empty" showLineNumbers -pub fn is_empty(self) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L168-L170 - - -True if the length of the hash map is empty. - -Example: - -```rust title="is_empty_example" showLineNumbers -assert(map.is_empty()); - - map.insert(1, 2); - assert(!map.is_empty()); - - map.remove(1); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L229-L237 - - -### len - -```rust title="len" showLineNumbers -pub fn len(self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L429-L431 - - -Returns the current length of this hash map. - -Example: - -```rust title="len_example" showLineNumbers -// This is equivalent to checking map.is_empty() - assert(map.len() == 0); - - map.insert(1, 2); - map.insert(3, 4); - map.insert(5, 6); - assert(map.len() == 3); - - // 3 was already present as a key in the hash map, so the length is unchanged - map.insert(3, 7); - assert(map.len() == 3); - - map.remove(1); - assert(map.len() == 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L239-L254 - - -### capacity - -```rust title="capacity" showLineNumbers -pub fn capacity(_self: Self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L451-L453 - - -Returns the maximum capacity of this hashmap. This is always equal to the capacity -specified in the hashmap's type. - -Unlike hashmaps in general purpose programming languages, hashmaps in Noir have a -static capacity that does not increase as the map grows larger. Thus, this capacity -is also the maximum possible element count that can be inserted into the hashmap. -Due to hash collisions (modulo the hashmap length), it is likely the actual maximum -element count will be lower than the full capacity. - -Example: - -```rust title="capacity_example" showLineNumbers -let empty_map: HashMap> = HashMap::default(); - assert(empty_map.len() == 0); - assert(empty_map.capacity() == 42); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L256-L260 - - -### clear - -```rust title="clear" showLineNumbers -pub fn clear(&mut self) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L122-L124 - - -Clears the hashmap, removing all key-value pairs from it. - -Example: - -```rust title="clear_example" showLineNumbers -assert(!map.is_empty()); - map.clear(); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L262-L266 - - -### contains_key - -```rust title="contains_key" showLineNumbers -pub fn contains_key( - self, - key: K - ) -> bool - where - K: Hash + Eq, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L142-L151 - - -True if the hashmap contains the given key. Unlike `get`, this will not also return -the value associated with the key. - -Example: - -```rust title="contains_key_example" showLineNumbers -if map.contains_key(7) { - let value = map.get(7); - assert(value.is_some()); - } else { - println("No value for key 7!"); - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L268-L275 - - -### entries - -```rust title="entries" showLineNumbers -pub fn entries(self) -> BoundedVec<(K, V), N> { -``` -> Source code: noir_stdlib/src/collections/map.nr#L192-L194 - - -Returns a vector of each key-value pair present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="entries_example" showLineNumbers -let entries = map.entries(); - - // The length of a hashmap may not be compile-time known, so we - // need to loop over its capacity instead - for i in 0..map.capacity() { - if i < entries.len() { - let (key, value) = entries.get(i); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L309-L320 - - -### keys - -```rust title="keys" showLineNumbers -pub fn keys(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L228-L230 - - -Returns a vector of each key present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="keys_example" showLineNumbers -let keys = map.keys(); - - for i in 0..keys.max_len() { - if i < keys.len() { - let key = keys.get_unchecked(i); - let value = map.get(key).unwrap_unchecked(); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L322-L332 - - -### values - -```rust title="values" showLineNumbers -pub fn values(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L262-L264 - - -Returns a vector of each value present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="values_example" showLineNumbers -let values = map.values(); - - for i in 0..values.max_len() { - if i < values.len() { - let value = values.get_unchecked(i); - println(f"Found value {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L334-L343 - - -### iter_mut - -```rust title="iter_mut" showLineNumbers -pub fn iter_mut( - &mut self, - f: fn(K, V) -> (K, V) - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L298-L307 - - -Iterates through each key-value pair of the HashMap, setting each key-value pair to the -result returned from the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If this is not desired, use `iter_values_mut` if only values need to be mutated, -or `entries` if neither keys nor values need to be mutated. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_mut_example" showLineNumbers -// Add 1 to each key in the map, and double the value associated with that key. - map.iter_mut(|k, v| (k + 1, v * 2)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L347-L350 - - -### iter_keys_mut - -```rust title="iter_keys_mut" showLineNumbers -pub fn iter_keys_mut( - &mut self, - f: fn(K) -> K - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L338-L347 - - -Iterates through the HashMap, mutating each key to the result returned from -the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If only iteration is desired and the keys are not intended to be mutated, -prefer using `entries` instead. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_keys_mut_example" showLineNumbers -// Double each key, leaving the value associated with that key untouched - map.iter_keys_mut(|k| k * 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L352-L355 - - -### iter_values_mut - -```rust title="iter_values_mut" showLineNumbers -pub fn iter_values_mut(&mut self, f: fn(V) -> V) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L372-L374 - - -Iterates through the HashMap, applying the given function to each value and mutating the -value to equal the result. This function is more efficient than `iter_mut` and `iter_keys_mut` -because the keys are untouched and the underlying hashmap thus does not need to be reordered. - -Example: - -```rust title="iter_values_mut_example" showLineNumbers -// Halve each value - map.iter_values_mut(|v| v / 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L357-L360 - - -### retain - -```rust title="retain" showLineNumbers -pub fn retain(&mut self, f: fn(K, V) -> bool) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L393-L395 - - -Retains only the key-value pairs for which the given function returns true. -Any key-value pairs for which the function returns false will be removed from the map. - -Example: - -```rust title="retain_example" showLineNumbers -map.retain(|k, v| (k != 0) & (v != 0)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L280-L282 - - -## Trait Implementations - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default { - /// Constructs an empty HashMap. - /// - /// Example: - /// - /// ```noir - /// let hashmap: HashMap> = HashMap::default(); - /// assert(hashmap.is_empty()); - /// ``` - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L694-L708 - - -Constructs an empty HashMap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L204 - - -### eq - -```rust title="eq" showLineNumbers -impl Eq for HashMap -where - K: Eq + Hash, - V: Eq, - B: BuildHasher, - H: Hasher { - /// Checks if two HashMaps are equal. - /// - /// Example: - /// - /// ```noir - /// let mut map1: HashMap> = HashMap::default(); - /// let mut map2: HashMap> = HashMap::default(); - /// - /// map1.insert(1, 2); - /// map1.insert(3, 4); - /// - /// map2.insert(3, 4); - /// map2.insert(1, 2); - /// - /// assert(map1 == map2); - /// ``` - fn eq(self, other: HashMap) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L643-L667 - - -Checks if two HashMaps are equal. - -Example: - -```rust title="eq_example" showLineNumbers -let mut map1: HashMap> = HashMap::default(); - let mut map2: HashMap> = HashMap::default(); - - map1.insert(1, 2); - map1.insert(3, 4); - - map2.insert(3, 4); - map2.insert(1, 2); - - assert(map1 == map2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L284-L295 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/index.md deleted file mode 100644 index ea84c6d5c21..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/index.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -title: Containers -description: Container types provided by Noir's standard library for storing and retrieving data -keywords: [containers, data types, vec, hashmap] ---- diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/vec.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/vec.mdx deleted file mode 100644 index 475011922f8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/containers/vec.mdx +++ /dev/null @@ -1,170 +0,0 @@ ---- -title: Vectors -description: Delve into the Vec data type in Noir. Learn about its methods, practical examples, and best practices for using Vectors in your Noir code. -keywords: [noir, vector type, methods, examples, dynamic arrays] -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A vector is a collection type similar to Rust's `Vec` type. In Noir, it is a convenient way to use slices as mutable arrays. - -Example: - -```rust -let mut vector: Vec = Vec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -``` - -## Methods - -### new - -Creates a new, empty vector. - -```rust -pub fn new() -> Self -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### from_slice - -Creates a vector containing each element from a given slice. Mutations to the resulting vector will not affect the original slice. - -```rust -pub fn from_slice(slice: [T]) -> Self -``` - -Example: - -```rust -let slice: [Field] = &[1, 2, 3]; -let vector_from_slice = Vec::from_slice(slice); -assert(vector_from_slice.len() == 3); -``` - -### len - -Returns the number of elements in the vector. - -```rust -pub fn len(self) -> Field -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### get - -Retrieves an element from the vector at a given index. Panics if the index points beyond the vector's end. - -```rust -pub fn get(self, index: Field) -> T -``` - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -``` - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -Panics if the index points beyond the vector's end. - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -vector.set(1, 42); -assert(vector.get(1) == 42); -``` - -### push - -Adds a new element to the vector's end, returning a new vector with a length one greater than the original unmodified vector. - -```rust -pub fn push(&mut self, elem: T) -``` - -Example: - -```rust -let mut vector: Vec = Vec::new(); -vector.push(10); -assert(vector.len() == 1); -``` - -### pop - -Removes an element from the vector's end, returning a new vector with a length one less than the original vector, along with the removed element. Panics if the vector's length is zero. - -```rust -pub fn pop(&mut self) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20]); -let popped_elem = vector.pop(); -assert(popped_elem == 20); -assert(vector.len() == 1); -``` - -### insert - -Inserts an element at a specified index, shifting subsequent elements to the right. - -```rust -pub fn insert(&mut self, index: Field, elem: T) -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 30]); -vector.insert(1, 20); -assert(vector.get(1) == 20); -``` - -### remove - -Removes an element at a specified index, shifting subsequent elements to the left, and returns the removed element. - -```rust -pub fn remove(&mut self, index: Field) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20, 30]); -let removed_elem = vector.remove(1); -assert(removed_elem == 20); -assert(vector.len() == 2); -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ciphers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ciphers.mdx deleted file mode 100644 index d6a5e1a79eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ciphers.mdx +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Ciphers -description: - Learn about the implemented ciphers ready to use for any Noir project -keywords: - [ciphers, Noir project, aes128, encrypt] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## aes128 - -Given a plaintext as an array of bytes, returns the corresponding aes128 ciphertext (CBC mode). Input padding is automatically performed using PKCS#7, so that the output length is `input.len() + (16 - input.len() % 16)`. - -```rust title="aes128" showLineNumbers -pub fn aes128_encrypt(input: [u8; N], iv: [u8; 16], key: [u8; 16]) -> [u8] {} -``` -> Source code: noir_stdlib/src/aes128.nr#L2-L4 - - -```rust -fn main() { - let input: [u8; 4] = [0, 12, 3, 15] // Random bytes, will be padded to 16 bytes. - let iv: [u8; 16] = [0; 16]; // Initialisation vector - let key: [u8; 16] = [0; 16] // AES key - let ciphertext = std::aes128::aes128_encrypt(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()); // In this case, the output length will be 16 bytes. -} -``` - - - \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ec_primitives.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ec_primitives.md deleted file mode 100644 index f262d8160d6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ec_primitives.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Elliptic Curve Primitives -keywords: [cryptographic primitives, Noir project] -sidebar_position: 4 ---- - -Data structures and methods on them that allow you to carry out computations involving elliptic -curves over the (mathematical) field corresponding to `Field`. For the field currently at our -disposal, applications would involve a curve embedded in BN254, e.g. the -[Baby Jubjub curve](https://eips.ethereum.org/EIPS/eip-2494). - -## Data structures - -### Elliptic curve configurations - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Curve`), i.e. the specific elliptic -curve you want to use, which would be specified using any one of the methods -`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::new` which take the coefficients in the -defining equation together with a generator point as parameters. You can find more detail in the -comments in -[`noir_stdlib/src/ec/mod.nr`](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr), but -the gist of it is that the elliptic curves of interest are usually expressed in one of the standard -forms implemented here (Twisted Edwards, Montgomery and Short Weierstraß), and in addition to that, -you could choose to use `affine` coordinates (Cartesian coordinates - the usual (x,y) - possibly -together with a point at infinity) or `curvegroup` coordinates (some form of projective coordinates -requiring more coordinates but allowing for more efficient implementations of elliptic curve -operations). Conversions between all of these forms are provided, and under the hood these -conversions are done whenever an operation is more efficient in a different representation (or a -mixed coordinate representation is employed). - -### Points - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Point`), i.e. points lying on the -elliptic curve. For a curve configuration `c` and a point `p`, it may be checked that `p` -does indeed lie on `c` by calling `c.contains(p1)`. - -## Methods - -(given a choice of curve representation, e.g. use `std::ec::tecurve::affine::Curve` and use -`std::ec::tecurve::affine::Point`) - -- The **zero element** is given by `Point::zero()`, and we can verify whether a point `p: Point` is - zero by calling `p.is_zero()`. -- **Equality**: Points `p1: Point` and `p2: Point` may be checked for equality by calling - `p1.eq(p2)`. -- **Addition**: For `c: Curve` and points `p1: Point` and `p2: Point` on the curve, adding these two - points is accomplished by calling `c.add(p1,p2)`. -- **Negation**: For a point `p: Point`, `p.negate()` is its negation. -- **Subtraction**: For `c` and `p1`, `p2` as above, subtracting `p2` from `p1` is accomplished by - calling `c.subtract(p1,p2)`. -- **Scalar multiplication**: For `c` as above, `p: Point` a point on the curve and `n: Field`, - scalar multiplication is given by `c.mul(n,p)`. If instead `n :: [u1; N]`, i.e. `n` is a bit - array, the `bit_mul` method may be used instead: `c.bit_mul(n,p)` -- **Multi-scalar multiplication**: For `c` as above and arrays `n: [Field; N]` and `p: [Point; N]`, - multi-scalar multiplication is given by `c.msm(n,p)`. -- **Coordinate representation conversions**: The `into_group` method converts a point or curve - configuration in the affine representation to one in the CurveGroup representation, and - `into_affine` goes in the other direction. -- **Curve representation conversions**: `tecurve` and `montcurve` curves and points are equivalent - and may be converted between one another by calling `into_montcurve` or `into_tecurve` on their - configurations or points. `swcurve` is more general and a curve c of one of the other two types - may be converted to this representation by calling `c.into_swcurve()`, whereas a point `p` lying - on the curve given by `c` may be mapped to its corresponding `swcurve` point by calling - `c.map_into_swcurve(p)`. -- **Map-to-curve methods**: The Elligator 2 method of mapping a field element `n: Field` into a - `tecurve` or `montcurve` with configuration `c` may be called as `c.elligator2_map(n)`. For all of - the curve configurations, the SWU map-to-curve method may be called as `c.swu_map(z,n)`, where - `z: Field` depends on `Field` and `c` and must be chosen by the user (the conditions it needs to - satisfy are specified in the comments - [here](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr)). - -## Examples - -The -[ec_baby_jubjub test](https://github.com/noir-lang/noir/blob/master/test_programs/compile_success_empty/ec_baby_jubjub/src/main.nr) -illustrates all of the above primitives on various forms of the Baby Jubjub curve. A couple of more -interesting examples in Noir would be: - -Public-key cryptography: Given an elliptic curve and a 'base point' on it, determine the public key -from the private key. This is a matter of using scalar multiplication. In the case of Baby Jubjub, -for example, this code would do: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; - -fn bjj_pub_key(priv_key: Field) -> Point -{ - - let bjj = Curve::new(168700, 168696, G::new(995203441582195749578291179787384436505546430278305826713579947235728471134,5472060717959818805561601436314318772137091100104008585924551046643952123905)); - - let base_pt = Point::new(5299619240641551281634865583518297030282874472190772894086521144482721001553, 16950150798460657717958625567821834550301663161624707787222815936182638968203); - - bjj.mul(priv_key,base_pt) -} -``` - -This would come in handy in a Merkle proof. - -- EdDSA signature verification: This is a matter of combining these primitives with a suitable hash - function. See - [feat(stdlib): EdDSA sig verification noir#1136](https://github.com/noir-lang/noir/pull/1136) for - the case of Baby Jubjub and the Poseidon hash function. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx deleted file mode 100644 index 4c22e70e8de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: ECDSA Signature Verification -description: Learn about the cryptographic primitives regarding ECDSA over the secp256k1 and secp256r1 curves -keywords: [cryptographic primitives, Noir project, ecdsa, secp256k1, secp256r1, signatures] -sidebar_position: 3 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports ECDSA signatures verification over the secp256k1 and secp256r1 curves. - -## ecdsa_secp256k1::verify_signature - -Verifier for ECDSA Secp256k1 signatures. -See ecdsa_secp256k1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256k1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256k1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256k1::verify_signature_slice - -Verifier for ECDSA Secp256k1 signatures where the message is a slice. - -```rust title="ecdsa_secp256k1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L13-L20 - - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures. -See ecdsa_secp256r1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256r1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256r1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures where the message is a slice. - -```rust title="ecdsa_secp256r1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/eddsa.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/eddsa.mdx deleted file mode 100644 index b283de693c8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/eddsa.mdx +++ /dev/null @@ -1,37 +0,0 @@ ---- -title: EdDSA Verification -description: Learn about the cryptographic primitives regarding EdDSA -keywords: [cryptographic primitives, Noir project, eddsa, signatures] -sidebar_position: 5 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## eddsa::eddsa_poseidon_verify - -Verifier for EdDSA signatures - -```rust -fn eddsa_poseidon_verify(public_key_x : Field, public_key_y : Field, signature_s: Field, signature_r8_x: Field, signature_r8_y: Field, message: Field) -> bool -``` - -It is also possible to specify the hash algorithm used for the signature by using the `eddsa_verify` function by passing a type implementing the Hasher trait with the turbofish operator. -For instance, if you want to use Poseidon2 instead, you can do the following: -```rust -use std::hash::poseidon2::Poseidon2Hasher; - -eddsa_verify::(pub_key_a.x, pub_key_a.y, s_a, r8_a.x, r8_a.y, msg); -``` - - - -## eddsa::eddsa_to_pub - -Private to public key conversion. - -Returns `(pub_key_x, pub_key_y)` - -```rust -fn eddsa_to_pub(secret : Field) -> (Field, Field) -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx deleted file mode 100644 index 2da1e34f008..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx +++ /dev/null @@ -1,95 +0,0 @@ ---- -title: Scalar multiplication -description: See how you can perform scalar multiplication in Noir -keywords: [cryptographic primitives, Noir project, scalar multiplication] -sidebar_position: 1 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -The following functions perform operations over the embedded curve whose coordinates are defined by the configured noir field. -For the BN254 scalar field, this is BabyJubJub or Grumpkin. - -:::note -Suffixes `_low` and `_high` denote low and high limbs of a scalar. -::: - -## embedded_curve_ops::multi_scalar_mul - -Performs multi scalar multiplication over the embedded curve. -The function accepts arbitrary amount of point-scalar pairs on the input, it multiplies the individual pairs over -the curve and returns a sum of the resulting points. - -Points represented as x and y coordinates [x1, y1, x2, y2, ...], scalars as low and high limbs [low1, high1, low2, high2, ...]. - -```rust title="multi_scalar_mul" showLineNumbers -pub fn multi_scalar_mul( - points: [EmbeddedCurvePoint; N], - scalars: [EmbeddedCurveScalar; N] -) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L88-L93 - - -example - -```rust -fn main(point_x: Field, point_y: Field, scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::multi_scalar_mul([point_x, point_y], [scalar_low, scalar_high]); - println(point); -} -``` - -## embedded_curve_ops::fixed_base_scalar_mul - -Performs fixed base scalar multiplication over the embedded curve (multiplies input scalar with a generator point). -The function accepts a single scalar on the input represented as 2 fields. - -```rust title="fixed_base_scalar_mul" showLineNumbers -pub fn fixed_base_scalar_mul(scalar: EmbeddedCurveScalar) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L105-L107 - - -example - -```rust -fn main(scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::fixed_base_scalar_mul(scalar_low, scalar_high); - println(point); -} -``` - -## embedded_curve_ops::embedded_curve_add - -Adds two points on the embedded curve. -This function takes two `EmbeddedCurvePoint` structures as parameters, representing points on the curve, and returns a new `EmbeddedCurvePoint` structure that represents their sum. - -### Parameters: -- `point1` (`EmbeddedCurvePoint`): The first point to add. -- `point2` (`EmbeddedCurvePoint`): The second point to add. - -### Returns: -- `EmbeddedCurvePoint`: The resulting point after the addition of `point1` and `point2`. - -```rust title="embedded_curve_add" showLineNumbers -fn embedded_curve_add( - point1: EmbeddedCurvePoint, - point2: EmbeddedCurvePoint -) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L115-L120 - - -example - -```rust -fn main() { - let point1 = EmbeddedCurvePoint { x: 1, y: 2 }; - let point2 = EmbeddedCurvePoint { x: 3, y: 4 }; - let result = std::embedded_curve_ops::embedded_curve_add(point1, point2); - println!("Resulting Point: ({}, {})", result.x, result.y); -} -``` - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/hashes.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/hashes.mdx deleted file mode 100644 index d6640d26f25..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/hashes.mdx +++ /dev/null @@ -1,253 +0,0 @@ ---- -title: Hash methods -description: - Learn about the cryptographic primitives ready to use for any Noir project, including sha256, - blake2s, pedersen, mimc_bn254 and mimc -keywords: - [cryptographic primitives, Noir project, sha256, blake2s, pedersen, mimc_bn254, mimc, hash] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## sha256 - -Given an array of bytes, returns the resulting sha256 hash. -Specify a message_size to hash only the first `message_size` bytes of the input. - -```rust title="sha256" showLineNumbers -pub fn sha256(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/sha256.nr#L8-L10 - - -example: -```rust title="sha256_var" showLineNumbers -let digest = std::hash::sha256_var([x as u8], 1); -``` -> Source code: test_programs/execution_success/sha256/src/main.nr#L16-L18 - - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::sha256::sha256_var(x, 4); -} -``` - - - - -## blake2s - -Given an array of bytes, returns an array with the Blake2 hash - -```rust title="blake2s" showLineNumbers -pub fn blake2s(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L18-L20 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake2s(x); -} -``` - - - -## blake3 - -Given an array of bytes, returns an array with the Blake3 hash - -```rust title="blake3" showLineNumbers -pub fn blake3(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L24-L26 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake3(x); -} -``` - - - -## pedersen_hash - -Given an array of Fields, returns the Pedersen hash. - -```rust title="pedersen_hash" showLineNumbers -pub fn pedersen_hash(input: [Field; N]) -> Field -``` -> Source code: noir_stdlib/src/hash/mod.nr#L77-L79 - - -example: - -```rust title="pedersen-hash" showLineNumbers -fn main(x: Field, y: Field, expected_hash: Field) { - let hash = std::hash::pedersen_hash([x, y]); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/pedersen_hash/src/main.nr#L1-L7 - - - - -## pedersen_commitment - -Given an array of Fields, returns the Pedersen commitment. - -```rust title="pedersen_commitment" showLineNumbers -pub fn pedersen_commitment(input: [Field; N]) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/hash/mod.nr#L29-L31 - - -example: - -```rust title="pedersen-commitment" showLineNumbers -fn main(x: Field, y: Field, expected_commitment: std::embedded_curve_ops::EmbeddedCurvePoint) { - let commitment = std::hash::pedersen_commitment([x, y]); - assert_eq(commitment.x, expected_commitment.x); - assert_eq(commitment.y, expected_commitment.y); -} -``` -> Source code: test_programs/execution_success/pedersen_commitment/src/main.nr#L1-L8 - - - - -## keccak256 - -Given an array of bytes (`u8`), returns the resulting keccak hash as an array of -32 bytes (`[u8; 32]`). Specify a message_size to hash only the first -`message_size` bytes of the input. - -```rust title="keccak256" showLineNumbers -pub fn keccak256(input: [u8; N], message_size: u32) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L128-L130 - - -example: - -```rust title="keccak256" showLineNumbers -fn main(x: Field, result: [u8; 32]) { - // We use the `as` keyword here to denote the fact that we want to take just the first byte from the x Field - // The padding is taken care of by the program - let digest = std::hash::keccak256([x as u8], 1); - assert(digest == result); - - //#1399: variable message size - let message_size = 4; - let hash_a = std::hash::keccak256([1, 2, 3, 4], message_size); - let hash_b = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size); - - assert(hash_a == hash_b); - - let message_size_big = 8; - let hash_c = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size_big); - - assert(hash_a != hash_c); -} -``` -> Source code: test_programs/execution_success/keccak256/src/main.nr#L1-L21 - - - - -## poseidon - -Given an array of Fields, returns a new Field with the Poseidon Hash. Mind that you need to specify -how many inputs are there to your Poseidon function. - -```rust -// example for hash_1, hash_2 accepts an array of length 2, etc -fn hash_1(input: [Field; 1]) -> Field -``` - -example: - -```rust title="poseidon" showLineNumbers -use std::hash::poseidon; - -fn main(x1: [Field; 2], y1: pub Field, x2: [Field; 4], y2: pub Field) { - let hash1 = poseidon::bn254::hash_2(x1); - assert(hash1 == y1); - - let hash2 = poseidon::bn254::hash_4(x2); - assert(hash2 == y2); -} -``` -> Source code: test_programs/execution_success/poseidon_bn254_hash/src/main.nr#L1-L11 - - -## poseidon 2 - -Given an array of Fields, returns a new Field with the Poseidon2 Hash. Contrary to the Poseidon -function, there is only one hash and you can specify a message_size to hash only the first -`message_size` bytes of the input, - -```rust -// example for hashing the first three elements of the input -Poseidon2::hash(input, 3); -``` - -example: - -```rust title="poseidon2" showLineNumbers -use std::hash::poseidon2; - -fn main(inputs: [Field; 4], expected_hash: Field) { - let hash = poseidon2::Poseidon2::hash(inputs, inputs.len()); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/poseidon2/src/main.nr#L1-L8 - - -## mimc_bn254 and mimc - -`mimc_bn254` is `mimc`, but with hardcoded parameters for the BN254 curve. You can use it by -providing an array of Fields, and it returns a Field with the hash. You can use the `mimc` method if -you're willing to input your own constants: - -```rust -fn mimc(x: Field, k: Field, constants: [Field; N], exp : Field) -> Field -``` - -otherwise, use the `mimc_bn254` method: - -```rust -fn mimc_bn254(array: [Field; N]) -> Field -``` - -example: - -```rust - -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::mimc::mimc_bn254(x); -} -``` - -## hash_to_field - -```rust -fn hash_to_field(_input : [Field]) -> Field {} -``` - -Calculates the `blake2s` hash of the inputs and returns the hash modulo the field modulus to return -a value which can be represented as a `Field`. - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/index.md deleted file mode 100644 index 650f30165d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/index.md +++ /dev/null @@ -1,14 +0,0 @@ ---- -title: Cryptographic Primitives -description: - Learn about the cryptographic primitives ready to use for any Noir project -keywords: - [ - cryptographic primitives, - Noir project, - ] ---- - -The Noir team is progressively adding new cryptographic primitives to the standard library. Reach out for news or if you would be interested in adding more of these calculations in Noir. - -Some methods are available thanks to the Aztec backend, not being performed using Noir. When using other backends, these methods may or may not be supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/schnorr.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/schnorr.mdx deleted file mode 100644 index 00e7f257612..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/cryptographic_primitives/schnorr.mdx +++ /dev/null @@ -1,64 +0,0 @@ ---- -title: Schnorr Signatures -description: Learn how you can verify Schnorr signatures using Noir -keywords: [cryptographic primitives, Noir project, schnorr, signatures] -sidebar_position: 2 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## schnorr::verify_signature - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin). -See schnorr::verify_signature_slice for a version that works directly on slices. - -```rust title="schnorr_verify" showLineNumbers -pub fn verify_signature( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L2-L9 - - -where `_signature` can be generated like so using the npm package -[@noir-lang/barretenberg](https://www.npmjs.com/package/@noir-lang/barretenberg) - -```js -const { BarretenbergWasm } = require('@noir-lang/barretenberg/dest/wasm'); -const { Schnorr } = require('@noir-lang/barretenberg/dest/crypto/schnorr'); - -... - -const barretenberg = await BarretenbergWasm.new(); -const schnorr = new Schnorr(barretenberg); -const pubKey = schnorr.computePublicKey(privateKey); -const message = ... -const signature = Array.from( - schnorr.constructSignature(hash, privateKey).toBuffer() -); - -... -``` - - - -## schnorr::verify_signature_slice - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin) -where the message is a slice. - -```rust title="schnorr_verify_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/fmtstr.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/fmtstr.md deleted file mode 100644 index 65a7da9996d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/fmtstr.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: fmtstr ---- - -`fmtstr` is the type resulting from using format string (`f"..."`). - -## Methods - -### quoted_contents - -```rust title="quoted_contents" showLineNumbers -comptime fn quoted_contents(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/format_string.nr#L3-L5 - - -Returns the format string contents (that is, without the leading and trailing double quotes) as a `Quoted` value. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/is_unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/is_unconstrained.md deleted file mode 100644 index 51bb1bda8f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/is_unconstrained.md +++ /dev/null @@ -1,69 +0,0 @@ ---- -title: Is Unconstrained Function -description: - The is_unconstrained function returns wether the context at that point of the program is unconstrained or not. -keywords: - [ - unconstrained - ] ---- - -It's very common for functions in circuits to take unconstrained hints of an expensive computation and then verify it. This is done by running the hint in an unconstrained context and then verifying the result in a constrained context. - -When a function is marked as unconstrained, any subsequent functions that it calls will also be run in an unconstrained context. However, if we are implementing a library function, other users might call it within an unconstrained context or a constrained one. Generally, in an unconstrained context we prefer just computing the result instead of taking a hint of it and verifying it, since that'd mean doing the same computation twice: - -```rust - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - my_expensive_computation_hint(); - // verify my_expensive_computation: If external_interface is called from unconstrained, this is redundant - ... -} - -``` - -In order to improve the performance in an unconstrained context you can use the function at `std::runtime::is_unconstrained() -> bool`: - - -```rust -use dep::std::runtime::is_unconstrained; - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - if is_unconstrained() { - my_expensive_computation(); - } else { - my_expensive_computation_hint(); - // verify my_expensive_computation - ... - } -} - -``` - -The is_unconstrained result is resolved at compile time, so in unconstrained contexts the compiler removes the else branch, and in constrained contexts the compiler removes the if branch, reducing the amount of compute necessary to run external_interface. - -Note that using `is_unconstrained` in a `comptime` context will also return `true`: - -``` -fn main() { - comptime { - assert(is_unconstrained()); - } -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/logging.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/logging.md deleted file mode 100644 index db75ef9f86f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/logging.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Logging -description: - Learn how to use the println statement for debugging in Noir with this tutorial. Understand the - basics of logging in Noir and how to implement it in your code. -keywords: - [ - noir logging, - println statement, - print statement, - debugging in noir, - noir std library, - logging tutorial, - basic logging in noir, - noir logging implementation, - noir debugging techniques, - rust, - ] ---- - -The standard library provides two familiar statements you can use: `println` and `print`. Despite being a limited implementation of rust's `println!` and `print!` macros, these constructs can be useful for debugging. - -You can print the output of both statements in your Noir code by using the `nargo execute` command or the `--show-output` flag when using `nargo test` (provided there are print statements in your tests). - -It is recommended to use `nargo execute` if you want to debug failing constraints with `println` or `print` statements. This is due to every input in a test being a constant rather than a witness, so we issue an error during compilation while we only print during execution (which comes after compilation). Neither `println`, nor `print` are callable for failed constraints caught at compile time. - -Both `print` and `println` are generic functions which can work on integers, fields, strings, and even structs or expressions. Note however, that slices are currently unsupported. For example: - -```rust -struct Person { - age: Field, - height: Field, -} - -fn main(age: Field, height: Field) { - let person = Person { - age: age, - height: height, - }; - println(person); - println(age + height); - println("Hello world!"); -} -``` - -You can print different types in the same statement (including strings) with a type called `fmtstr`. It can be specified in the same way as a normal string, just prepended with an "f" character: - -```rust - let fmt_str = f"i: {i}, j: {j}"; - println(fmt_str); - - let s = myStruct { y: x, x: y }; - println(s); - - println(f"i: {i}, s: {s}"); - - println(x); - println([x, y]); - - let foo = fooStruct { my_struct: s, foo: 15 }; - println(f"s: {s}, foo: {foo}"); - - println(15); // prints 0x0f, implicit Field - println(-1 as u8); // prints 255 - println(-1 as i8); // prints -1 -``` - -Examples shown above are interchangeable between the two `print` statements: - -```rust -let person = Person { age : age, height : height }; - -println(person); -print(person); - -println("Hello world!"); // Prints with a newline at the end of the input -print("Hello world!"); // Prints the input and keeps cursor on the same line -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/merkle_trees.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/merkle_trees.md deleted file mode 100644 index 6a9ebf72ada..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/merkle_trees.md +++ /dev/null @@ -1,58 +0,0 @@ ---- -title: Merkle Trees -description: Learn about Merkle Trees in Noir with this tutorial. Explore the basics of computing a merkle root using a proof, with examples. -keywords: - [ - Merkle trees in Noir, - Noir programming language, - check membership, - computing root from leaf, - Noir Merkle tree implementation, - Merkle tree tutorial, - Merkle tree code examples, - Noir libraries, - pedersen hash., - ] ---- - -## compute_merkle_root - -Returns the root of the tree from the provided leaf and its hash path, using a [Pedersen hash](./cryptographic_primitives/hashes.mdx#pedersen_hash). - -```rust -fn compute_merkle_root(leaf : Field, index : Field, hash_path: [Field]) -> Field -``` - -example: - -```rust -/** - // these values are for this example only - index = "0" - priv_key = "0x000000000000000000000000000000000000000000000000000000616c696365" - secret = "0x1929ea3ab8d9106a899386883d9428f8256cfedb3c4f6b66bf4aa4d28a79988f" - note_hash_path = [ - "0x1e61bdae0f027b1b2159e1f9d3f8d00fa668a952dddd822fda80dc745d6f65cc", - "0x0e4223f3925f98934393c74975142bd73079ab0621f4ee133cee050a3c194f1a", - "0x2fd7bb412155bf8693a3bd2a3e7581a679c95c68a052f835dddca85fa1569a40" - ] - */ -fn main(index: Field, priv_key: Field, secret: Field, note_hash_path: [Field; 3]) { - - let pubkey = std::scalar_mul::fixed_base_embedded_curve(priv_key); - let pubkey_x = pubkey[0]; - let pubkey_y = pubkey[1]; - let note_commitment = std::hash::pedersen(&[pubkey_x, pubkey_y, secret]); - - let root = std::merkle::compute_merkle_root(note_commitment[0], index, note_hash_path.as_slice()); - println(root); -} -``` - -To check merkle tree membership: - -1. Include a merkle root as a program input. -2. Compute the merkle root of a given leaf, index and hash path. -3. Assert the merkle roots are equal. - -For more info about merkle trees, see the Wikipedia [page](https://en.wikipedia.org/wiki/Merkle_tree). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/expr.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/expr.md deleted file mode 100644 index f226c1eebb3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/expr.md +++ /dev/null @@ -1,323 +0,0 @@ ---- -title: Expr ---- - -`std::meta::expr` contains methods on the built-in `Expr` type for quoted, syntactically valid expressions. - -## Methods - -### as_array - -```rust title="as_array" showLineNumbers -comptime fn as_array(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L7-L9 - - -If this expression is an array, this returns a slice of each element in the array. - -### as_assert - -```rust title="as_assert" showLineNumbers -comptime fn as_assert(self) -> Option<(Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L12-L14 - - -If this expression is an assert, this returns the assert expression and the optional message. - -### as_assert_eq - -```rust title="as_assert_eq" showLineNumbers -comptime fn as_assert_eq(self) -> Option<(Expr, Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L17-L19 - - -If this expression is an assert_eq, this returns the left-hand-side and right-hand-side -expressions, together with the optional message. - -### as_assign - -```rust title="as_assign" showLineNumbers -comptime fn as_assign(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L22-L24 - - -If this expression is an assignment, this returns a tuple with the left hand side -and right hand side in order. - -### as_binary_op - -```rust title="as_binary_op" showLineNumbers -comptime fn as_binary_op(self) -> Option<(Expr, BinaryOp, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L32-L34 - - -If this expression is a binary operator operation ` `, -return the left-hand side, operator, and the right-hand side of the operation. - -### as_block - -```rust title="as_block" showLineNumbers -comptime fn as_block(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L37-L39 - - -If this expression is a block `{ stmt1; stmt2; ...; stmtN }`, return -a slice containing each statement. - -### as_bool - -```rust title="as_bool" showLineNumbers -comptime fn as_bool(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L42-L44 - - -If this expression is a boolean literal, return that literal. - -### as_comptime - -```rust title="as_comptime" showLineNumbers -comptime fn as_comptime(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L50-L52 - - -If this expression is a `comptime { stmt1; stmt2; ...; stmtN }` block, -return each statement in the block. - -### as_function_call - -```rust title="as_function_call" showLineNumbers -comptime fn as_function_call(self) -> Option<(Expr, [Expr])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L55-L57 - - -If this expression is a function call `foo(arg1, ..., argN)`, return -the function and a slice of each argument. - -### as_if - -```rust title="as_if" showLineNumbers -comptime fn as_if(self) -> Option<(Expr, Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L60-L62 - - -If this expression is an `if condition { then_branch } else { else_branch }`, -return the condition, then branch, and else branch. If there is no else branch, -`None` is returned for that branch instead. - -### as_index - -```rust title="as_index" showLineNumbers -comptime fn as_index(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L65-L67 - - -If this expression is an index into an array `array[index]`, return the -array and the index. - -### as_integer - -```rust title="as_integer" showLineNumbers -comptime fn as_integer(self) -> Option<(Field, bool)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L27-L29 - - -If this expression is an integer literal, return the integer as a field -as well as whether the integer is negative (true) or not (false). - -### as_let - -```rust title="as_let" showLineNumbers -comptime fn as_let(self) -> Option<(Expr, Option, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L70-L72 - - -If this expression is a let statement, returns the let pattern as an `Expr`, -the optional type annotation, and the assigned expression. - -### as_member_access - -```rust title="as_member_access" showLineNumbers -comptime fn as_member_access(self) -> Option<(Expr, Quoted)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L75-L77 - - -If this expression is a member access `foo.bar`, return the struct/tuple -expression and the field. The field will be represented as a quoted value. - -### as_method_call - -```rust title="as_method_call" showLineNumbers -comptime fn as_method_call(self) -> Option<(Expr, Quoted, [UnresolvedType], [Expr])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L80-L82 - - -If this expression is a method call `foo.bar::(arg1, ..., argN)`, return -the receiver, method name, a slice of each generic argument, and a slice of each argument. - -### as_repeated_element_array - -```rust title="as_repeated_element_array" showLineNumbers -comptime fn as_repeated_element_array(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L85-L87 - - -If this expression is a repeated element array `[elem; length]`, return -the repeated element and the length expressions. - -### as_repeated_element_slice - -```rust title="as_repeated_element_slice" showLineNumbers -comptime fn as_repeated_element_slice(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L90-L92 - - -If this expression is a repeated element slice `[elem; length]`, return -the repeated element and the length expressions. - -### as_slice - -```rust title="as_slice" showLineNumbers -comptime fn as_slice(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L95-L97 - - -If this expression is a slice literal `&[elem1, ..., elemN]`, -return each element of the slice. - -### as_tuple - -```rust title="as_tuple" showLineNumbers -comptime fn as_tuple(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L100-L102 - - -If this expression is a tuple `(field1, ..., fieldN)`, -return each element of the tuple. - -### as_unary_op - -```rust title="as_unary_op" showLineNumbers -comptime fn as_unary_op(self) -> Option<(UnaryOp, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L105-L107 - - -If this expression is a unary operation ` `, -return the unary operator as well as the right-hand side expression. - -### as_unsafe - -```rust title="as_unsafe" showLineNumbers -comptime fn as_unsafe(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L110-L112 - - -If this expression is an `unsafe { stmt1; ...; stmtN }` block, -return each statement inside in a slice. - -### has_semicolon - -```rust title="has_semicolon" showLineNumbers -comptime fn has_semicolon(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L115-L117 - - -`true` if this expression is trailed by a semicolon. E.g. - -``` -comptime { - let expr1 = quote { 1 + 2 }.as_expr().unwrap(); - let expr2 = quote { 1 + 2; }.as_expr().unwrap(); - - assert(expr1.as_binary_op().is_some()); - assert(expr2.as_binary_op().is_some()); - - assert(!expr1.has_semicolon()); - assert(expr2.has_semicolon()); -} -``` - -### is_break - -```rust title="is_break" showLineNumbers -comptime fn is_break(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L120-L122 - - -`true` if this expression is `break`. - -### is_continue - -```rust title="is_continue" showLineNumbers -comptime fn is_continue(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L125-L127 - - -`true` if this expression is `continue`. - -### modify - -```rust title="modify" showLineNumbers -comptime fn modify(self, f: fn[Env](Expr) -> Option) -> Expr { -``` -> Source code: noir_stdlib/src/meta/expr.nr#L129-L131 - - -Applies a mapping function to this expression and to all of its sub-expressions. -`f` will be applied to each sub-expression first, then applied to the expression itself. - -This happens recursively for every expression within `self`. - -For example, calling `modify` on `(&[1], &[2, 3])` with an `f` that returns `Option::some` -for expressions that are integers, doubling them, would return `(&[2], &[4, 6])`. - -### quoted - -```rust title="quoted" showLineNumbers -comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/expr.nr#L161-L163 - - -Returns this expression as a `Quoted` value. It's the same as `quote { $self }`. - -### resolve - -```rust title="resolve" showLineNumbers -comptime fn resolve(self, in_function: Option) -> TypedExpr {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L168-L170 - - -Resolves and type-checks this expression and returns the result as a `TypedExpr`. - -The `in_function` argument specifies where the expression is resolved: -- If it's `none`, the expression is resolved in the function where `resolve` was called -- If it's `some`, the expression is resolved in the given function - -If any names used by this expression are not in scope or if there are any type errors, -this will give compiler errors as if the expression was written directly into -the current `comptime` function. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/function_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/function_def.md deleted file mode 100644 index e490af68f7f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/function_def.md +++ /dev/null @@ -1,166 +0,0 @@ ---- -title: FunctionDefinition ---- - -`std::meta::function_def` contains methods on the built-in `FunctionDefinition` type representing -a function definition in the source program. - -## Methods - -### add_attribute - -```rust title="add_attribute" showLineNumbers -comptime fn add_attribute(self, attribute: str) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L3-L5 - - -Adds an attribute to the function. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### body - -```rust title="body" showLineNumbers -comptime fn body(self) -> Expr {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L8-L10 - - -Returns the body of the function as an expression. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L13-L15 - - -Returns true if this function has a custom attribute with the given name. - -### is_unconstrained - -```rust title="is_unconstrained" showLineNumbers -comptime fn is_unconstrained(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L18-L20 - - -Returns true if this function is unconstrained. - -### module - -```rust title="module" showLineNumbers -comptime fn module(self) -> Module {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L23-L25 - - -Returns the module where the function is defined. - -### name - -```rust title="name" showLineNumbers -comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L28-L30 - - -Returns the name of the function. - -### parameters - -```rust title="parameters" showLineNumbers -comptime fn parameters(self) -> [(Quoted, Type)] {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L33-L35 - - -Returns each parameter of the function as a tuple of (parameter pattern, parameter type). - -### return_type - -```rust title="return_type" showLineNumbers -comptime fn return_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L38-L40 - - -The return type of the function. - -### set_body - -```rust title="set_body" showLineNumbers -comptime fn set_body(self, body: Expr) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L43-L45 - - -Mutate the function body to a new expression. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_parameters - -```rust title="set_parameters" showLineNumbers -comptime fn set_parameters(self, parameters: [(Quoted, Type)]) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L48-L50 - - -Mutates the function's parameters to a new set of parameters. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -Expects a slice of (parameter pattern, parameter type) for each parameter. Requires -each parameter pattern to be a syntactically valid parameter. - -### set_return_type - -```rust title="set_return_type" showLineNumbers -comptime fn set_return_type(self, return_type: Type) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L53-L55 - - -Mutates the function's return type to a new type. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_return_public - -```rust title="set_return_public" showLineNumbers -comptime fn set_return_public(self, public: bool) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L58-L60 - - -Mutates the function's return visibility to public (if `true` is given) or private (if `false` is given). -This is only valid on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_unconstrained - -```rust title="set_unconstrained" showLineNumbers -comptime fn set_unconstrained(self, value: bool) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L63-L65 - - -Mutates the function to be unconstrained (if `true` is given) or not (if `false` is given). -This is only valid on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -## Trait Implementations - -```rust -impl Eq for FunctionDefinition -impl Hash for FunctionDefinition -``` - -Note that each function is assigned a unique ID internally and this is what is used for -equality and hashing. So even functions with identical signatures and bodies may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/index.md deleted file mode 100644 index 73d6fe52285..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/index.md +++ /dev/null @@ -1,210 +0,0 @@ ---- -title: Metaprogramming -description: Noir's Metaprogramming API -keywords: [metaprogramming, comptime, macros, macro, quote, unquote] ---- - -`std::meta` is the entry point for Noir's metaprogramming API. This consists of `comptime` functions -and types used for inspecting and modifying Noir programs. - -## Functions - -### type_of - -```rust title="type_of" showLineNumbers -pub comptime fn type_of(x: T) -> Type {} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L26-L28 - - -Returns the type of a variable at compile-time. - -Example: -```rust -comptime { - let x: i32 = 1; - let x_type: Type = std::meta::type_of(x); - - assert_eq(x_type, quote { i32 }.as_type()); -} -``` - -### unquote - -```rust title="unquote" showLineNumbers -pub comptime fn unquote(code: Quoted) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L18-L20 - - -Unquotes the passed-in token stream where this function was called. - -Example: -```rust -comptime { - let code = quote { 1 + 2 }; - - // let x = 1 + 2; - let x = unquote!(code); -} -``` - -### derive - -```rust title="derive" showLineNumbers -#[varargs] -pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L46-L49 - - -Attribute placed on struct definitions. - -Creates a trait impl for each trait passed in as an argument. -To do this, the trait must have a derive handler registered -with `derive_via` beforehand. The traits in the stdlib that -can be derived this way are `Eq`, `Ord`, `Default`, and `Hash`. - -Example: -```rust -#[derive(Eq, Default)] -struct Foo { - x: i32, - y: T, -} - -fn main() { - let foo1 = Foo::default(); - let foo2 = Foo { x: 0, y: &[0] }; - assert_eq(foo1, foo2); -} -``` - -### derive_via - -```rust title="derive_via_signature" showLineNumbers -pub comptime fn derive_via(t: TraitDefinition, f: DeriveFunction) { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L68-L70 - - -Attribute placed on trait definitions. - -Registers a function to create impls for the given trait -when the trait is used in a `derive` call. Users may use -this to register their own functions to enable their traits -to be derived by `derive`. - -Because this function requires a function as an argument which -should produce a trait impl for any given struct, users may find -it helpful to use a function like `std::meta::make_trait_impl` to -help creating these impls. - -Example: -```rust -#[derive_via(derive_do_nothing)] -trait DoNothing { - fn do_nothing(self); -} - -comptime fn derive_do_nothing(s: StructDefinition) -> Quoted { - let typ = s.as_type(); - quote { - impl DoNothing for $typ { - fn do_nothing(self) { - println("Nothing"); - } - } - } -} -``` - -As another example, `derive_eq` in the stdlib is used to derive the `Eq` -trait for any struct. It makes use of `make_trait_impl` to do this: - -```rust title="derive_eq" showLineNumbers -comptime fn derive_eq(s: StructDefinition) -> Quoted { - let signature = quote { fn eq(_self: Self, _other: Self) -> bool }; - let for_each_field = |name| quote { (_self.$name == _other.$name) }; - let body = |fields| { - if s.fields().len() == 0 { - quote { true } - } else { - fields - } - }; - crate::meta::make_trait_impl(s, quote { Eq }, signature, for_each_field, quote { & }, body) -} -``` -> Source code: noir_stdlib/src/cmp.nr#L10-L23 - - -### make_trait_impl - -```rust title="make_trait_impl" showLineNumbers -pub comptime fn make_trait_impl( - s: StructDefinition, - trait_name: Quoted, - function_signature: Quoted, - for_each_field: fn[Env1](Quoted) -> Quoted, - join_fields_with: Quoted, - body: fn[Env2](Quoted) -> Quoted -) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L87-L96 - - -A helper function to more easily create trait impls while deriving traits. - -Note that this function only works for traits which: -1. Have only one method -2. Have no generics on the trait itself. - - E.g. Using this on a trait such as `trait Foo { ... }` will result in the - generated impl incorrectly missing the `T` generic. - -If your trait fits these criteria then `make_trait_impl` is likely the easiest -way to write your derive handler. The arguments are as follows: - -- `s`: The struct to make the impl for -- `trait_name`: The name of the trait to derive. E.g. `quote { Eq }`. -- `function_signature`: The signature of the trait method to derive. E.g. `fn eq(self, other: Self) -> bool`. -- `for_each_field`: An operation to be performed on each field. E.g. `|name| quote { (self.$name == other.$name) }`. -- `join_fields_with`: A separator to join each result of `for_each_field` with. - E.g. `quote { & }`. You can also use an empty `quote {}` for no separator. -- `body`: The result of the field operations are passed into this function for any final processing. - This is the place to insert any setup/teardown code the trait requires. If the trait doesn't require - any such code, you can return the body as-is: `|body| body`. - -Example deriving `Hash`: - -```rust title="derive_hash" showLineNumbers -comptime fn derive_hash(s: StructDefinition) -> Quoted { - let name = quote { Hash }; - let signature = quote { fn hash(_self: Self, _state: &mut H) where H: std::hash::Hasher }; - let for_each_field = |name| quote { _self.$name.hash(_state); }; - crate::meta::make_trait_impl(s, name, signature, for_each_field, quote {}, |fields| fields) -} -``` -> Source code: noir_stdlib/src/hash/mod.nr#L147-L154 - - -Example deriving `Ord`: - -```rust title="derive_ord" showLineNumbers -comptime fn derive_ord(s: StructDefinition) -> Quoted { - let signature = quote { fn cmp(_self: Self, _other: Self) -> std::cmp::Ordering }; - let for_each_field = |name| quote { - if result == std::cmp::Ordering::equal() { - result = _self.$name.cmp(_other.$name); - } - }; - let body = |fields| quote { - let mut result = std::cmp::Ordering::equal(); - $fields - result - }; - crate::meta::make_trait_impl(s, quote { Ord }, signature, for_each_field, quote {}, body) -} -``` -> Source code: noir_stdlib/src/cmp.nr#L181-L196 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/module.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/module.md deleted file mode 100644 index efd3e61e125..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/module.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: Module ---- - -`std::meta::module` contains methods on the built-in `Module` type which represents a module in the source program. -Note that this type represents a module generally, it isn't limited to only `mod my_submodule { ... }` -declarations in the source program. - -## Methods - -### add_item - -```rust title="add_item" showLineNumbers -comptime fn add_item(self, item: Quoted) {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L3-L5 - - -Adds a top-level item (a function, a struct, a global, etc.) to the module. -Adding multiple items in one go is also valid if the `Quoted` value has multiple items in it. -Note that the items are type-checked as if they are inside the module they are being added to. - -### functions - -```rust title="functions" showLineNumbers -comptime fn functions(self) -> [FunctionDefinition] {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L18-L20 - - -Returns each function defined in the module. - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L8-L10 - - -Returns true if this module has a custom attribute with the given name. - -### is_contract - -```rust title="is_contract" showLineNumbers -comptime fn is_contract(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L13-L15 - - -`true` if this module is a contract module (was declared via `contract foo { ... }`). - -### name - -```rust title="name" showLineNumbers -comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L28-L30 - - -Returns the name of the module. - -### structs - -```rust title="structs" showLineNumbers -comptime fn structs(self) -> [StructDefinition] {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L23-L25 - - -Returns each struct defined in the module. - -## Trait Implementations - -```rust -impl Eq for Module -impl Hash for Module -``` - -Note that each module is assigned a unique ID internally and this is what is used for -equality and hashing. So even modules with identical names and contents may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/op.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/op.md deleted file mode 100644 index 18d1f0768fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/op.md +++ /dev/null @@ -1,244 +0,0 @@ ---- -title: UnaryOp and BinaryOp ---- - -`std::meta::op` contains the `UnaryOp` and `BinaryOp` types as well as methods on them. -These types are used to represent a unary or binary operator respectively in Noir source code. - -## Types - -### UnaryOp - -Represents a unary operator. One of `-`, `!`, `&mut`, or `*`. - -### Methods - -#### is_minus - -```rust title="is_minus" showLineNumbers -pub fn is_minus(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L7-L9 - - -Returns `true` if this operator is `-`. - -#### is_not - -```rust title="is_not" showLineNumbers -pub fn is_not(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L13-L15 - - -`true` if this operator is `!` - -#### is_mutable_reference - -```rust title="is_mutable_reference" showLineNumbers -pub fn is_mutable_reference(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L19-L21 - - -`true` if this operator is `&mut` - -#### is_dereference - -```rust title="is_dereference" showLineNumbers -pub fn is_dereference(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L25-L27 - - -`true` if this operator is `*` - -#### quoted - -```rust title="unary_quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/op.nr#L31-L33 - - -Returns this operator as a `Quoted` value. - -### Trait Implementations - -```rust -impl Eq for UnaryOp -impl Hash for UnaryOp -``` - -### BinaryOp - -Represents a binary operator. One of `+`, `-`, `*`, `/`, `%`, `==`, `!=`, `<`, `<=`, `>`, `>=`, `&`, `|`, `^`, `>>`, or `<<`. - -### Methods - -#### is_add - -```rust title="is_add" showLineNumbers -pub fn is_add(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L55-L57 - - -`true` if this operator is `+` - -#### is_subtract - -```rust title="is_subtract" showLineNumbers -pub fn is_subtract(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L61-L63 - - -`true` if this operator is `-` - -#### is_multiply - -```rust title="is_multiply" showLineNumbers -pub fn is_multiply(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L67-L69 - - -`true` if this operator is `*` - -#### is_divide - -```rust title="is_divide" showLineNumbers -pub fn is_divide(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L73-L75 - - -`true` if this operator is `/` - -#### is_modulo - -```rust title="is_modulo" showLineNumbers -pub fn is_modulo(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L145-L147 - - -`true` if this operator is `%` - -#### is_equal - -```rust title="is_equal" showLineNumbers -pub fn is_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L79-L81 - - -`true` if this operator is `==` - -#### is_not_equal - -```rust title="is_not_equal" showLineNumbers -pub fn is_not_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L85-L87 - - -`true` if this operator is `!=` - -#### is_less_than - -```rust title="is_less_than" showLineNumbers -pub fn is_less_than(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L91-L93 - - -`true` if this operator is `<` - -#### is_less_than_or_equal - -```rust title="is_less_than_or_equal" showLineNumbers -pub fn is_less_than_or_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L97-L99 - - -`true` if this operator is `<=` - -#### is_greater_than - -```rust title="is_greater_than" showLineNumbers -pub fn is_greater_than(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L103-L105 - - -`true` if this operator is `>` - -#### is_greater_than_or_equal - -```rust title="is_greater_than_or_equal" showLineNumbers -pub fn is_greater_than_or_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L109-L111 - - -`true` if this operator is `>=` - -#### is_and - -```rust title="is_and" showLineNumbers -pub fn is_and(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L115-L117 - - -`true` if this operator is `&` - -#### is_or - -```rust title="is_or" showLineNumbers -pub fn is_or(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L121-L123 - - -`true` if this operator is `|` - -#### is_shift_right - -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L133-L135 - - -`true` if this operator is `>>` - -#### is_shift_left - -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L133-L135 - - -`true` if this operator is `<<` - -#### quoted - -```rust title="binary_quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/op.nr#L151-L153 - - -Returns this operator as a `Quoted` value. - -### Trait Implementations - -```rust -impl Eq for BinaryOp -impl Hash for BinaryOp -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/quoted.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/quoted.md deleted file mode 100644 index 8fc188b25da..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/quoted.md +++ /dev/null @@ -1,138 +0,0 @@ ---- -title: Quoted ---- - -`std::meta::quoted` contains methods on the built-in `Quoted` type which represents -quoted token streams and is the result of the `quote { ... }` expression. - -## Methods - -### as_expr - -```rust title="as_expr" showLineNumbers -comptime fn as_expr(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L6-L8 - - -Parses the quoted token stream as an expression. Returns `Option::none()` if -the expression failed to parse. - -Example: - -```rust title="as_expr_example" showLineNumbers -#[test] - fn test_expr_as_function_call() { - comptime - { - let expr = quote { foo(42) }.as_expr().unwrap(); - let (_function, args) = expr.as_function_call().unwrap(); - assert_eq(args.len(), 1); - assert_eq(args[0].as_integer().unwrap(), (42, false)); - } - } -``` -> Source code: test_programs/noir_test_success/comptime_expr/src/main.nr#L331-L342 - - -### as_module - -```rust title="as_module" showLineNumbers -comptime fn as_module(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L11-L13 - - -Interprets this token stream as a module path leading to the name of a module. -Returns `Option::none()` if the module isn't found or this token stream cannot be parsed as a path. - -Example: - -```rust title="as_module_example" showLineNumbers -mod baz { - mod qux {} -} - -#[test] -fn as_module_test() { - comptime - { - let my_mod = quote { baz::qux }.as_module().unwrap(); - assert_eq(my_mod.name(), quote { qux }); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_module/src/main.nr#L102-L115 - - -### as_trait_constraint - -```rust title="as_trait_constraint" showLineNumbers -comptime fn as_trait_constraint(self) -> TraitConstraint {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L16-L18 - - -Interprets this token stream as a trait constraint (without an object type). -Note that this function panics instead of returning `Option::none()` if the token -stream does not parse and resolve to a valid trait constraint. - -Example: - -```rust title="implements_example" showLineNumbers -fn function_with_where(_x: T) where T: SomeTrait { - comptime - { - let t = quote { T }.as_type(); - let some_trait_i32 = quote { SomeTrait }.as_trait_constraint(); - assert(t.implements(some_trait_i32)); - - assert(t.get_trait_impl(some_trait_i32).is_none()); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L154-L165 - - -### as_type - -```rust title="as_type" showLineNumbers -comptime fn as_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L21-L23 - - -Interprets this token stream as a resolved type. Panics if the token -stream doesn't parse to a type or if the type isn't a valid type in scope. - -```rust title="implements_example" showLineNumbers -fn function_with_where(_x: T) where T: SomeTrait { - comptime - { - let t = quote { T }.as_type(); - let some_trait_i32 = quote { SomeTrait }.as_trait_constraint(); - assert(t.implements(some_trait_i32)); - - assert(t.get_trait_impl(some_trait_i32).is_none()); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L154-L165 - - -### tokens - -```rust title="tokens" showLineNumbers -comptime fn tokens(self) -> [Quoted] {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L26-L28 - - -Returns a slice of the individual tokens that form this token stream. - -## Trait Implementations - -```rust -impl Eq for Quoted -impl Hash for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/struct_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/struct_def.md deleted file mode 100644 index 212c636d12a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/struct_def.md +++ /dev/null @@ -1,177 +0,0 @@ ---- -title: StructDefinition ---- - -`std::meta::struct_def` contains methods on the built-in `StructDefinition` type. -This type corresponds to `struct Name { field1: Type1, ... }` items in the source program. - -## Methods - -### add_attribute - -```rust title="add_attribute" showLineNumbers -comptime fn add_attribute(self, attribute: str) {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L3-L5 - - -Adds an attribute to the struct. - -### add_generic - -```rust title="add_generic" showLineNumbers -comptime fn add_generic(self, generic_name: str) -> Type {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L8-L10 - - -Adds an generic to the struct. Returns the new generic type. -Errors if the given generic name isn't a single identifier or if -the struct already has a generic with the same name. - -This method should be used carefully, if there is existing code referring -to the struct type it may be checked before this function is called and -see the struct with the original number of generics. This method should -thus be preferred to use on code generated from other macros and structs -that are not used in function signatures. - -Example: - -```rust title="add-generic-example" showLineNumbers -comptime fn add_generic(s: StructDefinition) { - assert_eq(s.generics().len(), 0); - let new_generic = s.add_generic("T"); - - let generics = s.generics(); - assert_eq(generics.len(), 1); - assert_eq(generics[0], new_generic); - } -``` -> Source code: test_programs/compile_success_empty/comptime_struct_definition/src/main.nr#L38-L47 - - -### as_type - -```rust title="as_type" showLineNumbers -comptime fn as_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L15-L17 - - -Returns this struct as a type in the source program. If this struct has -any generics, the generics are also included as-is. - -### generics - -```rust title="generics" showLineNumbers -comptime fn generics(self) -> [Type] {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L26-L28 - - -Returns each generic on this struct. - -Example: - -``` -#[example] -struct Foo { - bar: [T; 2], - baz: Baz, -} - -comptime fn example(foo: StructDefinition) { - assert_eq(foo.generics().len(), 2); - - // Fails because `T` isn't in scope - // let t = quote { T }.as_type(); - // assert_eq(foo.generics()[0], t); -} -``` - -### fields - -```rust title="fields" showLineNumbers -comptime fn fields(self) -> [(Quoted, Type)] {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L33-L35 - - -Returns each field of this struct as a pair of (field name, field type). - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L20-L22 - - -Returns true if this struct has a custom attribute with the given name. - -### module - -```rust title="module" showLineNumbers -comptime fn module(self) -> Module {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L38-L40 - - -Returns the module where the struct is defined. - -### name - -```rust title="name" showLineNumbers -comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L43-L45 - - -Returns the name of this struct - -Note that the returned quoted value will be just the struct name, it will -not be the full path to the struct, nor will it include any generics. - -### set_fields - -```rust title="set_fields" showLineNumbers -comptime fn set_fields(self, new_fields: [(Quoted, Type)]) {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L52-L54 - - -Sets the fields of this struct to the given fields list where each element -is a pair of the field's name and the field's type. Expects each field name -to be a single identifier. Note that this will override any previous fields -on this struct. If those should be preserved, use `.fields()` to retrieve the -current fields on the struct type and append the new fields from there. - -Example: - -```rust -// Change this struct to: -// struct Foo { -// a: u32, -// b: i8, -// } -#[mangle_fields] -struct Foo { x: Field } - -comptime fn mangle_fields(s: StructDefinition) { - s.set_fields(&[ - (quote { a }, quote { u32 }.as_type()), - (quote { b }, quote { i8 }.as_type()), - ]); -} -``` - -## Trait Implementations - -```rust -impl Eq for StructDefinition -impl Hash for StructDefinition -``` - -Note that each struct is assigned a unique ID internally and this is what is used for -equality and hashing. So even structs with identical generics and fields may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/trait_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/trait_def.md deleted file mode 100644 index a1f363d46ff..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/trait_def.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: TraitDefinition ---- - -`std::meta::trait_def` contains methods on the built-in `TraitDefinition` type. This type -represents trait definitions such as `trait Foo { .. }` at the top-level of a program. - -## Methods - -### as_trait_constraint - -```rust title="as_trait_constraint" showLineNumbers -comptime fn as_trait_constraint(_self: Self) -> TraitConstraint {} -``` -> Source code: noir_stdlib/src/meta/trait_def.nr#L6-L8 - - -Converts this trait into a trait constraint. If there are any generics on this -trait, they will be kept as-is without instantiating or replacing them. - -## Trait Implementations - -```rust -impl Eq for TraitDefinition -impl Hash for TraitDefinition -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/trait_impl.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/trait_impl.md deleted file mode 100644 index 66d31ed2560..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/trait_impl.md +++ /dev/null @@ -1,60 +0,0 @@ ---- -title: TraitImpl ---- - -`std::meta::trait_impl` contains methods on the built-in `TraitImpl` type which represents a trait -implementation such as `impl Foo for Bar { ... }`. - -## Methods - -### trait_generic_args - -```rust title="trait_generic_args" showLineNumbers -comptime fn trait_generic_args(self) -> [Type] {} -``` -> Source code: noir_stdlib/src/meta/trait_impl.nr#L3-L5 - - -Returns any generic arguments on the trait of this trait implementation, if any. - -```rs -impl Foo for Bar { ... } - -comptime { - let bar_type = quote { Bar }.as_type(); - let foo = quote { Foo }.as_trait_constraint(); - - let my_impl: TraitImpl = bar_type.get_trait_impl(foo).unwrap(); - - let generics = my_impl.trait_generic_args(); - assert_eq(generics.len(), 2); - - assert_eq(generics[0], quote { i32 }.as_type()); - assert_eq(generics[1], quote { Field }.as_type()); -} -``` - -### methods - -```rust title="methods" showLineNumbers -comptime fn methods(self) -> [FunctionDefinition] {} -``` -> Source code: noir_stdlib/src/meta/trait_impl.nr#L8-L10 - - -Returns each method in this trait impl. - -Example: - -```rs -comptime { - let i32_type = quote { i32 }.as_type(); - let eq = quote { Eq }.as_trait_constraint(); - - let impl_eq_for_i32: TraitImpl = i32_type.get_trait_impl(eq).unwrap(); - let methods = impl_eq_for_i32.methods(); - - assert_eq(methods.len(), 1); - assert_eq(methods[0].name(), quote { eq }); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/typ.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/typ.md deleted file mode 100644 index 6c9f4b8d087..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/typ.md +++ /dev/null @@ -1,239 +0,0 @@ ---- -title: Type ---- - -`std::meta::typ` contains methods on the built-in `Type` type used for representing -a type in the source program. - -## Functions - -```rust title="fresh_type_variable" showLineNumbers -pub comptime fn fresh_type_variable() -> Type {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L5-L7 - - -Creates and returns an unbound type variable. This is a special kind of type internal -to type checking which will type check with any other type. When it is type checked -against another type it will also be set to that type. For example, if `a` is a type -variable and we have the type equality `(a, i32) = (u8, i32)`, the compiler will set -`a` equal to `u8`. - -Unbound type variables will often be rendered as `_` while printing them. Bound type -variables will appear as the type they are bound to. - -This can be used in conjunction with functions which internally perform type checks -such as `Type::implements` or `Type::get_trait_impl` to potentially grab some of the types used. - -Note that calling `Type::implements` or `Type::get_trait_impl` on a type variable will always -fail. - -Example: - -```rust title="serialize-setup" showLineNumbers -trait Serialize {} - -impl Serialize<1> for Field {} - -impl Serialize for [T; N] - where T: Serialize {} - -impl Serialize for (T, U) - where T: Serialize, U: Serialize {} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L20-L30 - -```rust title="fresh-type-variable-example" showLineNumbers -let typevar1 = std::meta::typ::fresh_type_variable(); - let constraint = quote { Serialize<$typevar1> }.as_trait_constraint(); - let field_type = quote { Field }.as_type(); - - // Search for a trait impl (binding typevar1 to 1 when the impl is found): - assert(field_type.implements(constraint)); - - // typevar1 should be bound to the "1" generic now: - assert_eq(typevar1.as_constant().unwrap(), 1); - - // If we want to do the same with a different type, we need to - // create a new type variable now that `typevar1` is bound - let typevar2 = std::meta::typ::fresh_type_variable(); - let constraint = quote { Serialize<$typevar2> }.as_trait_constraint(); - let array_type = quote { [(Field, Field); 5] }.as_type(); - assert(array_type.implements(constraint)); - - // Now typevar2 should be bound to the serialized pair size 2 times the array length 5 - assert_eq(typevar2.as_constant().unwrap(), 10); -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L130-L150 - - -## Methods - -### as_array - -```rust title="as_array" showLineNumbers -comptime fn as_array(self) -> Option<(Type, Type)> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L11-L13 - - -If this type is an array, return a pair of (element type, size type). - -Example: - -```rust -comptime { - let array_type = quote { [Field; 3] }.as_type(); - let (field_type, three_type) = array_type.as_array().unwrap(); - - assert(field_type.is_field()); - assert_eq(three_type.as_constant().unwrap(), 3); -} -``` - -### as_constant - -```rust title="as_constant" showLineNumbers -comptime fn as_constant(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L16-L18 - - -If this type is a constant integer (such as the `3` in the array type `[Field; 3]`), -return the numeric constant. - -### as_integer - -```rust title="as_integer" showLineNumbers -comptime fn as_integer(self) -> Option<(bool, u8)> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L21-L23 - - -If this is an integer type, return a boolean which is `true` -if the type is signed, as well as the number of bits of this integer type. - -### as_slice - -```rust title="as_slice" showLineNumbers -comptime fn as_slice(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L26-L28 - - -If this is a slice type, return the element type of the slice. - -### as_str - -```rust title="as_str" showLineNumbers -comptime fn as_str(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L31-L33 - - -If this is a `str` type, returns the length `N` as a type. - -### as_struct - -```rust title="as_struct" showLineNumbers -comptime fn as_struct(self) -> Option<(StructDefinition, [Type])> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L36-L38 - - -If this is a struct type, returns the struct in addition to -any generic arguments on this type. - -### as_tuple - -```rust title="as_tuple" showLineNumbers -comptime fn as_tuple(self) -> Option<[Type]> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L41-L43 - - -If this is a tuple type, returns each element type of the tuple. - -### get_trait_impl - -```rust title="get_trait_impl" showLineNumbers -comptime fn get_trait_impl(self, constraint: TraitConstraint) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L46-L48 - - -Retrieves the trait implementation that implements the given -trait constraint for this type. If the trait constraint is not -found, `None` is returned. Note that since the concrete trait implementation -for a trait constraint specified from a `where` clause is unknown, -this function will return `None` in these cases. If you only want to know -whether a type implements a trait, use `implements` instead. - -Example: - -```rust -comptime { - let field_type = quote { Field }.as_type(); - let default = quote { Default }.as_trait_constraint(); - - let the_impl: TraitImpl = field_type.get_trait_impl(default).unwrap(); - assert(the_impl.methods().len(), 1); -} -``` - -### implements - -```rust title="implements" showLineNumbers -comptime fn implements(self, constraint: TraitConstraint) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L51-L53 - - -`true` if this type implements the given trait. Note that unlike -`get_trait_impl` this will also return true for any `where` constraints -in scope. - -Example: - -```rust -fn foo() where T: Default { - comptime { - let field_type = quote { Field }.as_type(); - let default = quote { Default }.as_trait_constraint(); - assert(field_type.implements(default)); - - let t = quote { T }.as_type(); - assert(t.implements(default)); - } -} -``` - -### is_bool - -```rust title="is_bool" showLineNumbers -comptime fn is_bool(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L56-L58 - - -`true` if this type is `bool`. - -### is_field - -```rust title="is_field" showLineNumbers -comptime fn is_field(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L61-L63 - - -`true` if this type is `Field`. - -## Trait Implementations - -```rust -impl Eq for Type -impl Hash for Type -``` -Note that this is syntactic equality, this is not the same as whether two types will type check -to be the same type. Unless type inference or generics are being used however, users should not -typically have to worry about this distinction unless `std::meta::typ::fresh_type_variable` is used. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/typed_expr.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/typed_expr.md deleted file mode 100644 index 1ee71c8b064..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/typed_expr.md +++ /dev/null @@ -1,27 +0,0 @@ ---- -title: TypedExpr ---- - -`std::meta::typed_expr` contains methods on the built-in `TypedExpr` type for resolved and type-checked expressions. - -## Methods - -### get_type - -```rust title="as_function_definition" showLineNumbers -comptime fn as_function_definition(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typed_expr.nr#L7-L9 - - -If this expression refers to a function definitions, returns it. Otherwise returns `Option::none()`. - -### get_type - -```rust title="get_type" showLineNumbers -comptime fn get_type(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typed_expr.nr#L13-L15 - - -Returns the type of the expression, or `Option::none()` if there were errors when the expression was previously resolved. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/unresolved_type.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/unresolved_type.md deleted file mode 100644 index d6f2b1494bb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/unresolved_type.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: UnresolvedType ---- - -`std::meta::unresolved_type` contains methods on the built-in `UnresolvedType` type for the syntax of types. - -## Methods - -### is_field - -```rust title="is_field" showLineNumbers -comptime fn is_field(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L3-L5 - - -Returns true if this type refers to the Field type. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/options.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/options.md deleted file mode 100644 index a1bd4e1de5f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/options.md +++ /dev/null @@ -1,101 +0,0 @@ ---- -title: Option Type ---- - -The `Option` type is a way to express that a value might be present (`Some(T))` or absent (`None`). It's a safer way to handle potential absence of values, compared to using nulls in many other languages. - -```rust -struct Option { - None, - Some(T), -} -``` - -The `Option` type, already imported into your Noir program, can be used directly: - -```rust -fn main() { - let none = Option::none(); - let some = Option::some(3); -} -``` - -See [this test](https://github.com/noir-lang/noir/blob/5cbfb9c4a06c8865c98ff2b594464b037d821a5c/crates/nargo_cli/tests/test_data/option/src/main.nr) for a more comprehensive set of examples of each of the methods described below. - -## Methods - -### none - -Constructs a none value. - -### some - -Constructs a some wrapper around a given value. - -### is_none - -Returns true if the Option is None. - -### is_some - -Returns true of the Option is Some. - -### unwrap - -Asserts `self.is_some()` and returns the wrapped value. - -### unwrap_unchecked - -Returns the inner value without asserting `self.is_some()`. This method can be useful within an if condition when we already know that `option.is_some()`. If the option is None, there is no guarantee what value will be returned, only that it will be of type T for an `Option`. - -### unwrap_or - -Returns the wrapped value if `self.is_some()`. Otherwise, returns the given default value. - -### unwrap_or_else - -Returns the wrapped value if `self.is_some()`. Otherwise, calls the given function to return a default value. - -### expect - -Asserts `self.is_some()` with a provided custom message and returns the contained `Some` value. The custom message is expected to be a format string. - -### map - -If self is `Some(x)`, this returns `Some(f(x))`. Otherwise, this returns `None`. - -### map_or - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns the given default value. - -### map_or_else - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns `default()`. - -### and - -Returns None if self is None. Otherwise, this returns `other`. - -### and_then - -If self is None, this returns None. Otherwise, this calls the given function with the Some value contained within self, and returns the result of that call. In some languages this function is called `flat_map` or `bind`. - -### or - -If self is Some, return self. Otherwise, return `other`. - -### or_else - -If self is Some, return self. Otherwise, return `default()`. - -### xor - -If only one of the two Options is Some, return that option. Otherwise, if both options are Some or both are None, None is returned. - -### filter - -Returns `Some(x)` if self is `Some(x)` and `predicate(x)` is true. Otherwise, this returns `None`. - -### flatten - -Flattens an `Option>` into a `Option`. This returns `None` if the outer Option is None. Otherwise, this returns the inner Option. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/recursion.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/recursion.mdx deleted file mode 100644 index 60414a2fa51..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/recursion.mdx +++ /dev/null @@ -1,85 +0,0 @@ ---- -title: Recursive Proofs -description: Learn about how to write recursive proofs in Noir. -keywords: [recursion, recursive proofs, verification_key, verify_proof] ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports recursively verifying proofs, meaning you verify the proof of a Noir program in another Noir program. This enables creating proofs of arbitrary size by doing step-wise verification of smaller components of a large proof. - -Read [the explainer on recursion](../../explainers/explainer-recursion.md) to know more about this function and the [guide on how to use it.](../../how_to/how-to-recursion.md) - -## The `#[recursive]` Attribute - -In Noir, the `#[recursive]` attribute is used to indicate that a circuit is designed for recursive proof generation. When applied, it informs the compiler and the tooling that the circuit should be compiled in a way that makes its proofs suitable for recursive verification. This attribute eliminates the need for manual flagging of recursion at the tooling level, streamlining the proof generation process for recursive circuits. - -### Example usage with `#[recursive]` - -```rust -#[recursive] -fn main(x: Field, y: pub Field) { - assert(x == y, "x and y are not equal"); -} - -// This marks the circuit as recursion-friendly and indicates that proofs generated from this circuit -// are intended for recursive verification. -``` - -By incorporating this attribute directly in the circuit's definition, tooling like Nargo and NoirJS can automatically execute recursive-specific duties for Noir programs (e.g. recursive-friendly proof artifact generation) without additional flags or configurations. - -## Verifying Recursive Proofs - -```rust -#[foreign(recursive_aggregation)] -pub fn verify_proof(verification_key: [Field], proof: [Field], public_inputs: [Field], key_hash: Field) {} -``` - - - -## Example usage - -```rust - -fn main( - verification_key : [Field; 114], - proof : [Field; 93], - public_inputs : [Field; 1], - key_hash : Field, - proof_b : [Field; 93], -) { - std::verify_proof( - verification_key, - proof, - public_inputs, - key_hash - ); - - std::verify_proof( - verification_key, - proof_b, - public_inputs, - key_hash - ); -} -``` - -You can see a full example of recursive proofs in [this example recursion demo repo](https://github.com/noir-lang/noir-examples/tree/master/recursion). - -## Parameters - -### `verification_key` - -The verification key for the zk program that is being verified. - -### `proof` - -The proof for the zk program that is being verified. - -### `public_inputs` - -These represent the public inputs of the proof we are verifying. - -### `key_hash` - -A key hash is used to check the validity of the verification key. The circuit implementing this opcode can use this hash to ensure that the key provided to the circuit matches the key produced by the circuit creator. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/traits.md deleted file mode 100644 index 0629623a15a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/traits.md +++ /dev/null @@ -1,625 +0,0 @@ ---- -title: Traits -description: Noir's stdlib provides a few commonly used traits. -keywords: [traits, trait, interface, protocol, default, add, eq] ---- - -## `std::default` - -### `std::default::Default` - -```rust title="default-trait" showLineNumbers -trait Default { - fn default() -> Self; -} -``` -> Source code: noir_stdlib/src/default.nr#L4-L8 - - -Constructs a default value of a type. - -Implementations: -```rust -impl Default for Field { .. } - -impl Default for i8 { .. } -impl Default for i16 { .. } -impl Default for i32 { .. } -impl Default for i64 { .. } - -impl Default for u8 { .. } -impl Default for u16 { .. } -impl Default for u32 { .. } -impl Default for u64 { .. } - -impl Default for () { .. } -impl Default for bool { .. } - -impl Default for [T; N] - where T: Default { .. } - -impl Default for [T] { .. } - -impl Default for (A, B) - where A: Default, B: Default { .. } - -impl Default for (A, B, C) - where A: Default, B: Default, C: Default { .. } - -impl Default for (A, B, C, D) - where A: Default, B: Default, C: Default, D: Default { .. } - -impl Default for (A, B, C, D, E) - where A: Default, B: Default, C: Default, D: Default, E: Default { .. } -``` - -For primitive integer types, the return value of `default` is `0`. Container -types such as arrays are filled with default values of their element type, -except slices whose length is unknown and thus defaulted to zero. - ---- - -## `std::convert` - -### `std::convert::From` - -```rust title="from-trait" showLineNumbers -trait From { - fn from(input: T) -> Self; -} -``` -> Source code: noir_stdlib/src/convert.nr#L1-L5 - - -The `From` trait defines how to convert from a given type `T` to the type on which the trait is implemented. - -The Noir standard library provides a number of implementations of `From` between primitive types. -```rust title="from-impls" showLineNumbers -// Unsigned integers - -impl From for u32 { - fn from(value: u8) -> u32 { - value as u32 - } -} - -impl From for u64 { - fn from(value: u8) -> u64 { - value as u64 - } -} -impl From for u64 { - fn from(value: u32) -> u64 { - value as u64 - } -} - -impl From for Field { - fn from(value: u8) -> Field { - value as Field - } -} -impl From for Field { - fn from(value: u32) -> Field { - value as Field - } -} -impl From for Field { - fn from(value: u64) -> Field { - value as Field - } -} - -// Signed integers - -impl From for i32 { - fn from(value: i8) -> i32 { - value as i32 - } -} - -impl From for i64 { - fn from(value: i8) -> i64 { - value as i64 - } -} -impl From for i64 { - fn from(value: i32) -> i64 { - value as i64 - } -} - -// Booleans -impl From for u8 { - fn from(value: bool) -> u8 { - value as u8 - } -} -impl From for u32 { - fn from(value: bool) -> u32 { - value as u32 - } -} -impl From for u64 { - fn from(value: bool) -> u64 { - value as u64 - } -} -impl From for i8 { - fn from(value: bool) -> i8 { - value as i8 - } -} -impl From for i32 { - fn from(value: bool) -> i32 { - value as i32 - } -} -impl From for i64 { - fn from(value: bool) -> i64 { - value as i64 - } -} -impl From for Field { - fn from(value: bool) -> Field { - value as Field - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L25-L116 - - -#### When to implement `From` - -As a general rule of thumb, `From` may be implemented in the [situations where it would be suitable in Rust](https://doc.rust-lang.org/std/convert/trait.From.html#when-to-implement-from): - -- The conversion is *infallible*: Noir does not provide an equivalent to Rust's `TryFrom`, if the conversion can fail then provide a named method instead. -- The conversion is *lossless*: semantically, it should not lose or discard information. For example, `u32: From` can losslessly convert any `u16` into a valid `u32` such that the original `u16` can be recovered. On the other hand, `u16: From` should not be implemented as `2**16` is a `u32` which cannot be losslessly converted into a `u16`. -- The conversion is *value-preserving*: the conceptual kind and meaning of the resulting value is the same, even though the Noir type and technical representation might be different. While it's possible to infallibly and losslessly convert a `u8` into a `str<2>` hex representation, `4u8` and `"04"` are too different for `str<2>: From` to be implemented. -- The conversion is *obvious*: it's the only reasonable conversion between the two types. If there's ambiguity on how to convert between them such that the same input could potentially map to two different values then a named method should be used. For instance rather than implementing `U128: From<[u8; 16]>`, the methods `U128::from_le_bytes` and `U128::from_be_bytes` are used as otherwise the endianness of the array would be ambiguous, resulting in two potential values of `U128` from the same byte array. - -One additional recommendation specific to Noir is: -- The conversion is *efficient*: it's relatively cheap to convert between the two types. Due to being a ZK DSL, it's more important to avoid unnecessary computation compared to Rust. If the implementation of `From` would encourage users to perform unnecessary conversion, resulting in additional proving time, then it may be preferable to expose functionality such that this conversion may be avoided. - -### `std::convert::Into` - -The `Into` trait is defined as the reciprocal of `From`. It should be easy to convince yourself that if we can convert to type `A` from type `B`, then it's possible to convert type `B` into type `A`. - -For this reason, implementing `From` on a type will automatically generate a matching `Into` implementation. One should always prefer implementing `From` over `Into` as implementing `Into` will not generate a matching `From` implementation. - -```rust title="into-trait" showLineNumbers -trait Into { - fn into(self) -> T; -} - -impl Into for U where T: From { - fn into(self) -> T { - T::from(self) - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L13-L23 - - -`Into` is most useful when passing function arguments where the types don't quite match up with what the function expects. In this case, the compiler has enough type information to perform the necessary conversion by just appending `.into()` onto the arguments in question. - ---- - -## `std::cmp` - -### `std::cmp::Eq` - -```rust title="eq-trait" showLineNumbers -trait Eq { - fn eq(self, other: Self) -> bool; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L4-L8 - - -Returns `true` if `self` is equal to `other`. Implementing this trait on a type -allows the type to be used with `==` and `!=`. - -Implementations: -```rust -impl Eq for Field { .. } - -impl Eq for i8 { .. } -impl Eq for i16 { .. } -impl Eq for i32 { .. } -impl Eq for i64 { .. } - -impl Eq for u8 { .. } -impl Eq for u16 { .. } -impl Eq for u32 { .. } -impl Eq for u64 { .. } - -impl Eq for () { .. } -impl Eq for bool { .. } - -impl Eq for [T; N] - where T: Eq { .. } - -impl Eq for [T] - where T: Eq { .. } - -impl Eq for (A, B) - where A: Eq, B: Eq { .. } - -impl Eq for (A, B, C) - where A: Eq, B: Eq, C: Eq { .. } - -impl Eq for (A, B, C, D) - where A: Eq, B: Eq, C: Eq, D: Eq { .. } - -impl Eq for (A, B, C, D, E) - where A: Eq, B: Eq, C: Eq, D: Eq, E: Eq { .. } -``` - -### `std::cmp::Ord` - -```rust title="ord-trait" showLineNumbers -trait Ord { - fn cmp(self, other: Self) -> Ordering; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L175-L179 - - -`a.cmp(b)` compares two values returning `Ordering::less()` if `a < b`, -`Ordering::equal()` if `a == b`, or `Ordering::greater()` if `a > b`. -Implementing this trait on a type allows `<`, `<=`, `>`, and `>=` to be -used on values of the type. - -`std::cmp` also provides `max` and `min` functions for any type which implements the `Ord` trait. - -Implementations: - -```rust -impl Ord for u8 { .. } -impl Ord for u16 { .. } -impl Ord for u32 { .. } -impl Ord for u64 { .. } - -impl Ord for i8 { .. } -impl Ord for i16 { .. } -impl Ord for i32 { .. } - -impl Ord for i64 { .. } - -impl Ord for () { .. } -impl Ord for bool { .. } - -impl Ord for [T; N] - where T: Ord { .. } - -impl Ord for [T] - where T: Ord { .. } - -impl Ord for (A, B) - where A: Ord, B: Ord { .. } - -impl Ord for (A, B, C) - where A: Ord, B: Ord, C: Ord { .. } - -impl Ord for (A, B, C, D) - where A: Ord, B: Ord, C: Ord, D: Ord { .. } - -impl Ord for (A, B, C, D, E) - where A: Ord, B: Ord, C: Ord, D: Ord, E: Ord { .. } -``` - ---- - -## `std::ops` - -### `std::ops::Add`, `std::ops::Sub`, `std::ops::Mul`, and `std::ops::Div` - -These traits abstract over addition, subtraction, multiplication, and division respectively. -Implementing these traits for a given type will also allow that type to be used with the corresponding operator -for that trait (`+` for Add, etc) in addition to the normal method names. - -```rust title="add-trait" showLineNumbers -trait Add { - fn add(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L1-L5 - -```rust title="sub-trait" showLineNumbers -trait Sub { - fn sub(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L60-L64 - -```rust title="mul-trait" showLineNumbers -trait Mul { - fn mul(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L119-L123 - -```rust title="div-trait" showLineNumbers -trait Div { - fn div(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L178-L182 - - -The implementations block below is given for the `Add` trait, but the same types that implement -`Add` also implement `Sub`, `Mul`, and `Div`. - -Implementations: -```rust -impl Add for Field { .. } - -impl Add for i8 { .. } -impl Add for i16 { .. } -impl Add for i32 { .. } -impl Add for i64 { .. } - -impl Add for u8 { .. } -impl Add for u16 { .. } -impl Add for u32 { .. } -impl Add for u64 { .. } -``` - -### `std::ops::Rem` - -```rust title="rem-trait" showLineNumbers -trait Rem{ - fn rem(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L237-L241 - - -`Rem::rem(a, b)` is the remainder function returning the result of what is -left after dividing `a` and `b`. Implementing `Rem` allows the `%` operator -to be used with the implementation type. - -Unlike other numeric traits, `Rem` is not implemented for `Field`. - -Implementations: -```rust -impl Rem for u8 { fn rem(self, other: u8) -> u8 { self % other } } -impl Rem for u16 { fn rem(self, other: u16) -> u16 { self % other } } -impl Rem for u32 { fn rem(self, other: u32) -> u32 { self % other } } -impl Rem for u64 { fn rem(self, other: u64) -> u64 { self % other } } - -impl Rem for i8 { fn rem(self, other: i8) -> i8 { self % other } } -impl Rem for i16 { fn rem(self, other: i16) -> i16 { self % other } } -impl Rem for i32 { fn rem(self, other: i32) -> i32 { self % other } } -impl Rem for i64 { fn rem(self, other: i64) -> i64 { self % other } } -``` - -### `std::ops::Neg` - -```rust title="neg-trait" showLineNumbers -trait Neg { - fn neg(self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L290-L294 - - -`Neg::neg` is equivalent to the unary negation operator `-`. - -Implementations: -```rust title="neg-trait-impls" showLineNumbers -impl Neg for Field { - fn neg(self) -> Field { - -self - } -} - -impl Neg for i8 { - fn neg(self) -> i8 { - -self - } -} -impl Neg for i16 { - fn neg(self) -> i16 { - -self - } -} -impl Neg for i32 { - fn neg(self) -> i32 { - -self - } -} -impl Neg for i64 { - fn neg(self) -> i64 { - -self - } -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L296-L323 - - -### `std::ops::Not` - -```rust title="not-trait" showLineNumbers -trait Not { - fn not(self: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L1-L5 - - -`Not::not` is equivalent to the unary bitwise NOT operator `!`. - -Implementations: -```rust title="not-trait-impls" showLineNumbers -impl Not for bool { - fn not(self) -> bool { - !self - } -} - -impl Not for u64 { - fn not(self) -> u64 { - !self - } -} -impl Not for u32 { - fn not(self) -> u32 { - !self - } -} -impl Not for u16 { - fn not(self) -> u16 { - !self - } -} -impl Not for u8 { - fn not(self) -> u8 { - !self - } -} -impl Not for u1 { - fn not(self) -> u1 { - !self - } -} - -impl Not for i8 { - fn not(self) -> i8 { - !self - } -} -impl Not for i16 { - fn not(self) -> i16 { - !self - } -} -impl Not for i32 { - fn not(self) -> i32 { - !self - } -} -impl Not for i64 { - fn not(self) -> i64 { - !self - } -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L7-L60 - - -### `std::ops::{ BitOr, BitAnd, BitXor }` - -```rust title="bitor-trait" showLineNumbers -trait BitOr { - fn bitor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L62-L66 - -```rust title="bitand-trait" showLineNumbers -trait BitAnd { - fn bitand(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L121-L125 - -```rust title="bitxor-trait" showLineNumbers -trait BitXor { - fn bitxor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L180-L184 - - -Traits for the bitwise operations `|`, `&`, and `^`. - -Implementing `BitOr`, `BitAnd` or `BitXor` for a type allows the `|`, `&`, or `^` operator respectively -to be used with the type. - -The implementations block below is given for the `BitOr` trait, but the same types that implement -`BitOr` also implement `BitAnd` and `BitXor`. - -Implementations: -```rust -impl BitOr for bool { fn bitor(self, other: bool) -> bool { self | other } } - -impl BitOr for u8 { fn bitor(self, other: u8) -> u8 { self | other } } -impl BitOr for u16 { fn bitor(self, other: u16) -> u16 { self | other } } -impl BitOr for u32 { fn bitor(self, other: u32) -> u32 { self | other } } -impl BitOr for u64 { fn bitor(self, other: u64) -> u64 { self | other } } - -impl BitOr for i8 { fn bitor(self, other: i8) -> i8 { self | other } } -impl BitOr for i16 { fn bitor(self, other: i16) -> i16 { self | other } } -impl BitOr for i32 { fn bitor(self, other: i32) -> i32 { self | other } } -impl BitOr for i64 { fn bitor(self, other: i64) -> i64 { self | other } } -``` - -### `std::ops::{ Shl, Shr }` - -```rust title="shl-trait" showLineNumbers -trait Shl { - fn shl(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L239-L243 - -```rust title="shr-trait" showLineNumbers -trait Shr { - fn shr(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L292-L296 - - -Traits for a bit shift left and bit shift right. - -Implementing `Shl` for a type allows the left shift operator (`<<`) to be used with the implementation type. -Similarly, implementing `Shr` allows the right shift operator (`>>`) to be used with the type. - -Note that bit shifting is not currently implemented for signed types. - -The implementations block below is given for the `Shl` trait, but the same types that implement -`Shl` also implement `Shr`. - -Implementations: -```rust -impl Shl for u8 { fn shl(self, other: u8) -> u8 { self << other } } -impl Shl for u16 { fn shl(self, other: u16) -> u16 { self << other } } -impl Shl for u32 { fn shl(self, other: u32) -> u32 { self << other } } -impl Shl for u64 { fn shl(self, other: u64) -> u64 { self << other } } -``` - ---- - -## `std::append` - -### `std::append::Append` - -`Append` can abstract over types that can be appended to - usually container types: - -```rust title="append-trait" showLineNumbers -trait Append { - fn empty() -> Self; - fn append(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/append.nr#L9-L14 - - -`Append` requires two methods: - -- `empty`: Constructs an empty value of `Self`. -- `append`: Append two values together, returning the result. - -Additionally, it is expected that for any implementation: - -- `T::empty().append(x) == x` -- `x.append(T::empty()) == x` - -Implementations: -```rust -impl Append for [T] -impl Append for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/zeroed.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/zeroed.md deleted file mode 100644 index f450fecdd36..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/zeroed.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Zeroed Function -description: - The zeroed function returns a zeroed value of any type. -keywords: - [ - zeroed - ] ---- - -Implements `fn zeroed() -> T` to return a zeroed value of any type. This function is generally unsafe to use as the zeroed bit pattern is not guaranteed to be valid for all types. It can however, be useful in cases when the value is guaranteed not to be used such as in a BoundedVec library implementing a growable vector, up to a certain length, backed by an array. The array can be initialized with zeroed values which are guaranteed to be inaccessible until the vector is pushed to. Similarly, enumerations in noir can be implemented using this method by providing zeroed values for the unused variants. - -You can access the function at `std::unsafe::zeroed`. - -This function currently supports the following types: - -- Field -- Bool -- Uint -- Array -- Slice -- String -- Tuple -- Function - -Using it on other types could result in unexpected behavior. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md deleted file mode 100644 index 42f065f4a4e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md +++ /dev/null @@ -1,141 +0,0 @@ -# BarretenbergBackend - -## Implements - -- [`Backend`](../index.md#backend) -- [`Backend`](../index.md#backend) - -## Constructors - -### new BarretenbergBackend(acirCircuit, options) - -```ts -new BarretenbergBackend(acirCircuit, options): BarretenbergBackend -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `acirCircuit` | `CompiledCircuit` | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`BarretenbergBackend`](BarretenbergBackend.md) - -## Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `acirComposer` | `any` | - | -| `acirUncompressedBytecode` | `Uint8Array` | - | -| `api` | `Barretenberg` | - | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - | - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### generateProof() - -```ts -generateProof(compressedWitness): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `compressedWitness` | `Uint8Array` | - -#### Returns - -`Promise`\<`ProofData`\> - -#### Description - -Generates a proof - -*** - -### generateRecursiveProofArtifacts() - -```ts -generateRecursiveProofArtifacts(proofData, numOfPublicInputs): Promise -``` - -Generates artifacts that will be passed to a circuit that will verify this proof. - -Instead of passing the proof and verification key as a byte array, we pass them -as fields which makes it cheaper to verify in a circuit. - -The proof that is passed here will have been created using a circuit -that has the #[recursive] attribute on its `main` method. - -The number of public inputs denotes how many public inputs are in the inner proof. - -#### Parameters - -| Parameter | Type | Default value | -| :------ | :------ | :------ | -| `proofData` | `ProofData` | `undefined` | -| `numOfPublicInputs` | `number` | `0` | - -#### Returns - -`Promise`\<`object`\> - -#### Example - -```typescript -const artifacts = await backend.generateRecursiveProofArtifacts(proof, numOfPublicInputs); -``` - -*** - -### getVerificationKey() - -```ts -getVerificationKey(): Promise -``` - -#### Returns - -`Promise`\<`Uint8Array`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md deleted file mode 100644 index 500276ea748..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md +++ /dev/null @@ -1,58 +0,0 @@ -# BarretenbergVerifier - -## Constructors - -### new BarretenbergVerifier(options) - -```ts -new BarretenbergVerifier(options): BarretenbergVerifier -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`BarretenbergVerifier`](BarretenbergVerifier.md) - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData, verificationKey): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | -| `verificationKey` | `Uint8Array` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md deleted file mode 100644 index be1cd9ad465..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md +++ /dev/null @@ -1,116 +0,0 @@ -# UltraHonkBackend - -## Implements - -- [`Backend`](../index.md#backend) -- [`Backend`](../index.md#backend) - -## Constructors - -### new UltraHonkBackend(acirCircuit, options) - -```ts -new UltraHonkBackend(acirCircuit, options): UltraHonkBackend -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `acirCircuit` | `CompiledCircuit` | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`UltraHonkBackend`](UltraHonkBackend.md) - -## Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `acirUncompressedBytecode` | `Uint8Array` | - | -| `api` | `Barretenberg` | - | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - | - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### generateProof() - -```ts -generateProof(decompressedWitness): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `decompressedWitness` | `Uint8Array` | - -#### Returns - -`Promise`\<`ProofData`\> - -*** - -### generateRecursiveProofArtifacts() - -```ts -generateRecursiveProofArtifacts(_proofData, _numOfPublicInputs): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `_proofData` | `ProofData` | -| `_numOfPublicInputs` | `number` | - -#### Returns - -`Promise`\<`object`\> - -*** - -### getVerificationKey() - -```ts -getVerificationKey(): Promise -``` - -#### Returns - -`Promise`\<`Uint8Array`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | - -#### Returns - -`Promise`\<`boolean`\> - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md deleted file mode 100644 index aee9460153f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md +++ /dev/null @@ -1,58 +0,0 @@ -# UltraHonkVerifier - -## Constructors - -### new UltraHonkVerifier(options) - -```ts -new UltraHonkVerifier(options): UltraHonkVerifier -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`UltraHonkVerifier`](UltraHonkVerifier.md) - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData, verificationKey): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | -| `verificationKey` | `Uint8Array` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/index.md deleted file mode 100644 index 4699e16dee6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/index.md +++ /dev/null @@ -1,42 +0,0 @@ -# backend_barretenberg - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [BarretenbergBackend](classes/BarretenbergBackend.md) | - | -| [BarretenbergVerifier](classes/BarretenbergVerifier.md) | - | -| [UltraHonkBackend](classes/UltraHonkBackend.md) | - | -| [UltraHonkVerifier](classes/UltraHonkVerifier.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [BackendOptions](type-aliases/BackendOptions.md) | - | - -## References - -### CompiledCircuit - -Renames and re-exports [Backend](index.md#backend) - -*** - -### ProofData - -Renames and re-exports [Backend](index.md#backend) - -## Variables - -### Backend - -```ts -Backend: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md deleted file mode 100644 index b49a479f4f4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md +++ /dev/null @@ -1,21 +0,0 @@ -# BackendOptions - -```ts -type BackendOptions: object; -``` - -## Description - -An options object, currently only used to specify the number of threads to use. - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `memory` | `object` | - | -| `memory.maximum` | `number` | - | -| `threads` | `number` | **Description**

Number of threads | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs deleted file mode 100644 index 8ecf05c0163..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend","label":"BarretenbergBackend"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier","label":"BarretenbergVerifier"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend","label":"UltraHonkBackend"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier","label":"UltraHonkVerifier"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions","label":"BackendOptions"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/classes/Noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/classes/Noir.md deleted file mode 100644 index ead255bc504..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/classes/Noir.md +++ /dev/null @@ -1,52 +0,0 @@ -# Noir - -## Constructors - -### new Noir(circuit) - -```ts -new Noir(circuit): Noir -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `circuit` | `CompiledCircuit` | - -#### Returns - -[`Noir`](Noir.md) - -## Methods - -### execute() - -```ts -execute(inputs, foreignCallHandler?): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `inputs` | `InputMap` | -| `foreignCallHandler`? | [`ForeignCallHandler`](../type-aliases/ForeignCallHandler.md) | - -#### Returns - -`Promise`\<`object`\> - -#### Description - -Allows to execute a circuit to get its witness and return value. - -#### Example - -```typescript -async execute(inputs) -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/and.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/and.md deleted file mode 100644 index c783283e396..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/and.md +++ /dev/null @@ -1,22 +0,0 @@ -# and() - -```ts -and(lhs, rhs): string -``` - -Performs a bitwise AND operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/blake2s256.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/blake2s256.md deleted file mode 100644 index 7882d0da8d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/blake2s256.md +++ /dev/null @@ -1,21 +0,0 @@ -# blake2s256() - -```ts -blake2s256(inputs): Uint8Array -``` - -Calculates the Blake2s256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md deleted file mode 100644 index 5e3cd53e9d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256k1\_verify() - -```ts -ecdsa_secp256k1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256k1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md deleted file mode 100644 index 0b20ff68957..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256r1\_verify() - -```ts -ecdsa_secp256r1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256r1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/keccak256.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/keccak256.md deleted file mode 100644 index d10f155ce86..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/keccak256.md +++ /dev/null @@ -1,21 +0,0 @@ -# keccak256() - -```ts -keccak256(inputs): Uint8Array -``` - -Calculates the Keccak256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/sha256.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/sha256.md deleted file mode 100644 index 6ba4ecac022..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/sha256.md +++ /dev/null @@ -1,21 +0,0 @@ -# sha256() - -```ts -sha256(inputs): Uint8Array -``` - -Calculates the SHA256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/xor.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/xor.md deleted file mode 100644 index 8d762b895d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/functions/xor.md +++ /dev/null @@ -1,22 +0,0 @@ -# xor() - -```ts -xor(lhs, rhs): string -``` - -Performs a bitwise XOR operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/index.md deleted file mode 100644 index 166508f7124..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_js - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [Noir](classes/Noir.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [ErrorWithPayload](type-aliases/ErrorWithPayload.md) | - | -| [ForeignCallHandler](type-aliases/ForeignCallHandler.md) | A callback which performs an foreign call and returns the response. | -| [ForeignCallInput](type-aliases/ForeignCallInput.md) | - | -| [ForeignCallOutput](type-aliases/ForeignCallOutput.md) | - | -| [WitnessMap](type-aliases/WitnessMap.md) | - | - -### Functions - -| Function | Description | -| :------ | :------ | -| [and](functions/and.md) | Performs a bitwise AND operation between `lhs` and `rhs` | -| [blake2s256](functions/blake2s256.md) | Calculates the Blake2s256 hash of the input bytes | -| [ecdsa\_secp256k1\_verify](functions/ecdsa_secp256k1_verify.md) | Verifies a ECDSA signature over the secp256k1 curve. | -| [ecdsa\_secp256r1\_verify](functions/ecdsa_secp256r1_verify.md) | Verifies a ECDSA signature over the secp256r1 curve. | -| [keccak256](functions/keccak256.md) | Calculates the Keccak256 hash of the input bytes | -| [sha256](functions/sha256.md) | Calculates the SHA256 hash of the input bytes | -| [xor](functions/xor.md) | Performs a bitwise XOR operation between `lhs` and `rhs` | - -## References - -### CompiledCircuit - -Renames and re-exports [InputMap](index.md#inputmap) - -## Variables - -### InputMap - -```ts -InputMap: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md deleted file mode 100644 index e8c2f4aef3d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md +++ /dev/null @@ -1,15 +0,0 @@ -# ErrorWithPayload - -```ts -type ErrorWithPayload: ExecutionError & object; -``` - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `decodedAssertionPayload` | `any` | - | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md deleted file mode 100644 index 812b8b16481..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md +++ /dev/null @@ -1,24 +0,0 @@ -# ForeignCallHandler - -```ts -type ForeignCallHandler: (name, inputs) => Promise; -``` - -A callback which performs an foreign call and returns the response. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | The identifier for the type of foreign call being performed. | -| `inputs` | [`ForeignCallInput`](ForeignCallInput.md)[] | An array of hex encoded inputs to the foreign call. | - -## Returns - -`Promise`\<[`ForeignCallOutput`](ForeignCallOutput.md)[]\> - -outputs - An array of hex encoded outputs containing the results of the foreign call. - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md deleted file mode 100644 index dd95809186a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallInput - -```ts -type ForeignCallInput: string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md deleted file mode 100644 index b71fb78a946..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallOutput - -```ts -type ForeignCallOutput: string | string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md deleted file mode 100644 index 258c46f9d0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md +++ /dev/null @@ -1,9 +0,0 @@ -# WitnessMap - -```ts -type WitnessMap: Map; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs deleted file mode 100644 index b3156097df6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/noir_js/classes/Noir","label":"Noir"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ErrorWithPayload","label":"ErrorWithPayload"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallHandler","label":"ForeignCallHandler"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallInput","label":"ForeignCallInput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallOutput","label":"ForeignCallOutput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/WitnessMap","label":"WitnessMap"}]},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_js/functions/and","label":"and"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/blake2s256","label":"blake2s256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify","label":"ecdsa_secp256k1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify","label":"ecdsa_secp256r1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/keccak256","label":"keccak256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/sha256","label":"sha256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/xor","label":"xor"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/compile.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/compile.md deleted file mode 100644 index 6faf763b37f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/compile.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile() - -```ts -compile( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ProgramCompilationArtifacts`](../index.md#programcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_program(fm); -``` - -```typescript -// Browser - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_program(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/compile_contract.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/compile_contract.md deleted file mode 100644 index 7d0b39a43ef..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/compile_contract.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile\_contract() - -```ts -compile_contract( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ContractCompilationArtifacts`](../index.md#contractcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_contract(fm); -``` - -```typescript -// Browser - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_contract(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/createFileManager.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/createFileManager.md deleted file mode 100644 index 7e65c1d69c7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/createFileManager.md +++ /dev/null @@ -1,21 +0,0 @@ -# createFileManager() - -```ts -createFileManager(dataDir): FileManager -``` - -Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `dataDir` | `string` | root of the file system | - -## Returns - -`FileManager` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md deleted file mode 100644 index fcea9275341..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md +++ /dev/null @@ -1,21 +0,0 @@ -# inflateDebugSymbols() - -```ts -inflateDebugSymbols(debugSymbols): any -``` - -Decompresses and decodes the debug symbols - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `debugSymbols` | `string` | The base64 encoded debug symbols | - -## Returns - -`any` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/index.md deleted file mode 100644 index b6e0f9d1bc0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_wasm - -## Exports - -### Functions - -| Function | Description | -| :------ | :------ | -| [compile](functions/compile.md) | Compiles a Noir project | -| [compile\_contract](functions/compile_contract.md) | Compiles a Noir project | -| [createFileManager](functions/createFileManager.md) | Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) | -| [inflateDebugSymbols](functions/inflateDebugSymbols.md) | Decompresses and decodes the debug symbols | - -## References - -### compile\_program - -Renames and re-exports [compile](functions/compile.md) - -## Interfaces - -### ContractCompilationArtifacts - -The compilation artifacts of a given contract. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `contract` | `ContractArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -### ProgramCompilationArtifacts - -The compilation artifacts of a given program. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | not part of the compilation output, injected later | -| `program` | `ProgramArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs deleted file mode 100644 index e0870710349..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"doc","id":"reference/NoirJS/noir_wasm/index","label":"API"},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile","label":"compile"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile_contract","label":"compile_contract"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/createFileManager","label":"createFileManager"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/inflateDebugSymbols","label":"inflateDebugSymbols"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/_category_.json deleted file mode 100644 index 5b6a20a609a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 4, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/_category_.json deleted file mode 100644 index 27869205ad3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugger", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_known_limitations.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_known_limitations.md deleted file mode 100644 index 936d416ac4b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_known_limitations.md +++ /dev/null @@ -1,59 +0,0 @@ ---- -title: Known limitations -description: - An overview of known limitations of the current version of the Noir debugger -keywords: - [ - Nargo, - Noir Debugger, - VS Code, - ] -sidebar_position: 2 ---- - -# Debugger Known Limitations - -There are currently some limits to what the debugger can observe. - -## Mutable references - -The debugger is currently blind to any state mutated via a mutable reference. For example, in: - -``` -let mut x = 1; -let y = &mut x; -*y = 2; -``` - -The update on `x` will not be observed by the debugger. That means, when running `vars` from the debugger REPL, or inspecting the _local variables_ pane in the VS Code debugger, `x` will appear with value 1 despite having executed `*y = 2;`. - -## Variables of type function or mutable references are opaque - -When inspecting variables, any variable of type `Function` or `MutableReference` will render its value as `<>` or `<>`. - -## Debugger instrumentation affects resulting ACIR - -In order to make the state of local variables observable, the debugger compiles Noir circuits interleaving foreign calls that track any mutations to them. While this works (except in the cases described above) and doesn't introduce any behavior changes, it does as a side effect produce bigger bytecode. In particular, when running the command `opcodes` on the REPL debugger, you will notice Unconstrained VM blocks that look like this: - -``` -... -5 BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [], q_c: 2 }), Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(2))], q_c: 0 })] - | outputs=[] - 5.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 5.1 | Mov { destination: RegisterIndex(3), source: RegisterIndex(1) } - 5.2 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 5.3 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 5.4 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 5.5 | Mov { destination: RegisterIndex(3), source: RegisterIndex(3) } - 5.6 | Call { location: 8 } - 5.7 | Stop - 5.8 | ForeignCall { function: "__debug_var_assign", destinations: [], inputs: [RegisterIndex(RegisterIndex(2)), RegisterIndex(RegisterIndex(3))] } -... -``` - -If you are interested in debugging/inspecting compiled ACIR without these synthetic changes, you can invoke the REPL debugger with the `--skip-instrumentation` flag or launch the VS Code debugger with the `skipConfiguration` property set to true in its launch configuration. You can find more details about those in the [Debugger REPL reference](debugger_repl.md) and the [VS Code Debugger reference](debugger_vscode.md). - -:::note -Skipping debugger instrumentation means you won't be able to inspect values of local variables. -::: - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_repl.md deleted file mode 100644 index 46e2011304e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_repl.md +++ /dev/null @@ -1,360 +0,0 @@ ---- -title: REPL Debugger -description: - Noir Debugger REPL options and commands. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -## Running the REPL debugger - -`nargo debug [OPTIONS] [WITNESS_NAME]` - -Runs the Noir REPL debugger. If a `WITNESS_NAME` is provided the debugger writes the resulting execution witness to a `WITNESS_NAME` file. - -### Options - -| Option | Description | -| --------------------- | ------------------------------------------------------------ | -| `-p, --prover-name ` | The name of the toml file which contains the inputs for the prover [default: Prover]| -| `--package ` | The name of the package to debug | -| `--print-acir` | Display the ACIR for compiled circuit | -| `--deny-warnings` | Treat all warnings as errors | -| `--silence-warnings` | Suppress warnings | -| `-h, --help` | Print help | - -None of these options are required. - -:::note -Since the debugger starts by compiling the target package, all Noir compiler options are also available. Check out the [compiler reference](../nargo_commands.md#nargo-compile) to learn more about the compiler options. -::: - -## REPL commands - -Once the debugger is running, it accepts the following commands. - -#### `help` (h) - -Displays the menu of available commands. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) value - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -### Stepping through programs - -#### `next` (n) - -Step until the next Noir source code location. While other commands, such as [`into`](#into-i) and [`step`](#step-s), allow for finer grained control of the program's execution at the opcode level, `next` is source code centric. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `next` here would cause the debugger to jump to the definition of `deep_entry_point` (if available). - -If you want to step over `deep_entry_point` and go straight to line 8, use [the `over` command](#over) instead. - -#### `over` - -Step until the next source code location, without diving into function calls. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `over` here would cause the debugger to execute until line 8 (`multiple_values_entry_point(x);`). - -If you want to step into `deep_entry_point` instead, use [the `next` command](#next-n). - -#### `out` - -Step until the end of the current function call. For example: - -``` - 3 ... - 4 fn main(x: u32) { - 5 assert(entry_point(x) == 2); - 6 swap_entry_point(x, x + 1); - 7 -> assert(deep_entry_point(x) == 4); - 8 multiple_values_entry_point(x); - 9 } - 10 - 11 unconstrained fn returns_multiple_values(x: u32) -> (u32, u32, u32, u32) { - 12 ... - ... - 55 - 56 unconstrained fn deep_entry_point(x: u32) -> u32 { - 57 -> level_1(x + 1) - 58 } - -``` - -Running `out` here will resume execution until line 8. - -#### `step` (s) - -Skips to the next ACIR code. A compiled Noir program is a sequence of ACIR opcodes. However, an unconstrained VM opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `step` command at this point would result in the debugger stopping at ACIR opcode 2, `EXPR`, skipping unconstrained computation steps. - -Use [the `into` command](#into-i) instead if you want to follow unconstrained computation step by step. - -#### `into` (i) - -Steps into the next opcode. A compiled Noir program is a sequence of ACIR opcodes. However, a BRILLIG opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `into` command at this point would result in the debugger stopping at opcode 1.0, `Mov ...`, allowing the debugger user to follow unconstrained computation step by step. - -Use [the `step` command](#step-s) instead if you want to skip to the next ACIR code directly. - -#### `continue` (c) - -Continues execution until the next breakpoint, or the end of the program. - -#### `restart` (res) - -Interrupts execution, and restarts a new debugging session from scratch. - -#### `opcodes` (o) - -Display the program's ACIR opcode sequence. For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -### Breakpoints - -#### `break [Opcode]` (or shorthand `b [Opcode]`) - -Sets a breakpoint on the specified opcode index. To get a list of the program opcode numbers, see [the `opcode` command](#opcodes-o). For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -In this example, issuing a `break 1.2` command adds break on opcode 1.2, as denoted by the `*` character: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | * Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -Running [the `continue` command](#continue-c) at this point would cause the debugger to execute the program until opcode 1.2. - -#### `delete [Opcode]` (or shorthand `d [Opcode]`) - -Deletes a breakpoint at an opcode location. Usage is analogous to [the `break` command](#). - -### Variable inspection - -#### vars - -Show variable values available at this point in execution. - -:::note -The ability to inspect variable values from the debugger depends on compilation to be run in a special debug instrumentation mode. This instrumentation weaves variable tracing code with the original source code. - -So variable value inspection comes at the expense of making the resulting ACIR bytecode bigger and harder to understand and optimize. - -If you find this compromise unacceptable, you can run the debugger with the flag `--skip-debug-instrumentation`. This will compile your circuit without any additional debug information, so the resulting ACIR bytecode will be identical to the one produced by standard Noir compilation. However, if you opt for this, the `vars` command will not be available while debugging. -::: - - -### Stacktrace - -#### `stacktrace` - -Displays the current stack trace. - - -### Witness map - -#### `witness` (w) - -Show witness map. For example: - -``` -_0 = 0 -_1 = 2 -_2 = 1 -``` - -#### `witness [Witness Index]` - -Display a single witness from the witness map. For example: - -``` -> witness 1 -_1 = 2 -``` - -#### `witness [Witness Index] [New value]` - -Overwrite the given index with a new value. For example: - -``` -> witness 1 3 -_1 = 3 -``` - - -### Unconstrained VM memory - -#### `memory` - -Show unconstrained VM memory state. For example: - -``` -> memory -At opcode 1.13: Store { destination_pointer: RegisterIndex(0), source: RegisterIndex(3) } -... -> registers -0 = 0 -1 = 10 -2 = 0 -3 = 1 -4 = 1 -5 = 2³² -6 = 1 -> into -At opcode 1.14: Const { destination: RegisterIndex(5), value: Value { inner: 1 } } -... -> memory -0 = 1 -> -``` - -In the example above: we start with clean memory, then step through a `Store` opcode which stores the value of register 3 (1) into the memory address stored in register 0 (0). Thus now `memory` shows memory address 0 contains value 1. - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: - -#### `memset [Memory address] [New value]` - -Update a memory cell with the given value. For example: - -``` -> memory -0 = 1 -> memset 0 2 -> memory -0 = 2 -> memset 1 4 -> memory -0 = 2 -1 = 4 -> -``` - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_vscode.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_vscode.md deleted file mode 100644 index c027332b3b0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/debugger/debugger_vscode.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: VS Code Debugger -description: - VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -# VS Code Noir Debugger Reference - -The Noir debugger enabled by the vscode-noir extension ships with default settings such that the most common scenario should run without any additional configuration steps. - -These defaults can nevertheless be overridden by defining a launch configuration file. This page provides a reference for the properties you can override via a launch configuration file, as well as documenting the Nargo `dap` command, which is a dependency of the VS Code Noir debugger. - - -## Creating and editing launch configuration files - -To create a launch configuration file from VS Code, open the _debug pane_, and click on _create a launch.json file_. - -![Creating a launch configuration file](@site/static/img/debugger/ref1-create-launch.png) - -A `launch.json` file will be created, populated with basic defaults. - -### Noir Debugger launch.json properties - -#### projectFolder - -_String, optional._ - -Absolute path to the Nargo project to debug. By default, it is dynamically determined by looking for the nearest `Nargo.toml` file to the active file at the moment of launching the debugger. - -#### proverName - -_String, optional._ - -Name of the prover input to use. Defaults to `Prover`, which looks for a file named `Prover.toml` at the `projectFolder`. - -#### generateAcir - -_Boolean, optional._ - -If true, generate ACIR opcodes instead of unconstrained opcodes which will be closer to release binaries but less convenient for debugging. Defaults to `false`. - -#### skipInstrumentation - -_Boolean, optional._ - -Skips variables debugging instrumentation of code, making debugging less convenient but the resulting binary smaller and closer to production. Defaults to `false`. - -:::note -Skipping instrumentation causes the debugger to be unable to inspect local variables. -::: - -## `nargo dap [OPTIONS]` - -When run without any option flags, it starts the Nargo Debug Adapter Protocol server, which acts as the debugging backend for the VS Code Noir Debugger. - -All option flags are related to preflight checks. The Debug Adapter Protocol specifies how errors are to be informed from a running DAP server, but it doesn't specify mechanisms to communicate server initialization errors between the DAP server and its client IDE. - -Thus `nargo dap` ships with a _preflight check_ mode. If flag `--preflight-check` and the rest of the `--preflight-*` flags are provided, Nargo will run the same initialization routine except it will not start the DAP server. - -`vscode-noir` will then run `nargo dap` in preflight check mode first before a debugging session starts. If the preflight check ends in error, vscode-noir will present stderr and stdout output from this process through its own Output pane in VS Code. This makes it possible for users to diagnose what pieces of configuration might be wrong or missing in case of initialization errors. - -If the preflight check succeeds, `vscode-noir` proceeds to start the DAP server normally but running `nargo dap` without any additional flags. - -### Options - -| Option | Description | -| --------------------------------------------------------- | --------------------------------------------------------------------------------------------------------- | -| `--preflight-check` | If present, dap runs in preflight check mode. | -| `--preflight-project-folder ` | Absolute path to the project to debug for preflight check. | -| `--preflight-prover-name ` | Name of prover file to use for preflight check | -| `--preflight-generate-acir` | Optional. If present, compile in ACIR mode while running preflight check. | -| `--preflight-skip-instrumentation` | Optional. If present, compile without introducing debug instrumentation while running preflight check. | -| `-h, --help` | Print help. | diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/nargo_commands.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/nargo_commands.md deleted file mode 100644 index b46ad4cda98..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/nargo_commands.md +++ /dev/null @@ -1,295 +0,0 @@ ---- -title: Nargo -description: - Noir CLI Commands for Noir Prover and Verifier to create, execute, prove and verify programs, - generate Solidity verifier smart contract and compile into JSON file containing ACIR - representation and ABI of circuit. -keywords: - [ - Nargo, - Noir CLI, - Noir Prover, - Noir Verifier, - generate Solidity verifier, - compile JSON file, - ACIR representation, - ABI of circuit, - TypeScript, - ] -sidebar_position: 0 ---- - -# Command-Line Help for `nargo` - -This document contains the help content for the `nargo` command-line program. - -**Command Overview:** - -* [`nargo`↴](#nargo) -* [`nargo check`↴](#nargo-check) -* [`nargo fmt`↴](#nargo-fmt) -* [`nargo compile`↴](#nargo-compile) -* [`nargo new`↴](#nargo-new) -* [`nargo init`↴](#nargo-init) -* [`nargo execute`↴](#nargo-execute) -* [`nargo debug`↴](#nargo-debug) -* [`nargo test`↴](#nargo-test) -* [`nargo info`↴](#nargo-info) -* [`nargo lsp`↴](#nargo-lsp) - -## `nargo` - -Noir's package manager - -**Usage:** `nargo ` - -###### **Subcommands:** - -* `check` — Checks the constraint system for errors -* `fmt` — Format the Noir files in a workspace -* `compile` — Compile the program and its secret execution trace into ACIR format -* `new` — Create a Noir project in a new directory -* `init` — Create a Noir project in the current directory -* `execute` — Executes a circuit to calculate its return value -* `debug` — Executes a circuit in debug mode -* `test` — Run the tests for this program -* `info` — Provides detailed information on each of a program's function (represented by a single circuit) -* `lsp` — Starts the Noir LSP server - -###### **Options:** - - - - -## `nargo check` - -Checks the constraint system for errors - -**Usage:** `nargo check [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to check -* `--workspace` — Check all packages in the workspace -* `--overwrite` — Force overwrite of existing files -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo fmt` - -Format the Noir files in a workspace - -**Usage:** `nargo fmt [OPTIONS]` - -###### **Options:** - -* `--check` — Run noirfmt in check mode - - - -## `nargo compile` - -Compile the program and its secret execution trace into ACIR format - -**Usage:** `nargo compile [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to compile -* `--workspace` — Compile all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo new` - -Create a Noir project in a new directory - -**Usage:** `nargo new [OPTIONS] ` - -###### **Arguments:** - -* `` — The path to save the new project - -###### **Options:** - -* `--name ` — Name of the package [default: package directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo init` - -Create a Noir project in the current directory - -**Usage:** `nargo init [OPTIONS]` - -###### **Options:** - -* `--name ` — Name of the package [default: current directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo execute` - -Executes a circuit to calculate its return value - -**Usage:** `nargo execute [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--workspace` — Execute all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo debug` - -Executes a circuit in debug mode - -**Usage:** `nargo debug [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--acir-mode` — Force ACIR output (disabling instrumentation) -* `--skip-instrumentation ` — Disable vars debug instrumentation (enabled by default) - - Possible values: `true`, `false` - - - - -## `nargo test` - -Run the tests for this program - -**Usage:** `nargo test [OPTIONS] [TEST_NAME]` - -###### **Arguments:** - -* `` — If given, only tests with names containing this string will be run - -###### **Options:** - -* `--show-output` — Display output of `println` statements -* `--exact` — Only run tests that match exactly -* `--package ` — The name of the package to test -* `--workspace` — Test all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo info` - -Provides detailed information on each of a program's function (represented by a single circuit) - -Current information provided per circuit: 1. The number of ACIR opcodes 2. Counts the final number gates in the circuit used by a backend - -**Usage:** `nargo info [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to detail -* `--workspace` — Detail all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo lsp` - -Starts the Noir LSP server - -Starts an LSP server which allows IDEs such as VS Code to display diagnostics in Noir source. - -VS Code Noir Language Support: https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir - -**Usage:** `nargo lsp` - - - -
- - - This document was generated automatically by - clap-markdown. - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/noir_codegen.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/noir_codegen.md deleted file mode 100644 index db8f07dc22e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/reference/noir_codegen.md +++ /dev/null @@ -1,114 +0,0 @@ ---- -title: Noir Codegen for TypeScript -description: Learn how to use Noir codegen to generate TypeScript bindings -keywords: [Nargo, Noir, compile, TypeScript] -sidebar_position: 3 ---- - -When using TypeScript, it is extra work to interpret Noir program outputs in a type-safe way. Third party libraries may exist for popular Noir programs, but they are either hard to find or unmaintained. - -Now you can generate TypeScript bindings for your Noir programs in two steps: -1. Exporting Noir functions using `nargo export` -2. Using the TypeScript module `noir_codegen` to generate TypeScript binding - -**Note:** you can only export functions from a Noir *library* (not binary or contract program types). - -## Installation - -### Your TypeScript project - -If you don't already have a TypeScript project you can add the module with `yarn` (or `npm`), then initialize it: - -```bash -yarn add typescript -D -npx tsc --init -``` - -### Add TypeScript module - `noir_codegen` - -The following command will add the module to your project's devDependencies: - -```bash -yarn add @noir-lang/noir_codegen -D -``` - -### Nargo library -Make sure you have Nargo, v0.25.0 or greater, installed. If you don't, follow the [installation guide](../getting_started/installation/index.md). - -If you're in a new project, make a `circuits` folder and create a new Noir library: - -```bash -mkdir circuits && cd circuits -nargo new --lib myNoirLib -``` - -## Usage - -### Export ABI of specified functions - -First go to the `.nr` files in your Noir library, and add the `#[export]` macro to each function that you want to use in TypeScript. - -```rust -#[export] -fn your_function(... -``` - -From your Noir library (where `Nargo.toml` is), run the following command: - -```bash -nargo export -``` - -You will now have an `export` directory with a .json file per exported function. - -You can also specify the directory of Noir programs using `--program-dir`, for example: - -```bash -nargo export --program-dir=./circuits/myNoirLib -``` - -### Generate TypeScript bindings from exported functions - -To use the `noir-codegen` package we added to the TypeScript project: - -```bash -yarn noir-codegen ./export/your_function.json -``` - -This creates an `exports` directory with an `index.ts` file containing all exported functions. - -**Note:** adding `--out-dir` allows you to specify an output dir for your TypeScript bindings to go. Eg: - -```bash -yarn noir-codegen ./export/*.json --out-dir ./path/to/output/dir -``` - -## Example .nr function to .ts output - -Consider a Noir library with this function: - -```rust -#[export] -fn not_equal(x: Field, y: Field) -> bool { - x != y -} -``` - -After the export and codegen steps, you should have an `index.ts` like: - -```typescript -export type Field = string; - - -export const is_equal_circuit: CompiledCircuit = -{"abi":{"parameters":[{"name":"x","type":{"kind":"field"},"visibility":"private"},{"name":"y","type":{"kind":"field"},"visibility":"private"}],"return_type":{"abi_type":{"kind":"boolean"},"visibility":"private"}},"bytecode":"H4sIAAAAAAAA/7WUMQ7DIAxFQ0Krrr2JjSGYLVcpKrn/CaqqDQN12WK+hPBgmWd/wEyHbF1SS923uhOs3pfoChI+wKXMAXzIKyNj4PB0TFTYc0w5RUjoqeAeEu1wqK0F54RGkWvW44LPzExnlkbMEs4JNZmN8PxS42uHv82T8a3Jeyn2Ks+VLPcO558HmyLMCDOXAXXtpPt4R/Rt9T36ss6dS9HGPx/eG17nGegKBQAA"}; - -export async function is_equal(x: Field, y: Field, foreignCallHandler?: ForeignCallHandler): Promise { - const program = new Noir(is_equal_circuit); - const args: InputMap = { x, y }; - const { returnValue } = await program.execute(args, foreignCallHandler); - return returnValue as boolean; -} -``` - -Now the `is_equal()` function and relevant types are readily available for use in TypeScript. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/debugger.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/debugger.md deleted file mode 100644 index 9b7565ba9ff..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/debugger.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Debugger -description: Learn about the Noir Debugger, in its REPL or VS Code versions. -keywords: [Nargo, VSCode, Visual Studio Code, REPL, Debugger] -sidebar_position: 2 ---- - -# Noir Debugger - -There are currently two ways of debugging Noir programs: - -1. From VS Code, via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). -2. Via the REPL debugger, which ships with Nargo. - -In order to use either version of the debugger, you will need to install recent enough versions of Noir, [Nargo](../getting_started/installation/index.md) and vscode-noir: - -- Noir & Nargo ≥0.28.0 -- Noir's VS Code extension ≥0.0.11 - -:::info -At the moment, the debugger supports debugging binary projects, but not contracts. -::: - -We cover the VS Code Noir debugger more in depth in [its VS Code debugger how-to guide](../how_to/debugger/debugging_with_vs_code.md) and [the reference](../reference/debugger/debugger_vscode.md). - -The REPL debugger is discussed at length in [the REPL debugger how-to guide](../how_to/debugger/debugging_with_the_repl.md) and [the reference](../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/language_server.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/language_server.md deleted file mode 100644 index 81e0356ef8a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/language_server.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Language Server -description: Learn about the Noir Language Server, how to install the components, and configuration that may be required. -keywords: [Nargo, Language Server, LSP, VSCode, Visual Studio Code] -sidebar_position: 0 ---- - -This section helps you install and configure the Noir Language Server. - -The Language Server Protocol (LSP) has two components, the [Server](#language-server) and the [Client](#language-client). Below we describe each in the context of Noir. - -## Language Server - -The Server component is provided by the Nargo command line tool that you installed at the beginning of this guide. -As long as Nargo is installed and you've used it to run other commands in this guide, it should be good to go! - -If you'd like to verify that the `nargo lsp` command is available, you can run `nargo --help` and look for `lsp` in the list of commands. If you see it, you're using a version of Noir with LSP support. - -## Language Client - -The Client component is usually an editor plugin that launches the Server. It communicates LSP messages between the editor and the Server. For example, when you save a file, the Client will alert the Server, so it can try to compile the project and report any errors. - -Currently, Noir provides a Language Client for Visual Studio Code via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). - -> **Note:** Noir's Language Server Protocol support currently assumes users' VSCode workspace root to be the same as users' Noir project root (i.e. where Nargo.toml lies). -> -> If LSP features seem to be missing / malfunctioning, make sure you are opening your Noir project directly (instead of as a sub-folder) in your VSCode instance. - -When your language server is running correctly and the VSCode plugin is installed, you should see handy codelens buttons for compilation, measuring circuit size, execution, and tests: - -![Compile and Execute](@site/static/img/codelens_compile_execute.png) -![Run test](@site/static/img/codelens_run_test.png) - -You should also see your tests in the `testing` panel: - -![Testing panel](@site/static/img/codelens_testing_panel.png) - -### Configuration - -- **Noir: Enable LSP** - If checked, the extension will launch the Language Server via `nargo lsp` and communicate with it. -- **Noir: Nargo Flags** - Additional flags may be specified if you require them to be added when the extension calls `nargo lsp`. -- **Noir: Nargo Path** - An absolute path to a Nargo binary with the `lsp` command. This may be useful if Nargo is not within the `PATH` of your editor. -- **Noir > Trace: Server** - Setting this to `"messages"` or `"verbose"` will log LSP messages between the Client and Server. Useful for debugging. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/testing.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/testing.md deleted file mode 100644 index 866677da567..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tooling/testing.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Testing in Noir -description: Learn how to use Nargo to test your Noir program in a quick and easy way -keywords: [Nargo, testing, Noir, compile, test] -sidebar_position: 1 ---- - -You can test your Noir programs using Noir circuits. - -Nargo will automatically compile and run any functions which have the decorator `#[test]` on them if -you run `nargo test`. - -For example if you have a program like: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test] -fn test_add() { - assert(add(2,2) == 4); - assert(add(0,1) == 1); - assert(add(1,0) == 1); -} -``` - -Running `nargo test` will test that the `test_add` function can be executed while satisfying all -the constraints which allows you to test that add returns the expected values. Test functions can't -have any arguments currently. - -### Test fail - -You can write tests that are expected to fail by using the decorator `#[test(should_fail)]`. For example: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test(should_fail)] -fn test_add() { - assert(add(2,2) == 5); -} -``` - -You can be more specific and make it fail with a specific reason by using `should_fail_with = ""`: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "What is the airspeed velocity of an unladen swallow")] -fn test_bridgekeeper() { - main(32); -} -``` - -The string given to `should_fail_with` doesn't need to exactly match the failure reason, it just needs to be a substring of it: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "airspeed velocity")] -fn test_bridgekeeper() { - main(32); -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tutorials/noirjs_app.md deleted file mode 100644 index eac28168445..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/tutorials/noirjs_app.md +++ /dev/null @@ -1,362 +0,0 @@ ---- -title: Building a web app with NoirJS -description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. -keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] -sidebar_position: 0 -pagination_next: noir/concepts/data_types/index ---- - -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! - -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/installation/index.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! - -### Nargo - -Run: - -```bash -nargo new circuit -``` - -And... That's about it. Your program is ready to be compiled and run. - -To compile, let's `cd` into the `circuit` folder to enter our project, and call: - -```bash -nargo compile -``` - -This compiles our circuit into `json` format and add it to a new `target` folder. - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` - -::: - -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/hello_noir/index.md) guide. However, we want our app to run on the browser, so we need Vite. - -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. - -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". - -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: - -```bash -cd vite-project -``` - -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; - -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); -``` - -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: - -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev -``` - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... -``` - -::: - -#### Some cleanup - -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. - -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) - -## HTML - -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: - -```html - - - - - - -

Noir app

-
- - -
-
-

Logs

-

Proof

-
- - -``` - -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). - -Start by pasting in this boilerplate code: - -```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} - -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } -}); -``` - -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html -``` - -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - -::: - -## Some NoirJS - -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: - -```ts -import circuit from '../circuit/target/circuit.json'; -``` - -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: - -```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -import { Noir } from '@noir-lang/noir_js'; -``` - -And instantiate them inside our try-catch block: - -```ts -// try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); -// } -``` - -:::note - -For the remainder of the tutorial, everything will be happening inside the `try` block - -::: - -## Our app - -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: - -```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; -``` - -Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: - -```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); -const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); -``` - -You're probably eager to see stuff happening, so go and run your app now! - -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. - -![Getting Started 0](@site/static/img/noir_getting_started_1.png) - -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! - -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. - -## Verifying - -Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: - -```js -display('logs', 'Verifying proof... ⌛'); -const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); -``` - -You have successfully generated a client-side Noir web app! - -![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) - -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. - -## UltraHonk Backend - -Barretenberg has recently exposed a new UltraHonk backend. We can use UltraHonk in NoirJS after version 0.33.0. Everything will be the same as the tutorial above, except that the class we need to import will change: -```js -import { UltraHonkBackend, UltraHonkVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -``` -The backend will then be instantiated as such: -```js -const backend = new UltraHonkBackend(circuit); -``` -Then all the commands to prove and verify your circuit will be same. - -The only feature currently unsupported with UltraHonk are [recursive proofs](../explainers/explainer-recursion.md). \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/cspell.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/cspell.json deleted file mode 100644 index c60b0a597b1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/cspell.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "words": [ - "Cryptdoku" - ] -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-oracle.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-oracle.md deleted file mode 100644 index 821e1f95c04..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-oracle.md +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: Oracles -description: This guide provides an in-depth understanding of how Oracles work in Noir programming. Learn how to use outside calculations in your programs, constrain oracles, and understand their uses and limitations. -keywords: - - Noir Programming - - Oracles - - JSON-RPC - - Foreign Call Handlers - - Constrained Functions - - Blockchain Programming -sidebar_position: 1 ---- - -If you've seen "The Matrix" you may recall "The Oracle" as Gloria Foster smoking cigarettes and baking cookies. While she appears to "know things", she is actually providing a calculation of a pre-determined future. Noir Oracles are similar, in a way. They don't calculate the future (yet), but they allow you to use outside calculations in your programs. - -![matrix oracle prediction](@site/static/img/memes/matrix_oracle.jpeg) - -A Noir program is usually self-contained. You can pass certain inputs to it, and it will generate a deterministic output for those inputs. But what if you wanted to defer some calculation to an outside process or source? - -Oracles are functions that provide this feature. - -## Use cases - -An example usage for Oracles is proving something on-chain. For example, proving that the ETH-USDC quote was below a certain target at a certain block time. Or even making more complex proofs like proving the ownership of an NFT as an anonymous login method. - -Another interesting use case is to defer expensive calculations to be made outside of the Noir program, and then constraining the result; similar to the use of [unconstrained functions](../noir/concepts//unconstrained.md). - -In short, anything that can be constrained in a Noir program but needs to be fetched from an external source is a great candidate to be used in oracles. - -## Constraining oracles - -Just like in The Matrix, Oracles are powerful. But with great power, comes great responsibility. Just because you're using them in a Noir program doesn't mean they're true. Noir has no superpowers. If you want to prove that Portugal won the Euro Cup 2016, you're still relying on potentially untrusted information. - -To give a concrete example, Alice wants to login to the [NounsDAO](https://nouns.wtf/) forum with her username "noir_nouner" by proving she owns a noun without revealing her ethereum address. Her Noir program could have an oracle call like this: - -```rust -#[oracle(getNoun)] -unconstrained fn get_noun(address: Field) -> Field -``` - -This oracle could naively resolve with the number of Nouns she possesses. However, it is useless as a trusted source, as the oracle could resolve to anything Alice wants. In order to make this oracle call actually useful, Alice would need to constrain the response from the oracle, by proving her address and the noun count belongs to the state tree of the contract. - -In short, **Oracles don't prove anything. Your Noir program does.** - -:::danger - -If you don't constrain the return of your oracle, you could be clearly opening an attack vector on your Noir program. Make double-triple sure that the return of an oracle call is constrained! - -::: - -## How to use Oracles - -On CLI, Nargo resolves oracles by making JSON RPC calls, which means it would require an RPC node to be running. - -In JavaScript, NoirJS accepts and resolves arbitrary call handlers (that is, not limited to JSON) as long as they match the expected types the developer defines. Refer to [Foreign Call Handler](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) to learn more about NoirJS's call handling. - -If you want to build using oracles, follow through to the [oracle guide](../how_to/how-to-oracles.md) for a simple example on how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-recursion.md deleted file mode 100644 index df8529ef4e0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-recursion.md +++ /dev/null @@ -1,176 +0,0 @@ ---- -title: Recursive proofs -description: Explore the concept of recursive proofs in Zero-Knowledge programming. Understand how recursion works in Noir, a language for writing smart contracts on the EVM blockchain. Learn through practical examples like Alice and Bob's guessing game, Charlie's recursive merkle tree, and Daniel's reusable components. Discover how to use recursive proofs to optimize computational resources and improve efficiency. - -keywords: - [ - "Recursive Proofs", - "Zero-Knowledge Programming", - "Noir", - "EVM Blockchain", - "Smart Contracts", - "Recursion in Noir", - "Alice and Bob Guessing Game", - "Recursive Merkle Tree", - "Reusable Components", - "Optimizing Computational Resources", - "Improving Efficiency", - "Verification Key", - "Aggregation", - "Recursive zkSNARK schemes", - "PLONK", - "Proving and Verification Keys" - ] -sidebar_position: 1 -pagination_next: how_to/how-to-recursion ---- - -In programming, we tend to think of recursion as something calling itself. A classic example would be the calculation of the factorial of a number: - -```js -function factorial(n) { - if (n === 0 || n === 1) { - return 1; - } else { - return n * factorial(n - 1); - } -} -``` - -In this case, while `n` is not `1`, this function will keep calling itself until it hits the base case, bubbling up the result on the call stack: - -```md - Is `n` 1? <--------- - /\ / - / \ n = n -1 - / \ / - Yes No -------- -``` - -In Zero-Knowledge, recursion has some similarities. - -It is not a Noir function calling itself, but a proof being used as an input to another circuit. In short, you verify one proof *inside* another proof, returning the proof that both proofs are valid. - -This means that, given enough computational resources, you can prove the correctness of any arbitrary number of proofs in a single proof. This could be useful to design state channels (for which a common example would be [Bitcoin's Lightning Network](https://en.wikipedia.org/wiki/Lightning_Network)), to save on gas costs by settling one proof on-chain, or simply to make business logic less dependent on a consensus mechanism. - -## Examples - -Let us look at some of these examples - -### Alice and Bob - Guessing game - -Alice and Bob are friends, and they like guessing games. They want to play a guessing game online, but for that, they need a trusted third-party that knows both of their secrets and finishes the game once someone wins. - -So, they use zero-knowledge proofs. Alice tries to guess Bob's number, and Bob will generate a ZK proof stating whether she succeeded or failed. - -This ZK proof can go on a smart contract, revealing the winner and even giving prizes. However, this means every turn needs to be verified on-chain. This incurs some cost and waiting time that may simply make the game too expensive or time-consuming to be worth it. - -As a solution, Alice proposes the following: "what if Bob generates his proof, and instead of sending it on-chain, I verify it *within* my own proof before playing my own turn?". - -She can then generate a proof that she verified his proof, and so on. - -```md - Did you fail? <-------------------------- - / \ / - / \ n = n -1 - / \ / - Yes No / - | | / - | | / - | You win / - | / - | / -Generate proof of that / - + / - my own guess ---------------- -``` - -### Charlie - Recursive merkle tree - -Charlie is a concerned citizen, and wants to be sure his vote in an election is accounted for. He votes with a ZK proof, but he has no way of knowing that his ZK proof was included in the total vote count! - -If the vote collector puts all of the votes into a [Merkle tree](https://en.wikipedia.org/wiki/Merkle_tree), everyone can prove the verification of two proofs within one proof, as such: - -```md - abcd - __________|______________ - | | - ab cd - _____|_____ ______|______ - | | | | - alice bob charlie daniel -``` - -Doing this recursively allows us to arrive on a final proof `abcd` which if true, verifies the correctness of all the votes. - -### Daniel - Reusable components - -Daniel has a big circuit and a big headache. A part of his circuit is a setup phase that finishes with some assertions that need to be made. But that section alone takes most of the proving time, and is largely independent of the rest of the circuit. - -He might find it more efficient to generate a proof for that setup phase separately, and verify that proof recursively in the actual business logic section of his circuit. This will allow for parallelization of both proofs, which results in a considerable speedup. - -## What params do I need - -As you can see in the [recursion reference](noir/standard_library/recursion.mdx), a simple recursive proof requires: - -- The proof to verify -- The Verification Key of the circuit that generated the proof -- A hash of this verification key, as it's needed for some backends -- The public inputs for the proof - -:::info - -Recursive zkSNARK schemes do not necessarily "verify a proof" in the sense that you expect a true or false to be spit out by the verifier. Rather an aggregation object is built over the public inputs. - -So, taking the example of Alice and Bob and their guessing game: - -- Alice makes her guess. Her proof is *not* recursive: it doesn't verify any proof within it! It's just a standard `assert(x != y)` circuit -- Bob verifies Alice's proof and makes his own guess. In this circuit, he doesn't exactly *prove* the verification of Alice's proof. Instead, he *aggregates* his proof to Alice's proof. The actual verification is done when the full proof is verified, for example when using `nargo verify` or through the verifier smart contract. - -We can imagine recursive proofs a [relay race](https://en.wikipedia.org/wiki/Relay_race). The first runner doesn't have to receive the baton from anyone else, as he/she already starts with it. But when his/her turn is over, the next runner needs to receive it, run a bit more, and pass it along. Even though every runner could theoretically verify the baton mid-run (why not? 🏃🔍), only at the end of the race does the referee verify that the whole race is valid. - -::: - -## Some architecture - -As with everything in computer science, there's no one-size-fits all. But there are some patterns that could help understanding and implementing them. To give three examples: - -### Adding some logic to a proof verification - -This would be an approach for something like our guessing game, where proofs are sent back and forth and are verified by each opponent. This circuit would be divided in two sections: - -- A `recursive verification` section, which would be just the call to `std::verify_proof`, and that would be skipped on the first move (since there's no proof to verify) -- A `guessing` section, which is basically the logic part where the actual guessing happens - -In such a situation, and assuming Alice is first, she would skip the first part and try to guess Bob's number. Bob would then verify her proof on the first section of his run, and try to guess Alice's number on the second part, and so on. - -### Aggregating proofs - -In some one-way interaction situations, recursion would allow for aggregation of simple proofs that don't need to be immediately verified on-chain or elsewhere. - -To give a practical example, a barman wouldn't need to verify a "proof-of-age" on-chain every time he serves alcohol to a customer. Instead, the architecture would comprise two circuits: - -- A `main`, non-recursive circuit with some logic -- A `recursive` circuit meant to verify two proofs in one proof - -The customer's proofs would be intermediate, and made on their phones, and the barman could just verify them locally. He would then aggregate them into a final proof sent on-chain (or elsewhere) at the end of the day. - -### Recursively verifying different circuits - -Nothing prevents you from verifying different circuits in a recursive proof, for example: - -- A `circuit1` circuit -- A `circuit2` circuit -- A `recursive` circuit - -In this example, a regulator could verify that taxes were paid for a specific purchase by aggregating both a `payer` circuit (proving that a purchase was made and taxes were paid), and a `receipt` circuit (proving that the payment was received) - -## How fast is it - -At the time of writing, verifying recursive proofs is surprisingly fast. This is because most of the time is spent on generating the verification key that will be used to generate the next proof. So you are able to cache the verification key and reuse it later. - -Currently, Noir JS packages don't expose the functionality of loading proving and verification keys, but that feature exists in the underlying `bb.js` package. - -## How can I try it - -Learn more about using recursion in Nargo and NoirJS in the [how-to guide](../how_to/how-to-recursion.md) and see a full example in [noir-examples](https://github.com/noir-lang/noir-examples). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-writing-noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-writing-noir.md deleted file mode 100644 index 3ef6e014a2f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/explainers/explainer-writing-noir.md +++ /dev/null @@ -1,177 +0,0 @@ ---- -title: Writing Performant Noir -description: Understand new considerations when writing Noir -keywords: [Noir, programming, rust] -tags: [Optimization] -sidebar_position: 0 ---- - - -This article intends to set you up with key concepts essential for writing more viable applications that use zero knowledge proofs, namely around efficient circuits. - -## Context - 'Efficient' is subjective - -When writing a web application for a performant computer with high-speed internet connection, writing efficient code sometimes is seen as an afterthought only if needed. Large multiplications running at the innermost of nested loops may not even be on a dev's radar. -When writing firmware for a battery-powered microcontroller, you think of cpu cycles as rations to keep within a product's power budget. - -> Code is written to create applications that perform specific tasks within specific constraints - -And these constraints differ depending on where the compiled code is execute. - -### The Ethereum Virtual Machine (EVM) - -In scenarios where extremely low gas costs are required for an Ethereum application to be viable/competitive, Ethereum smart contract developers get into what is colloquially known as: "*gas golfing*". Finding the lowest execution cost of their compiled code (EVM bytecode) to achieve a specific task. - -The equivalent optimization task when writing zk circuits is affectionately referred to as "*gate golfing*", finding the lowest gate representation of the compiled Noir code. - -### Coding for circuits - a paradigm shift - -In zero knowledge cryptography, code is compiled to "circuits" consisting of arithmetic gates, and gate count is the significant cost. Depending on the proving system this is linearly proportionate to proving time, and so from a product point this should be kept as low as possible. - -Whilst writing efficient code for web apps and Solidity has a few key differences, writing efficient circuits have a different set of considerations. It is a bit of a paradigm shift, like writing code for GPUs for the first time... - -For example, drawing a circle at (0, 0) of radius `r`: -- For a single CPU thread, -``` -for theta in 0..2*pi { - let x = r * cos(theta); - let y = r * sin(theta); - draw(x, y); -} // note: would do 0 - pi/2 and draw +ve/-ve x and y. -``` - -- For GPUs (simultaneous parallel calls with x, y across image), -``` -if (x^2 + y^2 = r^2) { - draw(x, y); -} -``` - -([Related](https://www.youtube.com/watch?v=-P28LKWTzrI)) - -Whilst this CPU -> GPU does not translate to circuits exactly, it is intended to exemplify the difference in intuition when coding for different machine capabilities/constraints. - -### Context Takeaway - -For those coming from a primarily web app background, this article will explain what you need to consider when writing circuits. Furthermore, for those experienced writing efficient machine code, prepare to shift what you think is efficient 😬 - -## Translating from Rust - -For some applications using Noir, existing code might be a convenient starting point to then proceed to optimize the gate count of. - -:::note -Many valuable functions and algorithms have been written in more established languages (C/C++), and converted to modern ones (like Rust). -::: - -Fortunately for Noir developers, when needing a particular function a Rust implementation can be readily compiled into Noir with some key changes. While the compiler does a decent amount of optimizations, it won't be able to change code that has been optimized for clock-cycles into code optimized for arithmetic gates. - -A few things to do when converting Rust code to Noir: -- `println!` is not a macro, use `println` function (same for `assert_eq`) -- No early `return` in function. Use constrain via assertion instead -- No passing by reference. Remove `&` operator to pass by value (copy) -- No boolean operators (`&&`, `||`). Use bitwise operators (`&`, `|`) with boolean values -- No type `usize`. Use types `u8`, `u32`, `u64`, ... -- `main` return must be public, `pub` -- No `const`, use `global` -- Noir's LSP is your friend, so error message should be informative enough to resolve syntax issues. - -## Writing efficient Noir for performant products - -The following points help refine our understanding over time. - -:::note -A Noir program makes a statement that can be verified. -::: - -It compiles to a structure that represents the calculation, and can assert results within the calculation at any stage (via the `constrain` keyword). - -A Noir program compiles to an Abstract Circuit Intermediate Representation which is: - - Conceptually a tree structure - - Leaves (inputs) are the `Field` type - - Nodes contain arithmetic operations to combine them (gates) - - The root is the final result (return value) - -:::tip -The command `nargo info` shows the programs circuit size, and is useful to compare the value of changes made. -You can dig deeper and use the `--print-acir` param to take a closer look at individual ACIR opcodes, and the proving backend to see its gate count (eg for barretenberg, `bb gates -b ./target/program.json`). -::: - -### Use the `Field` type - -Since the native type of values in circuits are `Field`s, using them for variables in Noir means less gates converting them under the hood. -Some things to be mindful of when using a Field type for a regular integer value: -- A variable of type `Field` can be cast `as` an integer type (eg `u8`, `u64`) - - Note: this retains only the bits of the integer type. Eg a Field value of 260 as a `u8` becomes 4 -- For Field types arithmetic operations meaningfully overflow/underflow, yet for integer types they are checked according to their size -- Comparisons and bitwise operations do not exist for `Field`s, cast to an appropriately sized integer type when you need to - -:::tip -Where possible, use `Field` type for values. Using smaller value types, and bit-packing strategies, will result in MORE gates -::: - - -### Use Arithmetic over non-arithmetic operations - -Since circuits are made of arithmetic gates, the cost of arithmetic operations tends to be one gate. Whereas for procedural code, they represent several clock cycles. - -Inversely, non-arithmetic operators are achieved with multiple gates, vs 1 clock cycle for procedural code. - -| (cost\op) | arithmetic
(`*`, `+`) | bit-wise ops
(eg `<`, `\|`, `>>`) | -| - | - | - | -| **cycles** | 10+ | 1 | -| **gates** | 1 | 10+ | - -Bit-wise operations (e.g. bit shifts `<<` and `>>`), albeit commonly used in general programming and especially for clock cycle optimizations, are on the contrary expensive in gates when performed within circuits. - -Translate away from bit shifts when writing constrained functions for the best performance. - -On the flip side, feel free to use bit shifts in unconstrained functions and tests if necessary, as they are executed outside of circuits and does not induce performance hits. - -### Use static over dynamic values - -Another general theme that manifests in different ways is that static reads are represented with less gates than dynamic ones. - -Reading from read-only memory (ROM) adds less gates than random-access memory (RAM), 2 vs ~3.25 due to the additional bounds checks. Arrays of fixed length (albeit used at a lower capacity), will generate less gates than dynamic storage. - -Related to this, if an index used to access an array is not known at compile time (ie unknown until run time), then ROM will be converted to RAM, expanding the gate count. - -:::tip -Use arrays and indices that are known at compile time where possible. -Using `assert_constant(i);` before an index, `i`, is used in an array will give a compile error if `i` is NOT known at compile time. -::: - -### Leverage unconstrained execution - -Constrained verification can leverage unconstrained execution, this is especially useful for operations that are represented by many gates. -Use an [unconstrained function](../noir/concepts/unconstrained.md) to perform gate-heavy calculations, then verify and constrain the result. - -Eg division generates more gates than multiplication, so calculating the quotient in an unconstrained function then constraining the product for the quotient and divisor (+ any remainder) equals the dividend will be more efficient. - -Use ` if is_unconstrained() { /`, to conditionally execute code if being called in an unconstrained vs constrained way. - -## Advanced - -Unless you're well into the depth of gate optimization, this advanced section can be ignored. - -### Combine arithmetic operations - -A Noir program can be honed further by combining arithmetic operators in a way that makes the most of each constraint of the backend proving system. This is in scenarios where the backend might not be doing this perfectly. - -Eg Barretenberg backend (current default for Noir) is a width-4 PLONKish constraint system -$ w_1*w_2*q_m + w_1*q_1 + w_2*q_2 + w_3*q_3 + w_4*q_4 + q_c = 0 $ - -Here we see there is one occurrence of witness 1 and 2 ($w_1$, $w_2$) being multiplied together, with addition to witnesses 1-4 ($w_1$ .. $w_4$) multiplied by 4 corresponding circuit constants ($q_1$ .. $q_4$) (plus a final circuit constant, $q_c$). - -Use `nargo info --print-acir`, to inspect the ACIR opcodes (and the proving system for gates), and it may present opportunities to amend the order of operations and reduce the number of constraints. - -#### Variable as witness vs expression - -If you've come this far and really know what you're doing at the equation level, a temporary lever (that will become unnecessary/useless over time) is: `std::as_witness`. This informs the compiler to save a variable as a witness not an expression. - -The compiler will mostly be correct and optimal, but this may help some near term edge cases that are yet to optimize. -Note: When used incorrectly it will create **less** efficient circuits (higher gate count). - -## References -- Guillaume's ["`Cryptdoku`" talk](https://www.youtube.com/watch?v=MrQyzuogxgg) (Jun'23) -- Tips from Tom, Jake and Zac. -- [Idiomatic Noir](https://www.vlayer.xyz/blog/idiomatic-noir-part-1-collections) blog post diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/backend/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/backend/_category_.json deleted file mode 100644 index b82e92beb0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/backend/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "position": 1, - "label": "Install Proving Backend", - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/backend/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/backend/index.md deleted file mode 100644 index 7192d954877..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/backend/index.md +++ /dev/null @@ -1,31 +0,0 @@ ---- -title: Proving Backend Installation -description: Proving backends offer command line tools for proving and verifying Noir programs. This page describes how to install `bb` as an example. -keywords: [ - Proving - Backend - Barretenberg - bb - bbup - Installation - Terminal - Command - CLI - Version -] -pagination_next: getting_started/hello_noir/index ---- - -Proving backends each provide their own tools for working with Noir programs, providing functionality like proof generation, proof verification, and verifier smart contract generation. - -For the latest information on tooling provided by each proving backend, installation instructions, Noir version compatibility... you may refer to the proving backends' own documentation. - -You can find the full list of proving backends compatible with Noir in [Awesome Noir](https://github.com/noir-lang/awesome-noir/?tab=readme-ov-file#proving-backends). - -## Example: Installing `bb` - -`bb` is the CLI tool provided by the [Barretenberg proving backend](https://github.com/AztecProtocol/barretenberg) developed by Aztec Labs. - -You can find the instructions for installation in [`bb`'s documentation](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/cpp/src/barretenberg/bb/readme.md#installation). - -Once installed, we are ready to start working on [our first Noir program](../hello_noir/index.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/_category_.json deleted file mode 100644 index 976a2325de0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/index.md deleted file mode 100644 index 6760e54aad1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/index.md +++ /dev/null @@ -1,157 +0,0 @@ ---- -title: Creating a Project -description: - Learn how to create and verify your first Noir program using Nargo, a programming language for - zero-knowledge proofs. -keywords: - [ - Nargo, - Noir, - zero-knowledge proofs, - programming language, - create Noir program, - verify Noir program, - step-by-step guide, - ] -sidebar_position: 1 - ---- - -Now that we have installed Nargo and a proving backend, it is time to make our first hello world program! - -### 1. Create a new project directory - -Noir code can live anywhere on your computer. Let us create a _projects_ folder in the home -directory to house our first Noir program. - -Create the directory and change directory into it by running: - -```sh -mkdir ~/projects -cd ~/projects -``` - -## Nargo - -Nargo provides the ability to initiate and execute Noir projects. Read the [Nargo installation](../installation/index.md) section to learn more about Nargo and how to install it. - -### 2. Create a new Noir project - -Now that we are in the projects directory, create a new Nargo project by running: - -```sh -nargo new hello_world -``` - -`hello_world` can be any arbitrary project name, we are simply using `hello_world` for demonstration. - -In production, it is common practice to name the project folder, `circuits`, for clarity amongst other folders in the codebase (like: `contracts`, `scripts`, and `test`). - -A `hello_world` folder would be created. Similar to Rust, the folder houses _src/main.nr_ and -_Nargo.toml_ which contain the source code and environmental options of your Noir program -respectively. - -#### Intro to Noir Syntax - -Let us take a closer look at _main.nr_. The default _main.nr_ generated should look like this: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -The first line of the program specifies the program's inputs: - -```rust -x : Field, y : pub Field -``` - -Program inputs in Noir are private by default (e.g. `x`), but can be labeled public using the -keyword `pub` (e.g. `y`). To learn more about private and public values, check the -[Data Types](../../noir/concepts/data_types/index.md) section. - -The next line of the program specifies its body: - -```rust -assert(x != y); -``` - -The Noir syntax `assert` can be interpreted as something similar to constraints in other zk-contract languages. - -For more Noir syntax, check the [Language Concepts](../../noir/concepts/comments.md) chapter. - -### 3. Build in/output files - -Change directory into _hello_world_ and build in/output files for your Noir program by running: - -```sh -cd hello_world -nargo check -``` - -A _Prover.toml_ file will be generated in your project directory, to allow specifying input values to the program. - -### 4. Execute the Noir program - -Now that the project is set up, we can execute our Noir program. - -Fill in input values for execution in the _Prover.toml_ file. For example: - -```toml -x = "1" -y = "2" -``` - -Execute your Noir program: - -```sh -nargo execute witness-name -``` - -The witness corresponding to this execution will then be written to the file `./target/witness-name.gz`. - -The command also automatically compiles your Noir program if it was not already / was edited, which you may notice the compiled artifacts being written to the file `./target/hello_world.json`. - -## Proving Backend - -Proving backends provide the ability to generate and verify proofs of executing Noir programs, following Noir's tooling that compiles and executes the programs. Read the [proving backend installation](../backend/index.md) section to learn more about proving backends and how to install them. - -Barretenberg is used as an example here to demonstrate how proving and verifying could be implemented and used. Read the [`bb` installation](../backend/index.md#example-installing-bb) section for how to install Barretenberg's CLI tool; refer to [`bb`'s documentation](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/cpp/src/barretenberg/bb/readme.md) for full details about the tool. - -### 5. Prove an execution of the Noir program - -Using Barretenberg as an example, prove the valid execution of your Noir program running: - -```sh -bb prove -b ./target/hello_world.json -w ./target/witness-name.gz -o ./target/proof -``` - -The proof generated will then be written to the file `./target/proof`. - -:::tip -Since the params for `nargo` and `bb` often specify multiple filenames to read from or write to, remember to check each command is referring to the desired filenames. -Or for greater certainty, delete the target folder and go through each step again (compile, witness, prove, ...) to ensure files generated in past commands are being referenced in future ones. -::: - -### 6. Verify the execution proof - -Once a proof is generated, we can verify correct execution of our Noir program by verifying the proof file. - -Using Barretenberg as an example, compute the verification key for the Noir program by running: - -```sh -bb write_vk -b ./target/hello_world.json -o ./target/vk -``` - -And verify your proof by running: - -```sh -bb verify -k ./target/vk -p ./target/proof -``` - -If successful, the verification will complete in silence; if unsuccessful, the command will trigger logging of the corresponding error. - -Congratulations, you have now created and verified a proof for your very first Noir program! - -In the [next section](./project_breakdown.md), we will go into more detail on each step performed. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/project_breakdown.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/project_breakdown.md deleted file mode 100644 index 96e653f6c08..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/hello_noir/project_breakdown.md +++ /dev/null @@ -1,159 +0,0 @@ ---- -title: Project Breakdown -description: - Learn about the anatomy of a Nargo project, including the purpose of the Prover TOML - file, and how to prove and verify your program. -keywords: - [Nargo, Nargo project, Prover.toml, proof verification, private asset transfer] -sidebar_position: 2 ---- - -This section breaks down our hello world program from the previous section. - -## Anatomy of a Nargo Project - -Upon creating a new project with `nargo new` and building the in/output files with `nargo check` -commands, you would get a minimal Nargo project of the following structure: - - - src - - Prover.toml - - Nargo.toml - -The source directory _src_ holds the source code for your Noir program. By default only a _main.nr_ -file will be generated within it. - -### Prover.toml - -_Prover.toml_ is used for specifying the input values for executing and proving the program. You can specify `toml` files with different names by using the `--prover-name` or `-p` flags, see the [Prover](#provertoml) section below. Optionally you may specify expected output values for prove-time checking as well. - -### Nargo.toml - -_Nargo.toml_ contains the environmental options of your project. It contains a "package" section and a "dependencies" section. - -Example Nargo.toml: - -```toml -[package] -name = "noir_starter" -type = "bin" -authors = ["Alice"] -compiler_version = "0.9.0" -description = "Getting started with Noir" -entry = "circuit/main.nr" -license = "MIT" - -[dependencies] -ecrecover = {tag = "v0.9.0", git = "https://github.com/colinnielsen/ecrecover-noir.git"} -``` - -Nargo.toml for a [workspace](../../noir/modules_packages_crates/workspaces.md) will look a bit different. For example: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -#### Package section - -The package section defines a number of fields including: - -- `name` (**required**) - the name of the package -- `type` (**required**) - can be "bin", "lib", or "contract" to specify whether its a binary, library or Aztec contract -- `authors` (optional) - authors of the project -- `compiler_version` - specifies the version of the compiler to use. This is enforced by the compiler and follow's [Rust's versioning](https://doc.rust-lang.org/cargo/reference/manifest.html#the-version-field), so a `compiler_version = 0.18.0` will enforce Nargo version 0.18.0, `compiler_version = ^0.18.0` will enforce anything above 0.18.0 but below 0.19.0, etc. For more information, see how [Rust handles these operators](https://docs.rs/semver/latest/semver/enum.Op.html) -- `description` (optional) -- `entry` (optional) - a relative filepath to use as the entry point into your package (overrides the default of `src/lib.nr` or `src/main.nr`) -- `backend` (optional) -- `license` (optional) -- `expression_width` (optional) - Sets the default backend expression width. This field will override the default backend expression width specified by the Noir compiler (currently set to width 4). - -#### Dependencies section - -This is where you will specify any dependencies for your project. See the [Dependencies page](../../noir/modules_packages_crates/dependencies.md) for more info. - -`./proofs/` and `./contract/` directories will not be immediately visible until you create a proof or -verifier contract respectively. - -### main.nr - -The _main.nr_ file contains a `main` method, this method is the entry point into your Noir program. - -In our sample program, _main.nr_ looks like this: - -```rust -fn main(x : Field, y : Field) { - assert(x != y); -} -``` - -The parameters `x` and `y` can be seen as the API for the program and must be supplied by the prover. Since neither `x` nor `y` is marked as public, the verifier does not supply any inputs, when verifying the proof. - -The prover supplies the values for `x` and `y` in the _Prover.toml_ file. - -As for the program body, `assert` ensures that the condition to be satisfied (e.g. `x != y`) is constrained by the proof of the execution of said program (i.e. if the condition was not met, the verifier would reject the proof as an invalid proof). - -### Prover.toml - -The _Prover.toml_ file is a file which the prover uses to supply the inputs to the Noir program (both private and public). - -In our hello world program the _Prover.toml_ file looks like this: - -```toml -x = "1" -y = "2" -``` - -When the command `nargo execute` is executed, nargo will execute the Noir program using the inputs specified in `Prover.toml`, aborting if it finds that these do not satisfy the constraints defined by `main`. In this example, `x` and `y` must satisfy the inequality constraint `assert(x != y)`. - -If an output name is specified such as `nargo execute foo`, the witness generated by this execution will be written to `./target/foo.gz`. This can then be used to generate a proof of the execution. - -#### Arrays of Structs - -The following code shows how to pass an array of structs to a Noir program to generate a proof. - -```rust -// main.nr -struct Foo { - bar: Field, - baz: Field, -} - -fn main(foos: [Foo; 3]) -> pub Field { - foos[2].bar + foos[2].baz -} -``` - -Prover.toml: - -```toml -[[foos]] # foos[0] -bar = 0 -baz = 0 - -[[foos]] # foos[1] -bar = 0 -baz = 0 - -[[foos]] # foos[2] -bar = 1 -baz = 2 -``` - -#### Custom toml files - -You can specify a `toml` file with a different name to use for execution by using the `--prover-name` or `-p` flags. - -This command looks for proof inputs in the default **Prover.toml** and generates the witness and saves it at `./target/foo.gz`: - -```bash -nargo execute foo -``` - -This command looks for proof inputs in the custom **OtherProver.toml** and generates the witness and saves it at `./target/bar.gz`: - -```bash -nargo execute -p OtherProver bar -``` - -Now that you understand the concepts, you'll probably want some editor feedback while you are writing more complex code. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/_category_.json deleted file mode 100644 index 0c02fb5d4d7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "position": 0, - "label": "Install Nargo", - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/index.md deleted file mode 100644 index 53ea9c7891c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/index.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Nargo Installation -description: - nargo is a command line tool for interacting with Noir programs. This page is a quick guide on how to install Nargo through the most common and easy method, noirup -keywords: [ - Nargo - Noir - Rust - Cargo - Noirup - Installation - Terminal Commands - Version Check - Nightlies - Specific Versions - Branches - Noirup Repository -] -pagination_next: getting_started/hello_noir/index ---- - -`nargo` is a tool for working with Noir programs on the CLI, providing you with the ability to start new projects, compile, execute and test Noir programs from the terminal. - -The name is inspired by Rust's package manager `cargo`; and similar to Rust's `rustup`, Noir also has an easy installation script `noirup`. - -## Installing Noirup - -Open a terminal on your machine, and write: - -```bash -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Close the terminal, open another one, and run - -```bash -noirup -``` - -Done. That's it. You should have the latest version working. You can check with `nargo --version`. - -You can also install nightlies, specific versions -or branches. Check out the [noirup repository](https://github.com/noir-lang/noirup) for more -information. - -Now we're ready to start working on [our first Noir program!](../hello_noir/index.md) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/other_install_methods.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/other_install_methods.md deleted file mode 100644 index 3634723562b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/getting_started/installation/other_install_methods.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Alternative Installations -description: There are different ways to install Nargo, the one-stop shop and command-line tool for developing Noir programs. This guide explains how to specify which version to install when using noirup, and using WSL for windows. -keywords: [ - Installation - Nargo - Noirup - Binaries - Compiling from Source - WSL for Windows - macOS - Linux - Nix - Direnv - Uninstalling Nargo - ] -sidebar_position: 1 ---- - -## Encouraged Installation Method: Noirup - -Noirup is the endorsed method for installing Nargo, streamlining the process of fetching binaries or compiling from source. It supports a range of options to cater to your specific needs, from nightly builds and specific versions to compiling from various sources. - -### Installing Noirup - -First, ensure you have `noirup` installed: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -### Fetching Binaries - -With `noirup`, you can easily switch between different Nargo versions, including nightly builds: - -- **Nightly Version**: Install the latest nightly build. - - ```sh - noirup --version nightly - ``` - -- **Specific Version**: Install a specific version of Nargo. - ```sh - noirup --version - ``` - -### Compiling from Source - -`noirup` also enables compiling Nargo from various sources: - -- **From a Specific Branch**: Install from the latest commit on a branch. - - ```sh - noirup --branch - ``` - -- **From a Fork**: Install from the main branch of a fork. - - ```sh - noirup --repo - ``` - -- **From a Specific Branch in a Fork**: Install from a specific branch in a fork. - - ```sh - noirup --repo --branch - ``` - -- **From a Specific Pull Request**: Install from a specific PR. - - ```sh - noirup --pr - ``` - -- **From a Specific Commit**: Install from a specific commit. - - ```sh - noirup -C - ``` - -- **From Local Source**: Compile and install from a local directory. - ```sh - noirup --path ./path/to/local/source - ``` - -## Installation on Windows - -The default backend for Noir (Barretenberg) doesn't provide Windows binaries at this time. For that reason, Noir cannot be installed natively. However, it is available by using Windows Subsystem for Linux (WSL). - -Step 1: Follow the instructions [here](https://learn.microsoft.com/en-us/windows/wsl/install) to install and run WSL. - -step 2: Follow the [Noirup instructions](#encouraged-installation-method-noirup). - -## Uninstalling Nargo - -If you installed Nargo with `noirup`, you can uninstall Nargo by removing the files in `~/.nargo`, `~/nargo`, and `~/noir_cache`. This ensures that all installed binaries, configurations, and cache related to Nargo are fully removed from your system. - -```bash -rm -r ~/.nargo -rm -r ~/nargo -rm -r ~/noir_cache -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/_category_.json deleted file mode 100644 index 23b560f610b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/_category_.json deleted file mode 100644 index cc2cbb1c253..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugging", - "position": 5, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/debugging_with_the_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/debugging_with_the_repl.md deleted file mode 100644 index 1d64dae3f37..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/debugging_with_the_repl.md +++ /dev/null @@ -1,164 +0,0 @@ ---- -title: Using the REPL Debugger -description: - Step-by-step guide on how to debug your Noir circuits with the REPL Debugger. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -#### Pre-requisites - -In order to use the REPL debugger, first you need to install recent enough versions of Nargo and vscode-noir. - -## Debugging a simple circuit - -Let's debug a simple circuit: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -To start the REPL debugger, using a terminal, go to a Noir circuit's home directory. Then: - -`$ nargo debug` - -You should be seeing this in your terminal: - -``` -[main] Starting debugger -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:9 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> -``` - -The debugger displays the current Noir code location, and it is now waiting for us to drive it. - -Let's first take a look at the available commands. For that we'll use the `help` command. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -Some commands operate only for unconstrained functions, such as `memory` and `memset`. If you try to use them while execution is paused at an ACIR opcode, the debugger will simply inform you that you are not executing unconstrained code: - -``` -> memory -Unconstrained VM memory not available -> -``` - -Before continuing, we can take a look at the initial witness map: - -``` -> witness -_0 = 1 -_1 = 2 -> -``` - -Cool, since `x==1`, `y==2`, and we want to check that `x != y`, our circuit should succeed. At this point we could intervene and use the witness setter command to change one of the witnesses. Let's set `y=3`, then back to 2, so we don't affect the expected result: - -``` -> witness -_0 = 1 -_1 = 2 -> witness 1 3 -_1 = 3 -> witness -_0 = 1 -_1 = 3 -> witness 1 2 -_1 = 2 -> witness -_0 = 1 -_1 = 2 -> -``` - -Now we can inspect the current state of local variables. For that we use the `vars` command. - -``` -> vars -> -``` - -We currently have no vars in context, since we are at the entry point of the program. Let's use `next` to execute until the next point in the program. - -``` -> vars -> next -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:20 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> vars -x:Field = 0x01 -``` - -As a result of stepping, the variable `x`, whose initial value comes from the witness map, is now in context and returned by `vars`. - -``` -> next - 1 fn main(x : Field, y : pub Field) { - 2 -> assert(x != y); - 3 } -> vars -y:Field = 0x02 -x:Field = 0x01 -``` - -Stepping again we can finally see both variables and their values. And now we can see that the next assertion should succeed. - -Let's continue to the end: - -``` -> continue -(Continuing execution...) -Finished execution -> q -[main] Circuit witness successfully solved -``` - -Upon quitting the debugger after a solved circuit, the resulting circuit witness gets saved, equivalent to what would happen if we had run the same circuit with `nargo execute`. - -We just went through the basics of debugging using Noir REPL debugger. For a comprehensive reference, check out [the reference page](../../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/debugging_with_vs_code.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/debugging_with_vs_code.md deleted file mode 100644 index a5858c1a5eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/debugger/debugging_with_vs_code.md +++ /dev/null @@ -1,68 +0,0 @@ ---- -title: Using the VS Code Debugger -description: - Step by step guide on how to debug your Noir circuits with the VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -This guide will show you how to use VS Code with the vscode-noir extension to debug a Noir project. - -#### Pre-requisites - -- Nargo -- vscode-noir -- A Noir project with a `Nargo.toml`, `Prover.toml` and at least one Noir (`.nr`) containing an entry point function (typically `main`). - -## Running the debugger - -The easiest way to start debugging is to open the file you want to debug, and press `F5`. This will cause the debugger to launch, using your `Prover.toml` file as input. - -You should see something like this: - -![Debugger launched](@site/static/img/debugger/1-started.png) - -Let's inspect the state of the program. For that, we open VS Code's _Debug pane_. Look for this icon: - -![Debug pane icon](@site/static/img/debugger/2-icon.png) - -You will now see two categories of variables: Locals and Witness Map. - -![Debug pane expanded](@site/static/img/debugger/3-debug-pane.png) - -1. **Locals**: variables of your program. At this point in execution this section is empty, but as we step through the code it will get populated by `x`, `result`, `digest`, etc. - -2. **Witness map**: these are initially populated from your project's `Prover.toml` file. In this example, they will be used to populate `x` and `result` at the beginning of the `main` function. - -Most of the time you will probably be focusing mostly on locals, as they represent the high level state of your program. - -You might be interested in inspecting the witness map in case you are trying to solve a really low level issue in the compiler or runtime itself, so this concerns mostly advanced or niche users. - -Let's step through the program, by using the debugger buttons or their corresponding keyboard shortcuts. - -![Debugger buttons](@site/static/img/debugger/4-debugger-buttons.png) - -Now we can see in the variables pane that there's values for `digest`, `result` and `x`. - -![Inspecting locals](@site/static/img/debugger/5-assert.png) - -We can also inspect the values of variables by directly hovering on them on the code. - -![Hover locals](@site/static/img/debugger/6-hover.png) - -Let's set a break point at the `keccak256` function, so we can continue execution up to the point when it's first invoked without having to go one step at a time. - -We just need to click the to the right of the line number 18. Once the breakpoint appears, we can click the `continue` button or use its corresponding keyboard shortcut (`F5` by default). - -![Breakpoint](@site/static/img/debugger/7-break.png) - -Now we are debugging the `keccak256` function, notice the _Call Stack pane_ at the lower right. This lets us inspect the current call stack of our process. - -That covers most of the current debugger functionalities. Check out [the reference](../../reference/debugger/debugger_vscode.md) for more details on how to configure the debugger. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-oracles.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-oracles.md deleted file mode 100644 index 392dd8b452e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-oracles.md +++ /dev/null @@ -1,275 +0,0 @@ ---- -title: How to use Oracles -description: Learn how to use oracles in your Noir program with examples in both Nargo and NoirJS. This guide also covers writing a JSON RPC server and providing custom foreign call handlers for NoirJS. -keywords: - - Noir Programming - - Oracles - - Nargo - - NoirJS - - JSON RPC Server - - Foreign Call Handlers -sidebar_position: 1 ---- - -This guide shows you how to use oracles in your Noir program. For the sake of clarity, it assumes that: - -- You have read the [explainer on Oracles](../explainers/explainer-oracle.md) and are comfortable with the concept. -- You have a Noir program to add oracles to. You can create one using the [vite-hardhat starter](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) as a boilerplate. -- You understand the concept of a JSON-RPC server. Visit the [JSON-RPC website](https://www.jsonrpc.org/) if you need a refresher. -- You are comfortable with server-side JavaScript (e.g. Node.js, managing packages, etc.). - -## Rundown - -This guide has 3 major steps: - -1. How to modify our Noir program to make use of oracle calls as unconstrained functions -2. How to write a JSON RPC Server to resolve these oracle calls with Nargo -3. How to use them in Nargo and how to provide a custom resolver in NoirJS - -## Step 1 - Modify your Noir program - -An oracle is defined in a Noir program by defining two methods: - -- An unconstrained method - This tells the compiler that it is executing an [unconstrained functions](../noir/concepts//unconstrained.md). -- A decorated oracle method - This tells the compiler that this method is an RPC call. - -An example of an oracle that returns a `Field` would be: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(number: Field) -> Field { } - -unconstrained fn get_sqrt(number: Field) -> Field { - sqrt(number) -} -``` - -In this example, we're wrapping our oracle function in an unconstrained method, and decorating it with `oracle(getSqrt)`. We can then call the unconstrained function as we would call any other function: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); -} -``` - -In the next section, we will make this `getSqrt` (defined on the `sqrt` decorator) be a method of the RPC server Noir will use. - -:::danger - -As explained in the [Oracle Explainer](../explainers/explainer-oracle.md), this `main` function is unsafe unless you constrain its return value. For example: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); - assert(sqrt.pow_32(2) as u64 == input as u64); // <---- constrain the return of an oracle! -} -``` - -::: - -:::info - -Currently, oracles only work with single params or array params. For example: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt([Field; 2]) -> [Field; 2] { } -``` - -::: - -## Step 2 - Write an RPC server - -Brillig will call *one* RPC server. Most likely you will have to write your own, and you can do it in whatever language you prefer. In this guide, we will do it in Javascript. - -Let's use the above example of an oracle that consumes an array with two `Field` and returns their square roots: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(input: [Field; 2]) -> [Field; 2] { } - -unconstrained fn get_sqrt(input: [Field; 2]) -> [Field; 2] { - sqrt(input) -} - -fn main(input: [Field; 2]) { - let sqrt = get_sqrt(input); - assert(sqrt[0].pow_32(2) as u64 == input[0] as u64); - assert(sqrt[1].pow_32(2) as u64 == input[1] as u64); -} - -#[test] -fn test() { - let input = [4, 16]; - main(input); -} -``` - -:::info - -Why square root? - -In general, computing square roots is computationally more expensive than multiplications, which takes a toll when speaking about ZK applications. In this case, instead of calculating the square root in Noir, we are using our oracle to offload that computation to be made in plain. In our circuit we can simply multiply the two values. - -::: - -Now, we should write the correspondent RPC server, starting with the [default JSON-RPC 2.0 boilerplate](https://www.npmjs.com/package/json-rpc-2.0#example): - -```js -import { JSONRPCServer } from "json-rpc-2.0"; -import express from "express"; -import bodyParser from "body-parser"; - -const app = express(); -app.use(bodyParser.json()); - -const server = new JSONRPCServer(); -app.post("/", (req, res) => { - const jsonRPCRequest = req.body; - server.receive(jsonRPCRequest).then((jsonRPCResponse) => { - if (jsonRPCResponse) { - res.json(jsonRPCResponse); - } else { - res.sendStatus(204); - } - }); -}); - -app.listen(5555); -``` - -Now, we will add our `getSqrt` method, as expected by the `#[oracle(getSqrt)]` decorator in our Noir code. It maps through the params array and returns their square roots: - -```js -server.addMethod("resolve_foreign_call", async (params) => { - if (params[0].function !== "getSqrt") { - throw Error("Unexpected foreign call") - }; - const values = params[0].inputs[0].map((field) => { - return `${Math.sqrt(parseInt(field, 16))}`; - }); - return { values: [values] }; -}); -``` - -If you're using Typescript, the following types may be helpful in understanding the expected return value and making sure they're easy to follow: - -```js -export type ForeignCallSingle = string; - -export type ForeignCallArray = string[]; - -export type ForeignCallResult = { - values: (ForeignCallSingle | ForeignCallArray)[]; -}; -``` - -:::info Multidimensional Arrays - -If the Oracle function is returning an array containing other arrays, such as `[['1','2],['3','4']]`, you need to provide the values in JSON as flattened values. In the previous example, it would be `['1', '2', '3', '4']`. In the Noir program, the Oracle signature can use a nested type, the flattened values will be automatically converted to the nested type. - -::: - -## Step 3 - Usage with Nargo - -Using the [`nargo` CLI tool](../getting_started/installation/index.md), you can use oracles in the `nargo test` and `nargo execute` commands by passing a value to `--oracle-resolver`. For example: - -```bash -nargo test --oracle-resolver http://localhost:5555 -``` - -This tells `nargo` to use your RPC Server URL whenever it finds an oracle decorator. - -## Step 4 - Usage with NoirJS - -In a JS environment, an RPC server is not strictly necessary, as you may want to resolve your oracles without needing any JSON call at all. NoirJS simply expects that you pass a callback function when you generate proofs, and that callback function can be anything. - -For example, if your Noir program expects the host machine to provide CPU pseudo-randomness, you could simply pass it as the `foreignCallHandler`. You don't strictly need to create an RPC server to serve pseudo-randomness, as you may as well get it directly in your app: - -```js -const foreignCallHandler = (name, inputs) => crypto.randomBytes(16) // etc - -await noir.execute(inputs, foreignCallHandler) -``` - -As one can see, in NoirJS, the [`foreignCallHandler`](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) function simply means "a callback function that returns a value of type [`ForeignCallOutput`](../reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md). It doesn't have to be an RPC call like in the case for Nargo. - -:::tip - -Does this mean you don't have to write an RPC server like in [Step #2](#step-2---write-an-rpc-server)? - -You don't technically have to, but then how would you run `nargo test`? To use both `Nargo` and `NoirJS` in your development flow, you will have to write a JSON RPC server. - -::: - -In this case, let's make `foreignCallHandler` call the JSON RPC Server we created in [Step #2](#step-2---write-an-rpc-server), by making it a JSON RPC Client. - -For example, using the same `getSqrt` program in [Step #1](#step-1---modify-your-noir-program) (comments in the code): - -```js -import { JSONRPCClient } from "json-rpc-2.0"; - -// declaring the JSONRPCClient -const client = new JSONRPCClient((jsonRPCRequest) => { -// hitting the same JSON RPC Server we coded above - return fetch("http://localhost:5555", { - method: "POST", - headers: { - "content-type": "application/json", - }, - body: JSON.stringify(jsonRPCRequest), - }).then((response) => { - if (response.status === 200) { - return response - .json() - .then((jsonRPCResponse) => client.receive(jsonRPCResponse)); - } else if (jsonRPCRequest.id !== undefined) { - return Promise.reject(new Error(response.statusText)); - } - }); -}); - -// declaring a function that takes the name of the foreign call (getSqrt) and the inputs -const foreignCallHandler = async (name, input) => { - const inputs = input[0].map((i) => i.toString("hex")) - // notice that the "inputs" parameter contains *all* the inputs - // in this case we to make the RPC request with the first parameter "numbers", which would be input[0] - const oracleReturn = await client.request("resolve_foreign_call", [ - { - function: name, - inputs: [inputs] - }, - ]); - return [oracleReturn.values[0]]; -}; - -// the rest of your NoirJS code -const input = { input: [4, 16] }; -const { witness } = await noir.execute(input, foreignCallHandler); -``` - -:::tip - -If you're in a NoirJS environment running your RPC server together with a frontend app, you'll probably hit a familiar problem in full-stack development: requests being blocked by [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) policy. For development only, you can simply install and use the [`cors` npm package](https://www.npmjs.com/package/cors) to get around the problem: - -```bash -yarn add cors -``` - -and use it as a middleware: - -```js -import cors from "cors"; - -const app = express(); -app.use(cors()) -``` - -::: - -## Conclusion - -Hopefully by the end of this guide, you should be able to: - -- Write your own logic around Oracles and how to write a JSON RPC server to make them work with your Nargo commands. -- Provide custom foreign call handlers for NoirJS. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-recursion.md deleted file mode 100644 index c8c4dc9f5b4..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-recursion.md +++ /dev/null @@ -1,180 +0,0 @@ ---- -title: How to use recursion on NoirJS -description: Learn how to implement recursion with NoirJS, a powerful tool for creating smart contracts on the EVM blockchain. This guide assumes familiarity with NoirJS, solidity verifiers, and the Barretenberg proving backend. Discover how to generate both final and intermediate proofs using `noir_js` and `backend_barretenberg`. -keywords: - [ - "NoirJS", - "EVM blockchain", - "smart contracts", - "recursion", - "solidity verifiers", - "Barretenberg backend", - "noir_js", - "backend_barretenberg", - "intermediate proofs", - "final proofs", - "nargo compile", - "json import", - "recursive circuit", - "recursive app" - ] -sidebar_position: 1 ---- - -This guide shows you how to use recursive proofs in your NoirJS app. For the sake of clarity, it is assumed that: - -- You already have a NoirJS app. If you don't, please visit the [NoirJS tutorial](../tutorials/noirjs_app.md) and the [reference](../reference/NoirJS/noir_js/index.md). -- You are familiar with what are recursive proofs and you have read the [recursion explainer](../explainers/explainer-recursion.md) -- You already built a recursive circuit following [the reference](../noir/standard_library/recursion.mdx), and understand how it works. - -It is also assumed that you're not using `noir_wasm` for compilation, and instead you've used [`nargo compile`](../reference/nargo_commands.md) to generate the `json` you're now importing into your project. However, the guide should work just the same if you're using `noir_wasm`. - -:::info - -As you've read in the [explainer](../explainers/explainer-recursion.md), a recursive proof is an intermediate proof. This means that it doesn't necessarily generate the final step that makes it verifiable in a smart contract. However, it is easy to verify within another circuit. - -While "standard" usage of NoirJS packages abstracts final proofs, it currently lacks the necessary interface to abstract away intermediate proofs. This means that these proofs need to be created by using the backend directly. - -In short: - -- `noir_js` generates *only* final proofs -- `backend_barretenberg` generates both types of proofs - -::: - -In a standard recursive app, you're also dealing with at least two circuits. For the purpose of this guide, we will assume the following: - -- `main`: a circuit of type `assert(x != y)`, where `main` is marked with a `#[recursive]` attribute. This attribute states that the backend should generate proofs that are friendly for verification within another circuit. -- `recursive`: a circuit that verifies `main` - -For a full example of how recursive proofs work, please refer to the [noir-examples](https://github.com/noir-lang/noir-examples) repository. We will *not* be using it as a reference for this guide. - -## Step 1: Setup - -In a common NoirJS app, you need to instantiate a backend with something like `const backend = new Backend(circuit)`. Then you feed it to the `noir_js` interface. - -For recursion, this doesn't happen, and the only need for `noir_js` is only to `execute` a circuit and get its witness and return value. Everything else is not interfaced, so it needs to happen on the `backend` object. - -It is also recommended that you instantiate the backend with as many threads as possible, to allow for maximum concurrency: - -```js -const backend = new Backend(circuit, { threads: 8 }) -``` - -:::tip -You can use the [`os.cpus()`](https://nodejs.org/api/os.html#oscpus) object in `nodejs` or [`navigator.hardwareConcurrency`](https://developer.mozilla.org/en-US/docs/Web/API/Navigator/hardwareConcurrency) on the browser to make the most out of those glorious cpu cores -::: - -## Step 2: Generating the witness and the proof for `main` - -After instantiating the backend, you should also instantiate `noir_js`. We will use it to execute the circuit and get the witness. - -```js -const noir = new Noir(circuit) -const { witness } = noir.execute(input) -``` - -With this witness, you are now able to generate the intermediate proof for the main circuit: - -```js -const { proof, publicInputs } = await backend.generateProof(witness) -``` - -:::warning - -Always keep in mind what is actually happening on your development process, otherwise you'll quickly become confused about what circuit we are actually running and why! - -In this case, you can imagine that Alice (running the `main` circuit) is proving something to Bob (running the `recursive` circuit), and Bob is verifying her proof within his proof. - -With this in mind, it becomes clear that our intermediate proof is the one *meant to be verified within another circuit*, so it must be Alice's. Actually, the only final proof in this theoretical scenario would be the last one, sent on-chain. - -::: - -## Step 3 - Verification and proof artifacts - -Optionally, you are able to verify the intermediate proof: - -```js -const verified = await backend.verifyProof({ proof, publicInputs }) -``` - -This can be useful to make sure our intermediate proof was correctly generated. But the real goal is to do it within another circuit. For that, we need to generate recursive proof artifacts that will be passed to the circuit that is verifying the proof we just generated. Instead of passing the proof and verification key as a byte array, we pass them as fields which makes it cheaper to verify in a circuit: - -```js -const { proofAsFields, vkAsFields, vkHash } = await backend.generateRecursiveProofArtifacts( { publicInputs, proof }, publicInputsCount) -``` - -This call takes the public inputs and the proof, but also the public inputs count. While this is easily retrievable by simply counting the `publicInputs` length, the backend interface doesn't currently abstract it away. - -:::info - -The `proofAsFields` has a constant size `[Field; 93]` and verification keys in Barretenberg are always `[Field; 114]`. - -::: - -:::warning - -One common mistake is to forget *who* makes this call. - -In a situation where Alice is generating the `main` proof, if she generates the proof artifacts and sends them to Bob, which gladly takes them as true, this would mean Alice could prove anything! - -Instead, Bob needs to make sure *he* extracts the proof artifacts, using his own instance of the `main` circuit backend. This way, Alice has to provide a valid proof for the correct `main` circuit. - -::: - -## Step 4 - Recursive proof generation - -With the artifacts, generating a recursive proof is no different from a normal proof. You simply use the `backend` (with the recursive circuit) to generate it: - -```js -const recursiveInputs = { - verification_key: vkAsFields, // array of length 114 - proof: proofAsFields, // array of length 93 + size of public inputs - publicInputs: [mainInput.y], // using the example above, where `y` is the only public input - key_hash: vkHash, -} - -const { witness, returnValue } = noir.execute(recursiveInputs) // we're executing the recursive circuit now! -const { proof, publicInputs } = backend.generateProof(witness) -const verified = backend.verifyProof({ proof, publicInputs }) -``` - -You can obviously chain this proof into another proof. In fact, if you're using recursive proofs, you're probably interested of using them this way! - -:::tip - -Managing circuits and "who does what" can be confusing. To make sure your naming is consistent, you can keep them in an object. For example: - -```js -const circuits = { - main: mainJSON, - recursive: recursiveJSON -} -const backends = { - main: new BarretenbergBackend(circuits.main), - recursive: new BarretenbergBackend(circuits.recursive) -} -const noir_programs = { - main: new Noir(circuits.main), - recursive: new Noir(circuits.recursive) -} -``` - -This allows you to neatly call exactly the method you want without conflicting names: - -```js -// Alice runs this 👇 -const { witness: mainWitness } = await noir_programs.main.execute(input) -const proof = await backends.main.generateProof(mainWitness) - -// Bob runs this 👇 -const verified = await backends.main.verifyProof(proof) -const { proofAsFields, vkAsFields, vkHash } = await backends.main.generateRecursiveProofArtifacts( - proof, - numPublicInputs, -); -const { witness: recursiveWitness } = await noir_programs.recursive.execute(recursiveInputs) -const recursiveProof = await backends.recursive.generateProof(recursiveWitness); -``` - -::: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-solidity-verifier.md deleted file mode 100644 index a8169595b3d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/how-to-solidity-verifier.md +++ /dev/null @@ -1,259 +0,0 @@ ---- -title: Generate a Solidity Verifier -description: - Learn how to run the verifier as a smart contract on the blockchain. Compile a Solidity verifier - contract for your Noir program and deploy it on any EVM blockchain acting as a verifier smart - contract. Read more to find out -keywords: - [ - solidity verifier, - smart contract, - blockchain, - compiler, - plonk_vk.sol, - EVM blockchain, - verifying Noir programs, - proving backend, - Barretenberg, - ] -sidebar_position: 0 -pagination_next: tutorials/noirjs_app ---- - -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. - -This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. - -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: - -- You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network -- You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/installation/index.md) with Nargo and the example Hello Noir circuit -- You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. - -## Rundown - -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: - -1. How to generate a solidity smart contract -2. How to compile the smart contract in the RemixIDE -3. How to deploy it to a testnet - -## Step 1 - Generate a contract - -This is by far the most straightforward step. Just run: - -```sh -nargo compile -``` - -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: - -```sh -# Here we pass the path to the newly generated Noir artifact. -bb write_vk -b ./target/.json -bb contract -``` - -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. - -You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/hello_noir/index.md#proving-backend). - -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. - -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: - -## Step 2 - Compiling - -We will mostly skip the details of RemixIDE, as the UI can change from version to version. For now, we can just open -Remix and create a blank workspace. - -![Create Workspace](@site/static/img/how-tos/solidity_verifier_1.png) - -We will create a new file to contain the contract Nargo generated, and copy-paste its content. - -:::warning - -You'll likely see a warning advising you to not trust pasted code. While it is an important warning, it is irrelevant in the context of this guide and can be ignored. We will not be deploying anywhere near a mainnet. - -::: - -To compile our the verifier, we can navigate to the compilation tab: - -![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) - -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: - -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) - -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: - -![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) - -## Step 3 - Deploying - -At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. - -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": - -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) - -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. - -:::note - -Why "UltraVerifier"? - -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. - -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: - -## Step 4 - Verifying - -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: - -```solidity -function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) -``` - -When using the default example in the [Hello Noir](../getting_started/hello_noir/index.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. - -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/hello_noir/index.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the - -```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') - -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS - -echo "Proof:" -echo "0x$HEX_PROOF" -``` - -Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. - -A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): - -```solidity -function castVote(bytes calldata proof, uint proposalId, uint vote, bytes32 nullifierHash) public returns (bool) { - // ... - bytes32[] memory publicInputs = new bytes32[](4); - publicInputs[0] = merkleRoot; - publicInputs[1] = bytes32(proposalId); - publicInputs[2] = bytes32(vote); - publicInputs[3] = nullifierHash; - require(verifier.verify(proof, publicInputs), "Invalid proof"); -``` - -:::info[Return Values] - -A circuit doesn't have the concept of a return value. Return values are just syntactic sugar in Noir. - -Under the hood, the return value is passed as an input to the circuit and is checked at the end of the circuit program. - -For example, if you have Noir program like this: - -```rust -fn main( - // Public inputs - pubkey_x: pub Field, - pubkey_y: pub Field, - // Private inputs - priv_key: Field, -) -> pub Field -``` - -the `verify` function will expect the public inputs array (second function parameter) to be of length 3, the two inputs and the return value. - -Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. - -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. - -::: - -:::tip[Structs] - -You can pass structs to the verifier contract. They will be flattened so that the array of inputs is 1-dimensional array. - -For example, consider the following program: - -```rust -struct Type1 { - val1: Field, - val2: Field, -} - -struct Nested { - t1: Type1, - is_true: bool, -} - -fn main(x: pub Field, nested: pub Nested, y: pub Field) { - //... -} -``` - -The order of these inputs would be flattened to: `[x, nested.t1.val1, nested.t1.val2, nested.is_true, y]` - -::: - -The other function you can call is our entrypoint `verify` function, as defined above. - -:::tip - -It's worth noticing that the `verify` function is actually a `view` function. A `view` function does not alter the blockchain state, so it doesn't need to be distributed (i.e. it will run only on the executing node), and therefore doesn't cost any gas. - -This can be particularly useful in some situations. If Alice generated a proof and wants Bob to verify its correctness, Bob doesn't need to run Nargo, NoirJS, or any Noir specific infrastructure. He can simply make a call to the blockchain with the proof and verify it is correct without paying any gas. - -It would be incorrect to say that a Noir proof verification costs any gas at all. However, most of the time the result of `verify` is used to modify state (for example, to update a balance, a game state, etc). In that case the whole network needs to execute it, which does incur gas costs (calldata and execution, but not storage). - -::: - -## A Note on EVM chains - -Noir proof verification requires the ecMul, ecAdd and ecPairing precompiles. Not all EVM chains support EC Pairings, notably some of the ZK-EVMs. This means that you won't be able to use the verifier contract in all of them. You can find an incomplete list of which EVM chains support these precompiles [here](https://www.evmdiff.com/features?feature=precompiles). - -For example, chains like `zkSync ERA` and `Polygon zkEVM` do not currently support these precompiles, so proof verification via Solidity verifier contracts won't work. Here's a quick list of EVM chains that have been tested and are known to work: - -- Optimism -- Arbitrum -- Polygon PoS -- Scroll -- Celo -- BSC -- Blast L2 -- Avalanche C-Chain -- Mode -- Linea -- Moonbeam - -If you test any other chains, please open a PR on this page to update the list. See [this doc](https://github.com/noir-lang/noir-starter/tree/main/with-foundry#testing-on-chain) for more info about testing verifier contracts on different EVM chains. - -## What's next - -Now that you know how to call a Noir Solidity Verifier on a smart contract using Remix, you should be comfortable with using it with some programmatic frameworks, such as [hardhat](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) and [foundry](https://github.com/noir-lang/noir-starter/tree/main/with-foundry). - -You can find other tools, examples, boilerplates and libraries in the [awesome-noir](https://github.com/noir-lang/awesome-noir) repository. - -You should also be ready to write and deploy your first NoirJS app and start generating proofs on websites, phones, and NodeJS environments! Head on to the [NoirJS tutorial](../tutorials/noirjs_app.md) to learn how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/merkle-proof.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/merkle-proof.mdx deleted file mode 100644 index 0a128adb2de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/merkle-proof.mdx +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Prove Merkle Tree Membership -description: - Learn how to use merkle membership proof in Noir to prove that a given leaf is a member of a - merkle tree with a specified root, at a given index. -keywords: - [merkle proof, merkle membership proof, Noir, rust, hash function, Pedersen, sha256, merkle tree] -sidebar_position: 4 ---- - -Let's walk through an example of a merkle membership proof in Noir that proves that a given leaf is -in a merkle tree. - -```rust - -fn main(message : [Field; 62], index : Field, hashpath : [Field; 40], root : Field) { - let leaf = std::hash::hash_to_field(message.as_slice()); - let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); - assert(merkle_root == root); -} - -``` - -The message is hashed using `hash_to_field`. The specific hash function that is being used is chosen -by the backend. The only requirement is that this hash function can heuristically be used as a -random oracle. If only collision resistance is needed, then one can call `std::hash::pedersen_hash` -instead. - -```rust -let leaf = std::hash::hash_to_field(message.as_slice()); -``` - -The leaf is then passed to a compute_merkle_root function with the root, index and hashpath. The returned root can then be asserted to be the same as the provided root. - -```rust -let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); -assert (merkle_root == root); -``` - -> **Note:** It is possible to re-implement the merkle tree implementation without standard library. -> However, for most usecases, it is enough. In general, the standard library will always opt to be -> as conservative as possible, while striking a balance with efficiency. - -An example, the merkle membership proof, only requires a hash function that has collision -resistance, hence a hash function like Pedersen is allowed, which in most cases is more efficient -than the even more conservative sha256. - -[View an example on the starter repo](https://github.com/noir-lang/noir-examples/blob/3ea09545cabfa464124ec2f3ea8e60c608abe6df/stealthdrop/circuits/src/main.nr#L20) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/using-devcontainers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/using-devcontainers.mdx deleted file mode 100644 index 727ec6ca667..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/how_to/using-devcontainers.mdx +++ /dev/null @@ -1,110 +0,0 @@ ---- -title: Developer Containers and Codespaces -description: "Learn how to set up a devcontainer in your GitHub repository for a seamless coding experience with Codespaces. Follow our easy 8-step guide to create your own Noir environment without installing Nargo locally." -keywords: ["Devcontainer", "Codespaces", "GitHub", "Noir Environment", "Docker Image", "Development Environment", "Remote Coding", "GitHub Codespaces", "Noir Programming", "Nargo", "VSCode Extensions", "Noirup"] -sidebar_position: 1 ---- - -Adding a developer container configuration file to your Noir project is one of the easiest way to unlock coding in browser. - -## What's a devcontainer after all? - -A [Developer Container](https://containers.dev/) (devcontainer for short) is a Docker image that comes preloaded with tools, extensions, and other tools you need to quickly get started or continue a project, without having to install Nargo locally. Think of it as a development environment in a box. - -There are many advantages to this: - -- It's platform and architecture agnostic -- You don't need to have an IDE installed, or Nargo, or use a terminal at all -- It's safer for using on a public machine or public network - -One of the best ways of using devcontainers is... not using your machine at all, for maximum control, performance, and ease of use. -Enter Codespaces. - -## Codespaces - -If a devcontainer is just a Docker image, then what stops you from provisioning a `p3dn.24xlarge` AWS EC2 instance with 92 vCPUs and 768 GiB RAM and using it to prove your 10-gate SNARK proof? - -Nothing! Except perhaps the 30-40$ per hour it will cost you. - -The problem is that provisioning takes time, and I bet you don't want to see the AWS console every time you want to code something real quick. - -Fortunately, there's an easy and free way to get a decent remote machine ready and loaded in less than 2 minutes: Codespaces. [Codespaces is a Github feature](https://github.com/features/codespaces) that allows you to code in a remote machine by using devcontainers, and it's pretty cool: - -- You can start coding Noir in less than a minute -- It uses the resources of a remote machine, so you can code on your grandma's phone if needed be -- It makes it easy to share work with your frens -- It's fully reusable, you can stop and restart whenever you need to - -:::info - -Don't take out your wallet just yet. Free GitHub accounts get about [15-60 hours of coding](https://github.com/features/codespaces) for free per month, depending on the size of your provisioned machine. - -::: - -## Tell me it's _actually_ easy - -It is! - -Github comes with a default codespace and you can use it to code your own devcontainer. That's exactly what we will be doing in this guide. - - - -8 simple steps: - -#### 1. Create a new repository on GitHub. - -#### 2. Click "Start coding with Codespaces". This will use the default image. - -#### 3. Create a folder called `.devcontainer` in the root of your repository. - -#### 4. Create a Dockerfile in that folder, and paste the following code: - -```docker -FROM --platform=linux/amd64 node:lts-bookworm-slim -SHELL ["/bin/bash", "-c"] -RUN apt update && apt install -y curl bash git tar gzip libc++-dev -RUN curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -ENV PATH="/root/.nargo/bin:$PATH" -RUN noirup -ENTRYPOINT ["nargo"] -``` -#### 5. Create a file called `devcontainer.json` in the same folder, and paste the following code: - -```json -{ - "name": "Noir on Codespaces", - "build": { - "context": ".", - "dockerfile": "Dockerfile" - }, - "customizations": { - "vscode": { - "extensions": ["noir-lang.vscode-noir"] - } - } -} -``` -#### 6. Commit and push your changes - -This will pull the new image and build it, so it could take a minute or so - -#### 8. Done! -Just wait for the build to finish, and there's your easy Noir environment. - - -Refer to [noir-starter](https://github.com/noir-lang/noir-starter/) as an example of how devcontainers can be used together with codespaces. - - - -## How do I use it? - -Using the codespace is obviously much easier than setting it up. -Just navigate to your repository and click "Code" -> "Open with Codespaces". It should take a few seconds to load, and you're ready to go. - -:::info - -If you really like the experience, you can add a badge to your readme, links to existing codespaces, and more. -Check out the [official docs](https://docs.github.com/en/codespaces/setting-up-your-project-for-codespaces/setting-up-your-repository/facilitating-quick-creation-and-resumption-of-codespaces) for more info. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/index.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/index.mdx deleted file mode 100644 index a6bd306f91d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/index.mdx +++ /dev/null @@ -1,67 +0,0 @@ ---- -title: Noir Lang -hide_title: true -description: - Learn about the public alpha release of Noir, a domain specific language heavily influenced by Rust that compiles to - an intermediate language which can be compiled to an arithmetic circuit or a rank-1 constraint system. -keywords: - [Noir, - Domain Specific Language, - Rust, - Intermediate Language, - Arithmetic Circuit, - Rank-1 Constraint System, - Ethereum Developers, - Protocol Developers, - Blockchain Developers, - Proving System, - Smart Contract Language] -sidebar_position: 0 ---- - -import Tabs from '@theme/Tabs'; -import TabItem from '@theme/TabItem'; - -Noir Logo - -Noir is an open-source Domain-Specific Language for safe and seamless construction of privacy-preserving Zero-Knowledge programs, requiring no previous knowledge on the underlying mathematics or cryptography. - -ZK programs are programs that can generate short proofs of statements without revealing all inputs to the statements. You can read more about Zero-Knowledge Proofs [here](https://dev.to/spalladino/a-beginners-intro-to-coding-zero-knowledge-proofs-c56). - -## What's new about Noir? - -Noir works differently from most ZK languages by taking a two-pronged path. First, it compiles the program to an adaptable intermediate language known as ACIR. From there, depending on a given project's needs, ACIR can be further compiled into an arithmetic circuit for integration with the proving backend. - -:::info - -Noir is backend agnostic, which means it makes no assumptions on which proving backend powers the ZK proof. Being the language that powers [Aztec Contracts](https://docs.aztec.network/developers/contracts/main), it defaults to Aztec's Barretenberg proving backend. - -However, the ACIR output can be transformed to be compatible with other PLONK-based backends, or into a [rank-1 constraint system](https://www.rareskills.io/post/rank-1-constraint-system) suitable for backends such as Arkwork's Marlin. - -::: - -## Who is Noir for? - -Noir can be used both in complex cloud-based backends and in user's smartphones, requiring no knowledge on the underlying math or cryptography. From authorization systems that keep a password in the user's device, to complex on-chain verification of recursive proofs, Noir is designed to abstract away complexity without any significant overhead. Here are some examples of situations where Noir can be used: - - - - Noir Logo - - Aztec Contracts leverage Noir to allow for the storage and execution of private information. Writing an Aztec Contract is as easy as writing Noir, and Aztec developers can easily interact with the network storage and execution through the [Aztec.nr](https://docs.aztec.network/developers/contracts/main) library. - - - Soliditry Verifier Example - Noir can auto-generate Solidity verifier contracts that verify Noir proofs. This allows for non-interactive verification of proofs containing private information in an immutable system. This feature powers a multitude of use-case scenarios, from P2P chess tournaments, to [Aztec Layer-2 Blockchain](https://docs.aztec.network/) - - - Aztec Labs developed NoirJS, an easy interface to generate and verify Noir proofs in a Javascript environment. This allows for Noir to be used in webpages, mobile apps, games, and any other environment supporting JS execution in a standalone manner. - - - - -## Libraries - -Noir is meant to be easy to extend by simply importing Noir libraries just like in Rust. -The [awesome-noir repo](https://github.com/noir-lang/awesome-noir#libraries) is a collection of libraries developed by the Noir community. -Writing a new library is easy and makes code be composable and easy to reuse. See the section on [dependencies](noir/modules_packages_crates/dependencies.md) for more information. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/migration_notes.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/migration_notes.md deleted file mode 100644 index 6bd740024e5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/migration_notes.md +++ /dev/null @@ -1,105 +0,0 @@ ---- -title: Migration notes -description: Read about migration notes from previous versions, which could solve problems while updating -keywords: [Noir, notes, migration, updating, upgrading] ---- - -Noir is in full-speed development. Things break fast, wild, and often. This page attempts to leave some notes on errors you might encounter when upgrading and how to resolve them until proper patches are built. - -### `backend encountered an error: libc++.so.1` - -Depending on your OS, you may encounter the following error when running `nargo prove` for the first time: - -```text -The backend encountered an error: "/home/codespace/.nargo/backends/acvm-backend-barretenberg/backend_binary: error while loading shared libraries: libc++.so.1: cannot open shared object file: No such file or directory\n" -``` - -Install the `libc++-dev` library with: - -```bash -sudo apt install libc++-dev -``` - -## ≥0.19 - -### Enforcing `compiler_version` - -From this version on, the compiler will check for the `compiler_version` field in `Nargo.toml`, and will error if it doesn't match the current Nargo version in use. - -To update, please make sure this field in `Nargo.toml` matches the output of `nargo --version`. - -## ≥0.14 - -The index of the [for loops](noir/concepts/control_flow.md#loops) is now of type `u64` instead of `Field`. An example refactor would be: - -```rust -for i in 0..10 { - let i = i as Field; -} -``` - -## ≥v0.11.0 and Nargo backend - -From this version onwards, Nargo starts managing backends through the `nargo backend` command. Upgrading to the versions per usual steps might lead to: - -### `backend encountered an error` - -This is likely due to the existing locally installed version of proving backend (e.g. barretenberg) is incompatible with the version of Nargo in use. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo prove -``` - -with your Noir program. - -This will trigger the download and installation of the latest version of barretenberg compatible with your Nargo in use. - -### `backend encountered an error: illegal instruction` - -On certain Intel-based systems, an `illegal instruction` error may arise due to incompatibility of barretenberg with certain CPU instructions. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo backend install acvm-backend-barretenberg https://github.com/noir-lang/barretenberg-js-binary/raw/master/run-bb.tar.gz -``` - -This downloads and installs a specific bb.js based version of barretenberg binary from GitHub. - -The gzipped file is running [this bash script](https://github.com/noir-lang/barretenberg-js-binary/blob/master/run-bb-js.sh), where we need to gzip it as the Nargo currently expect the backend to be zipped up. - -Then run: - -``` -DESIRED_BINARY_VERSION=0.8.1 nargo info -``` - -This overrides the bb native binary with a bb.js node application instead, which should be compatible with most if not all hardware. This does come with the drawback of being generally slower than native binary. - -0.8.1 indicates bb.js version 0.8.1, so if you change that it will update to a different version or the default version in the script if none was supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/_category_.json deleted file mode 100644 index 7da08f8a8c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Concepts", - "position": 0, - "collapsible": true, - "collapsed": true -} \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/assert.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/assert.md deleted file mode 100644 index 2132de42072..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/assert.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Assert Function -description: - Learn about the `assert` and `static_assert` functions in Noir, which can be used to explicitly - constrain the predicate or comparison expression that follows to be true, and what happens if - the expression is false at runtime or compile-time, respectively. -keywords: [Noir programming language, assert statement, predicate expression, comparison expression] -sidebar_position: 4 ---- - -Noir includes a special `assert` function which will explicitly constrain the predicate/comparison -expression that follows to be true. If this expression is false at runtime, the program will fail to -be proven. Example: - -```rust -fn main(x : Field, y : Field) { - assert(x == y); -} -``` - -> Assertions only work for predicate operations, such as `==`. If there's any ambiguity on the operation, the program will fail to compile. For example, it is unclear if `assert(x + y)` would check for `x + y == 0` or simply would return `true`. - -You can optionally provide a message to be logged when the assertion fails: - -```rust -assert(x == y, "x and y are not equal"); -``` - -Aside string literals, the optional message can be a format string or any other type supported as input for Noir's [print](../standard_library/logging.md) functions. This feature lets you incorporate runtime variables into your failed assertion logs: - -```rust -assert(x == y, f"Expected x == y, but got {x} == {y}"); -``` - -Using a variable as an assertion message directly: - -```rust -struct myStruct { - myField: Field -} - -let s = myStruct { myField: y }; -assert(s.myField == x, s); -``` - -There is also a special `static_assert` function that behaves like `assert`, -but that runs at compile-time. - -```rust -fn main(xs: [Field; 3]) { - let x = 2 + 2; - let y = 4; - static_assert(x == y, "expected 2 + 2 to equal 4"); - - // This passes since the length of `xs` is known at compile-time - static_assert(xs.len() == 3, "expected the input to have 3 elements"); -} -``` - -This function fails when passed a dynamic (run-time) argument: - -```rust -fn main(x : Field, y : Field) { - // this fails because `x` is not known at compile-time - static_assert(x == 2, "expected x to be known at compile-time and equal to 2"); - - let mut example_slice = &[]; - if y == 4 { - example_slice = example_slice.push_back(0); - } - - // This fails because the length of `example_slice` is not known at - // compile-time - let error_message = "expected an empty slice, known at compile-time"; - static_assert(example_slice.len() == 0, error_message); -} -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/comments.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/comments.md deleted file mode 100644 index b51a85f5c94..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/comments.md +++ /dev/null @@ -1,33 +0,0 @@ ---- -title: Comments -description: - Learn how to write comments in Noir programming language. A comment is a line of code that is - ignored by the compiler, but it can be read by programmers. Single-line and multi-line comments - are supported in Noir. -keywords: [Noir programming language, comments, single-line comments, multi-line comments] -sidebar_position: 10 ---- - -A comment is a line in your codebase which the compiler ignores, however it can be read by -programmers. - -Here is a single line comment: - -```rust -// This is a comment and is ignored -``` - -`//` is used to tell the compiler to ignore the rest of the line. - -Noir also supports multi-line block comments. Start a block comment with `/*` and end the block with `*/`. - -Noir does not natively support doc comments. You may be able to use [Rust doc comments](https://doc.rust-lang.org/reference/comments.html) in your code to leverage some Rust documentation build tools with Noir code. - -```rust -/* - This is a block comment describing a complex function. -*/ -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/comptime.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/comptime.md deleted file mode 100644 index 50d7e686083..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/comptime.md +++ /dev/null @@ -1,440 +0,0 @@ ---- -title: Compile-time Code & Metaprogramming -description: Learn how to use metaprogramming in Noir to create macros or derive your own traits -keywords: [Noir, comptime, compile-time, metaprogramming, macros, quote, unquote] -sidebar_position: 15 ---- - -# Overview - -Metaprogramming in Noir is comprised of three parts: -1. `comptime` code -2. Quoting and unquoting -3. The metaprogramming API in `std::meta` - -Each of these are explained in more detail in the next sections but the wide picture is that -`comptime` allows us to write code which runs at compile-time. In this `comptime` code we -can quote and unquote snippets of the program, manipulate them, and insert them in other -parts of the program. Comptime functions which do this are said to be macros. Additionally, -there's a compile-time API of built-in types and functions provided by the compiler which allows -for greater analysis and modification of programs. - ---- - -# Comptime - -`comptime` is a new keyword in Noir which marks an item as executing or existing at compile-time. It can be used in several ways: - -- `comptime fn` to define functions which execute exclusively during compile-time. -- `comptime global` to define a global variable which is evaluated at compile-time. - - Unlike runtime globals, `comptime global`s can be mutable. -- `comptime { ... }` to execute a block of statements during compile-time. -- `comptime let` to define a variable whose value is evaluated at compile-time. -- `comptime for` to run a for loop at compile-time. Syntax sugar for `comptime { for .. }`. - -## Scoping - -Note that while in a `comptime` context, any runtime variables _local to the current function_ are never visible. - -## Evaluating - -Evaluation rules of `comptime` follows the normal unconstrained evaluation rules for other Noir code. There are a few things to note though: - -- Certain built-in functions may not be available, although more may be added over time. -- Evaluation order of global items is currently unspecified. For example, given the following two functions we can't guarantee -which `println` will execute first. The ordering of the two printouts will be arbitrary, but should be stable across multiple compilations with the same `nargo` version as long as the program is also unchanged. - -```rust -fn one() { - comptime { println("one"); } -} - -fn two() { - comptime { println("two"); } -} -``` - -- Since evaluation order is unspecified, care should be taken when using mutable globals so that they do not rely on a particular ordering. -For example, using globals to generate unique ids should be fine but relying on certain ids always being produced (especially after edits to the program) should be avoided. -- Although most ordering of globals is unspecified, two are: - - Dependencies of a crate will always be evaluated before the dependent crate. - - Any annotations on a function will be run before the function itself is resolved. This is to allow the annotation to modify the function if necessary. Note that if the - function itself was called at compile-time previously, it will already be resolved and cannot be modified. To prevent accidentally calling functions you wish to modify - at compile-time, it may be helpful to sort your `comptime` annotation functions into a different crate along with any dependencies they require. - -## Lowering - -When a `comptime` value is used in runtime code it must be lowered into a runtime value. This means replacing the expression with the literal that it evaluated to. For example, the code: - -```rust -struct Foo { array: [Field; 2], len: u32 } - -fn main() { - println(comptime { - let mut foo = std::mem::zeroed::(); - foo.array[0] = 4; - foo.len = 1; - foo - }); -} -``` - -will be converted to the following after `comptime` expressions are evaluated: - -```rust -struct Foo { array: [Field; 2], len: u32 } - -fn main() { - println(Foo { array: [4, 0], len: 1 }); -} -``` - -Not all types of values can be lowered. For example, `Type`s and `TypeDefinition`s (among other types) cannot be lowered at all. - -```rust -fn main() { - // There's nothing we could inline here to create a Type value at runtime - // let _ = get_type!(); -} - -comptime fn get_type() -> Type { ... } -``` - ---- - -# (Quasi) Quote - -Macros in Noir are `comptime` functions which return code as a value which is inserted into the call site when it is lowered there. -A code value in this case is of type `Quoted` and can be created by a `quote { ... }` expression. -More specifically, the code value `quote` creates is a token stream - a representation of source code as a series of words, numbers, string literals, or operators. -For example, the expression `quote { Hi "there reader"! }` would quote three tokens: the word "hi", the string "there reader", and an exclamation mark. -You'll note that snippets that would otherwise be invalid syntax can still be quoted. - -When a `Quoted` value is used in runtime code, it is lowered into a `quote { ... }` expression. Since this expression is only valid -in compile-time code however, we'd get an error if we tried this. Instead, we can use macro insertion to insert each token into the -program at that point, and parse it as an expression. To do this, we have to add a `!` after the function name returning the `Quoted` value. -If the value was created locally and there is no function returning it, `std::meta::unquote!(_)` can be used instead. -Calling such a function at compile-time without `!` will just return the `Quoted` value to be further manipulated. For example: - -```rust title="quote-example" showLineNumbers -comptime fn quote_one() -> Quoted { - quote { 1 } - } - - #[test] - fn returning_versus_macro_insertion() { - comptime - { - // let _a: Quoted = quote { 1 }; - let _a: Quoted = quote_one(); - - // let _b: Field = 1; - let _b: Field = quote_one!(); - - // Since integers default to fields, if we - // want a different type we have to explicitly cast - // let _c: i32 = 1 as i32; - let _c: i32 = quote_one!() as i32; - } - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L121-L142 - - -For those familiar with quoting from other languages (primarily lisps), Noir's `quote` is actually a _quasiquote_. -This means we can escape the quoting by using the unquote operator to splice values in the middle of quoted code. - -# Unquote - -The unquote operator `$` is usable within a `quote` expression. -It takes a variable as an argument, evaluates the variable, and splices the resulting value into the quoted token stream at that point. For example, - -```rust -comptime { - let x = 1 + 2; - let y = quote { $x + 4 }; -} -``` - -The value of `y` above will be the token stream containing `3`, `+`, and `4`. We can also use this to combine `Quoted` values into larger token streams: - -```rust -comptime { - let x = quote { 1 + 2 }; - let y = quote { $x + 4 }; -} -``` - -The value of `y` above is now the token stream containing five tokens: `1 + 2 + 4`. - -Note that to unquote something, a variable name _must_ follow the `$` operator in a token stream. -If it is an expression (even a parenthesized one), it will do nothing. Most likely a parse error will be given when the macro is later unquoted. - -Unquoting can also be avoided by escaping the `$` with a backslash: - -``` -comptime { - let x = quote { 1 + 2 }; - - // y contains the four tokens: `$x + 4` - let y = quote { \$x + 4 }; -} -``` - ---- - -# Annotations - -Annotations provide a way to run a `comptime` function on an item in the program. -When you use an annotation, the function with the same name will be called with that item as an argument: - -```rust -#[my_struct_annotation] -struct Foo {} - -comptime fn my_struct_annotation(s: StructDefinition) { - println("Called my_struct_annotation!"); -} - -#[my_function_annotation] -fn foo() {} - -comptime fn my_function_annotation(f: FunctionDefinition) { - println("Called my_function_annotation!"); -} -``` - -Anything returned from one of these functions will be inserted at top-level along with the original item. -Note that expressions are not valid at top-level so you'll get an error trying to return `3` or similar just as if you tried to write a program containing `3; struct Foo {}`. -You can insert other top-level items such as trait impls, structs, or functions this way though. -For example, this is the mechanism used to insert additional trait implementations into the program when deriving a trait impl from a struct: - -```rust title="derive-field-count-example" showLineNumbers -trait FieldCount { - fn field_count() -> u32; - } - - #[derive_field_count] - struct Bar { x: Field, y: [Field; 2] } - - comptime fn derive_field_count(s: StructDefinition) -> Quoted { - let typ = s.as_type(); - let field_count = s.fields().len(); - quote { - impl FieldCount for $typ { - fn field_count() -> u32 { - $field_count - } - } - } - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L144-L163 - - -## Calling annotations with additional arguments - -Arguments may optionally be given to annotations. -When this is done, these additional arguments are passed to the annotation function after the item argument. - -```rust title="annotation-arguments-example" showLineNumbers -#[assert_field_is_type(quote { i32 }.as_type())] - struct MyStruct { my_field: i32 } - - comptime fn assert_field_is_type(s: StructDefinition, typ: Type) { - // Assert the first field in `s` has type `typ` - let fields = s.fields(); - assert_eq(fields[0].1, typ); - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L165-L174 - - -We can also take any number of arguments by adding the `varargs` annotation: - -```rust title="annotation-varargs-example" showLineNumbers -#[assert_three_args(1, 2, 3)] - struct MyOtherStruct { my_other_field: u32 } - - #[varargs] - comptime fn assert_three_args(_s: StructDefinition, args: [Field]) { - assert_eq(args.len(), 3); - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L176-L184 - - ---- - -# Comptime API - -Although `comptime`, `quote`, and unquoting provide a flexible base for writing macros, -Noir's true metaprogramming ability comes from being able to interact with the compiler through a compile-time API. -This API can be accessed through built-in functions in `std::meta` as well as on methods of several `comptime` types. - -The following is an incomplete list of some `comptime` types along with some useful methods on them. - -- `Quoted`: A token stream -- `Type`: The type of a Noir type - - `fn implements(self, constraint: TraitConstraint) -> bool` - - Returns true if `self` implements the given trait constraint -- `Expr`: A syntactically valid expression. Can be used to recur on a program's parse tree to inspect how it is structured. - - Methods: - - `fn as_function_call(self) -> Option<(Expr, [Expr])>` - - If this is a function call expression, return `(function, arguments)` - - `fn as_block(self) -> Option<[Expr]>` - - If this is a block, return each statement in the block -- `FunctionDefinition`: A function definition - - Methods: - - `fn parameters(self) -> [(Quoted, Type)]` - - Returns a slice of `(name, type)` pairs for each parameter -- `StructDefinition`: A struct definition - - Methods: - - `fn as_type(self) -> Type` - - Returns this `StructDefinition` as a `Type`. Any generics are kept as-is - - `fn generics(self) -> [Quoted]` - - Return the name of each generic on this struct - - `fn fields(self) -> [(Quoted, Type)]` - - Return the name and type of each field -- `TraitConstraint`: A trait constraint such as `From` -- `TypedExpr`: A type-checked expression. -- `UnresolvedType`: A syntactic notation that refers to a Noir type that hasn't been resolved yet - -There are many more functions available by exploring the `std::meta` module and its submodules. -Using these methods is the key to writing powerful metaprogramming libraries. - -## `#[use_callers_scope]` - -Since certain functions such as `Quoted::as_type`, `Expression::as_type`, or `Quoted::as_trait_constraint` will attempt -to resolve their contents in a particular scope - it can be useful to change the scope they resolve in. By default -these functions will resolve in the current function's scope which is usually the attribute function they are called in. -If you're working on a library however, this may be a completely different module or crate to the item you're trying to -use the attribute on. If you want to be able to use `Quoted::as_type` to refer to types local to the caller's scope for -example, you can annotate your attribute function with `#[use_callers_scope]`. This will ensure your attribute, and any -closures it uses, can refer to anything in the caller's scope. `#[use_callers_scope]` also works recursively. So if both -your attribute function and a helper function it calls use it, then they can both refer to the same original caller. - ---- - -# Example: Derive - -Using all of the above, we can write a `derive` macro that behaves similarly to Rust's but is not built into the language. -From the user's perspective it will look like this: - -```rust -// Example usage -#[derive(Default, Eq, Ord)] -struct MyStruct { my_field: u32 } -``` - -To implement `derive` we'll have to create a `comptime` function that accepts -a variable amount of traits. - -```rust title="derive_example" showLineNumbers -// These are needed for the unconstrained hashmap we're using to store derive functions -use crate::collections::umap::UHashMap; -use crate::hash::BuildHasherDefault; -use crate::hash::poseidon2::Poseidon2Hasher; - -// A derive function is one that given a struct definition can -// create us a quoted trait impl from it. -pub type DeriveFunction = fn(StructDefinition) -> Quoted; - -// We'll keep a global HANDLERS map to keep track of the derive handler for each trait -comptime mut global HANDLERS: UHashMap> = UHashMap::default(); - -// Given a struct and a slice of traits to derive, create trait impls for each. -// This function is as simple as iterating over the slice, checking if we have a trait -// handler registered for the given trait, calling it, and appending the result. -#[varargs] -pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted { - let mut result = quote {}; - - for trait_to_derive in traits { - let handler = unsafe { - HANDLERS.get(trait_to_derive) - }; - assert(handler.is_some(), f"No derive function registered for `{trait_to_derive}`"); - - let trait_impl = handler.unwrap()(s); - result = quote { $result $trait_impl }; - } - - result -} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L31-L65 - - -Registering a derive function could be done as follows: - -```rust title="derive_via" showLineNumbers -// To register a handler for a trait, just add it to our handlers map -pub comptime fn derive_via(t: TraitDefinition, f: DeriveFunction) { - HANDLERS.insert(t, f); -} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L67-L74 - - -```rust title="big-derive-usage-example" showLineNumbers -// Finally, to register a handler we call the above function as an annotation - // with our handler function. - #[derive_via(derive_do_nothing)] - trait DoNothing { - fn do_nothing(self); - } - - comptime fn derive_do_nothing(s: StructDefinition) -> Quoted { - // This is simplified since we don't handle generics or where clauses! - // In a real example we'd likely also need to introduce each of - // `s.generics()` as well as a trait constraint for each generic - // to ensure they also implement the trait. - let typ = s.as_type(); - quote { - impl DoNothing for $typ { - fn do_nothing(self) { - // Traits can't tell us what to do - println("something"); - } - } - } - } - - // Since `DoNothing` is a simple trait which: - // 1. Only has one method - // 2. Does not have any generics on the trait itself - // We can use `std::meta::make_trait_impl` to help us out. - // This helper function will generate our impl for us along with any - // necessary where clauses and still provides a flexible interface - // for us to work on each field on the struct. - comptime fn derive_do_nothing_alt(s: StructDefinition) -> Quoted { - let trait_name = quote { DoNothing }; - let method_signature = quote { fn do_nothing(self) }; - - // Call `do_nothing` recursively on each field in the struct - let for_each_field = |field_name| quote { self.$field_name.do_nothing(); }; - - // Some traits like Eq want to join each field expression with something like `&`. - // We don't need that here - let join_fields_with = quote {}; - - // The body function is a spot to insert any extra setup/teardown needed. - // We'll insert our println here. Since we recur on each field, we should see - // one println for the struct itself, followed by a println for every field (recursively). - let body = |body| quote { - println("something"); - $body - }; - crate::meta::make_trait_impl( - s, - trait_name, - method_signature, - for_each_field, - join_fields_with, - body - ) - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L186-L244 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/control_flow.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/control_flow.md deleted file mode 100644 index 045d3c3a5f5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/control_flow.md +++ /dev/null @@ -1,77 +0,0 @@ ---- -title: Control Flow -description: - Learn how to use loops and if expressions in the Noir programming language. Discover the syntax - and examples for for loops and if-else statements. -keywords: [Noir programming language, loops, for loop, if-else statements, Rust syntax] -sidebar_position: 2 ---- - -## If Expressions - -Noir supports `if-else` statements. The syntax is most similar to Rust's where it is not required -for the statement's conditional to be surrounded by parentheses. - -```rust -let a = 0; -let mut x: u32 = 0; - -if a == 0 { - if a != 0 { - x = 6; - } else { - x = 2; - } -} else { - x = 5; - assert(x == 5); -} -assert(x == 2); -``` - -## Loops - -Noir has one kind of loop: the `for` loop. `for` loops allow you to repeat a block of code multiple -times. - -The following block of code between the braces is run 10 times. - -```rust -for i in 0..10 { - // do something -} -``` - -The index for loops is of type `u64`. - -### Break and Continue - -In unconstrained code, `break` and `continue` are also allowed in `for` loops. These are only allowed -in unconstrained code since normal constrained code requires that Noir knows exactly how many iterations -a loop may have. `break` and `continue` can be used like so: - -```rust -for i in 0 .. 10 { - println("Iteration start") - - if i == 2 { - continue; - } - - if i == 5 { - break; - } - - println(i); -} -println("Loop end") -``` - -When used, `break` will end the current loop early and jump to the statement after the for loop. In the example -above, the `break` will stop the loop and jump to the `println("Loop end")`. - -`continue` will stop the current iteration of the loop, and jump to the start of the next iteration. In the example -above, `continue` will jump to `println("Iteration start")` when used. Note that the loop continues as normal after this. -The iteration variable `i` is still increased by one as normal when `continue` is used. - -`break` and `continue` cannot currently be used to jump out of more than a single loop at a time. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_bus.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_bus.mdx deleted file mode 100644 index e55e58622ce..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_bus.mdx +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: Data Bus -sidebar_position: 13 ---- -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -The data bus is an optimization that the backend can use to make recursion more efficient. -In order to use it, you must define some inputs of the program entry points (usually the `main()` -function) with the `call_data` modifier, and the return values with the `return_data` modifier. -These modifiers are incompatible with `pub` and `mut` modifiers. - -## Example - -```rust -fn main(mut x: u32, y: call_data u32, z: call_data [u32;4] ) -> return_data u32 { - let a = z[x]; - a+y -} -``` - -As a result, both call_data and return_data will be treated as private inputs and encapsulated into a read-only array each, for the backend to process. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/arrays.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/arrays.md deleted file mode 100644 index bb68e60fe53..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/arrays.md +++ /dev/null @@ -1,274 +0,0 @@ ---- -title: Arrays -description: - Dive into the Array data type in Noir. Grasp its methods, practical examples, and best practices for efficiently using Arrays in your Noir code. -keywords: - [ - noir, - array type, - methods, - examples, - indexing, - ] -sidebar_position: 4 ---- - -An array is one way of grouping together values into one compound type. Array types can be inferred -or explicitly specified via the syntax `[; ]`: - -```rust -fn main(x : Field, y : Field) { - let my_arr = [x, y]; - let your_arr: [Field; 2] = [x, y]; -} -``` - -Here, both `my_arr` and `your_arr` are instantiated as an array containing two `Field` elements. - -Array elements can be accessed using indexing: - -```rust -fn main() { - let a = [1, 2, 3, 4, 5]; - - let first = a[0]; - let second = a[1]; -} -``` - -All elements in an array must be of the same type (i.e. homogeneous). That is, an array cannot group -a `Field` value and a `u8` value together for example. - -You can write mutable arrays, like: - -```rust -fn main() { - let mut arr = [1, 2, 3, 4, 5]; - assert(arr[0] == 1); - - arr[0] = 42; - assert(arr[0] == 42); -} -``` - -You can instantiate a new array of a fixed size with the same value repeated for each element. The following example instantiates an array of length 32 where each element is of type Field and has the value 0. - -```rust -let array: [Field; 32] = [0; 32]; -``` - -Like in Rust, arrays in Noir are a fixed size. However, if you wish to convert an array to a [slice](./slices.mdx), you can just call `as_slice` on your array: - -```rust -let array: [Field; 32] = [0; 32]; -let sl = array.as_slice() -``` - -You can define multidimensional arrays: - -```rust -let array : [[Field; 2]; 2]; -let element = array[0][0]; -``` - -However, multidimensional slices are not supported. For example, the following code will error at compile time: - -```rust -let slice : [[Field]] = &[]; -``` - -## Types - -You can create arrays of primitive types or structs. There is not yet support for nested arrays -(arrays of arrays) or arrays of structs that contain arrays. - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for arrays. -Each of these functions are located within the generic impl `impl [T; N] {`. -So anywhere `self` appears, it refers to the variable `self: [T; N]`. - -### len - -Returns the length of an array - -```rust -fn len(self) -> Field -``` - -example - -```rust -fn main() { - let array = [42, 42]; - assert(array.len() == 2); -} -``` - -### sort - -Returns a new sorted array. The original array remains untouched. Notice that this function will -only work for arrays of fields or integers, not for any arbitrary type. This is because the sorting -logic it uses internally is optimized specifically for these values. If you need a sort function to -sort any type, you should use the function `sort_via` described below. - -```rust -fn sort(self) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32]; - let sorted = arr.sort(); - assert(sorted == [32, 42]); -} -``` - -### sort_via - -Sorts the array with a custom comparison function. The ordering function must return true if the first argument should be sorted to be before the second argument or is equal to the second argument. - -Using this method with an operator like `<` that does not return `true` for equal values will result in an assertion failure for arrays with equal elements. - -```rust -fn sort_via(self, ordering: fn(T, T) -> bool) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32] - let sorted_ascending = arr.sort_via(|a, b| a <= b); - assert(sorted_ascending == [32, 42]); // verifies - - let sorted_descending = arr.sort_via(|a, b| a >= b); - assert(sorted_descending == [32, 42]); // does not verify -} -``` - -### map - -Applies a function to each element of the array, returning a new array containing the mapped elements. - -```rust -fn map(self, f: fn(T) -> U) -> [U; N] -``` - -example - -```rust -let a = [1, 2, 3]; -let b = a.map(|a| a * 2); // b is now [2, 4, 6] -``` - -### fold - -Applies a function to each element of the array, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn(U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the array, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = [1]; -let a2 = [1, 2]; -let a3 = [1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let arr = [2, 2, 2, 2, 2]; - let folded = arr.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn(T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let reduced = arr.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let all = arr.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 5]; - let any = arr.any(|a| a == 5); - assert(any); -} -``` - -### as_str_unchecked - -Converts a byte array of type `[u8; N]` to a string. Note that this performs no UTF-8 validation - -the given array is interpreted as-is as a string. - -```rust -impl [u8; N] { - pub fn as_str_unchecked(self) -> str -} -``` - -example: - -```rust -fn main() { - let hi = [104, 105].as_str_unchecked(); - assert_eq(hi, "hi"); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/booleans.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/booleans.md deleted file mode 100644 index 2507af710e7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/booleans.md +++ /dev/null @@ -1,28 +0,0 @@ ---- -title: Booleans -description: - Delve into the Boolean data type in Noir. Understand its methods, practical examples, and best practices for using Booleans in your Noir programs. -keywords: - [ - noir, - boolean type, - methods, - examples, - logical operations, - ] -sidebar_position: 2 ---- - - -The `bool` type in Noir has two possible values: `true` and `false`: - -```rust -fn main() { - let t = true; - let f: bool = false; -} -``` - -The boolean type is most commonly used in conditionals like `if` expressions and `assert` -statements. More about conditionals is covered in the [Control Flow](../control_flow.md) and -[Assert Function](../assert.md) sections. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/fields.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/fields.md deleted file mode 100644 index 814b996facb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/fields.md +++ /dev/null @@ -1,240 +0,0 @@ ---- -title: Fields -description: - Dive deep into the Field data type in Noir. Understand its methods, practical examples, and best practices to effectively use Fields in your Noir programs. -keywords: - [ - noir, - field type, - methods, - examples, - best practices, - ] -sidebar_position: 0 ---- - -The field type corresponds to the native field type of the proving backend. - -The size of a Noir field depends on the elliptic curve's finite field for the proving backend -adopted. For example, a field would be a 254-bit integer when paired with the default backend that -spans the Grumpkin curve. - -Fields support integer arithmetic and are often used as the default numeric type in Noir: - -```rust -fn main(x : Field, y : Field) { - let z = x + y; -} -``` - -`x`, `y` and `z` are all private fields in this example. Using the `let` keyword we defined a new -private value `z` constrained to be equal to `x + y`. - -If proving efficiency is of priority, fields should be used as a default for solving problems. -Smaller integer types (e.g. `u64`) incur extra range constraints. - -## Methods - -After declaring a Field, you can use these common methods on it: - -### to_le_bits - -Transforms the field into an array of bits, Little Endian. - -```rust title="to_le_bits" showLineNumbers -pub fn to_le_bits(self: Self) -> [u1; N] {} -``` -> Source code: noir_stdlib/src/field/mod.nr#L33-L35 - - -example: - -```rust title="to_le_bits_example" showLineNumbers -fn test_to_le_bits() { - let field = 2; - let bits: [u1; 8] = field.to_le_bits(); - assert_eq(bits, [0, 1, 0, 0, 0, 0, 0, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L255-L261 - - - -### to_be_bits - -Transforms the field into an array of bits, Big Endian. - -```rust title="to_be_bits" showLineNumbers -pub fn to_be_bits(self: Self) -> [u1; N] {} -``` -> Source code: noir_stdlib/src/field/mod.nr#L49-L51 - - -example: - -```rust title="to_be_bits_example" showLineNumbers -fn test_to_be_bits() { - let field = 2; - let bits: [u1; 8] = field.to_be_bits(); - assert_eq(bits, [0, 0, 0, 0, 0, 0, 1, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L246-L252 - - - -### to_le_bytes - -Transforms into an array of bytes, Little Endian - -```rust title="to_le_bytes" showLineNumbers -pub fn to_le_bytes(self: Self) -> [u8; N] { -``` -> Source code: noir_stdlib/src/field/mod.nr#L62-L64 - - -example: - -```rust title="to_le_bytes_example" showLineNumbers -fn test_to_le_bytes() { - let field = 2; - let bits: [u8; 8] = field.to_le_bytes(); - assert_eq(bits, [2, 0, 0, 0, 0, 0, 0, 0]); - assert_eq(Field::from_le_bytes::<8>(bits), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L274-L281 - - -### to_be_bytes - -Transforms into an array of bytes, Big Endian - -```rust title="to_be_bytes" showLineNumbers -pub fn to_be_bytes(self: Self) -> [u8; N] { -``` -> Source code: noir_stdlib/src/field/mod.nr#L95-L97 - - -example: - -```rust title="to_be_bytes_example" showLineNumbers -fn test_to_be_bytes() { - let field = 2; - let bits: [u8; 8] = field.to_be_bytes(); - assert_eq(bits, [0, 0, 0, 0, 0, 0, 0, 2]); - assert_eq(Field::from_be_bytes::<8>(bits), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L264-L271 - - - -### to_le_radix - -Decomposes into an array over the specified base, Little Endian - -```rust title="to_le_radix" showLineNumbers -pub fn to_le_radix(self: Self, radix: u32) -> [u8; N] { - crate::assert_constant(radix); - self.__to_le_radix(radix) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L119-L124 - - - -example: - -```rust title="to_le_radix_example" showLineNumbers -fn test_to_le_radix() { - let field = 2; - let bits: [u8; 8] = field.to_le_radix(256); - assert_eq(bits, [2, 0, 0, 0, 0, 0, 0, 0]); - assert_eq(Field::from_le_bytes::<8>(bits), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L294-L301 - - - -### to_be_radix - -Decomposes into an array over the specified base, Big Endian - -```rust title="to_be_radix" showLineNumbers -pub fn to_be_radix(self: Self, radix: u32) -> [u8; N] { - crate::assert_constant(radix); - self.__to_be_radix(radix) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L126-L131 - - -example: - -```rust title="to_be_radix_example" showLineNumbers -fn test_to_be_radix() { - let field = 2; - let bits: [u8; 8] = field.to_be_radix(256); - assert_eq(bits, [0, 0, 0, 0, 0, 0, 0, 2]); - assert_eq(Field::from_be_bytes::<8>(bits), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L284-L291 - - - -### pow_32 - -Returns the value to the power of the specified exponent - -```rust -fn pow_32(self, exponent: Field) -> Field -``` - -example: - -```rust -fn main() { - let field = 2 - let pow = field.pow_32(4); - assert(pow == 16); -} -``` - -### assert_max_bit_size - -Adds a constraint to specify that the field can be represented with `bit_size` number of bits - -```rust title="assert_max_bit_size" showLineNumbers -pub fn assert_max_bit_size(self, bit_size: u32) { -``` -> Source code: noir_stdlib/src/field/mod.nr#L10-L12 - - -example: - -```rust -fn main() { - let field = 2 - field.assert_max_bit_size(32); -} -``` - -### sgn0 - -Parity of (prime) Field element, i.e. sgn0(x mod p) = 0 if x ∈ \{0, ..., p-1\} is even, otherwise sgn0(x mod p) = 1. - -```rust -fn sgn0(self) -> u1 -``` - - -### lt - -Returns true if the field is less than the other field - -```rust -pub fn lt(self, another: Field) -> bool -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/function_types.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/function_types.md deleted file mode 100644 index f6121af17e2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/function_types.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Function types -sidebar_position: 10 ---- - -Noir supports higher-order functions. The syntax for a function type is as follows: - -```rust -fn(arg1_type, arg2_type, ...) -> return_type -``` - -Example: - -```rust -fn assert_returns_100(f: fn() -> Field) { // f takes no args and returns a Field - assert(f() == 100); -} - -fn main() { - assert_returns_100(|| 100); // ok - assert_returns_100(|| 150); // fails -} -``` - -A function type also has an optional capture environment - this is necessary to support closures. -See [Lambdas](../lambdas.md) for more details. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/index.md deleted file mode 100644 index 11f51e2b65a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/index.md +++ /dev/null @@ -1,126 +0,0 @@ ---- -title: Data Types -description: - Get a clear understanding of the two categories of Noir data types - primitive types and compound - types. Learn about their characteristics, differences, and how to use them in your Noir - programming. -keywords: - [ - noir, - data types, - primitive types, - compound types, - private types, - public types, - ] ---- - -Every value in Noir has a type, which determines which operations are valid for it. - -All values in Noir are fundamentally composed of `Field` elements. For a more approachable -developing experience, abstractions are added on top to introduce different data types in Noir. - -Noir has two category of data types: primitive types (e.g. `Field`, integers, `bool`) and compound -types that group primitive types (e.g. arrays, tuples, structs). Each value can either be private or -public. - -## Private & Public Types - -A **private value** is known only to the Prover, while a **public value** is known by both the -Prover and Verifier. Mark values as `private` when the value should only be known to the prover. All -primitive types (including individual fields of compound types) in Noir are private by default, and -can be marked public when certain values are intended to be revealed to the Verifier. - -> **Note:** For public values defined in Noir programs paired with smart contract verifiers, once -> the proofs are verified on-chain the values can be considered known to everyone that has access to -> that blockchain. - -Public data types are treated no differently to private types apart from the fact that their values -will be revealed in proofs generated. Simply changing the value of a public type will not change the -circuit (where the same goes for changing values of private types as well). - -_Private values_ are also referred to as _witnesses_ sometimes. - -> **Note:** The terms private and public when applied to a type (e.g. `pub Field`) have a different -> meaning than when applied to a function (e.g. `pub fn foo() {}`). -> -> The former is a visibility modifier for the Prover to interpret if a value should be made known to -> the Verifier, while the latter is a visibility modifier for the compiler to interpret if a -> function should be made accessible to external Noir programs like in other languages. - -### pub Modifier - -All data types in Noir are private by default. Types are explicitly declared as public using the -`pub` modifier: - -```rust -fn main(x : Field, y : pub Field) -> pub Field { - x + y -} -``` - -In this example, `x` is **private** while `y` and `x + y` (the return value) are **public**. Note -that visibility is handled **per variable**, so it is perfectly valid to have one input that is -private and another that is public. - -> **Note:** Public types can only be declared through parameters on `main`. - -## Type Aliases - -A type alias is a new name for an existing type. Type aliases are declared with the keyword `type`: - -```rust -type Id = u8; - -fn main() { - let id: Id = 1; - let zero: u8 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can also be used with [generics](../generics.md): - -```rust -type Id = Size; - -fn main() { - let id: Id = 1; - let zero: u32 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can even refer to other aliases. An error will be issued if they form a cycle: - -```rust -// Ok! -type A = B; -type B = Field; - -type Bad1 = Bad2; - -// error: Dependency cycle found -type Bad2 = Bad1; -// ^^^^^^^^^^^ 'Bad2' recursively depends on itself: Bad2 -> Bad1 -> Bad2 -``` - -By default, like functions, type aliases are private to the module the exist in. You can use `pub` -to make the type alias public or `pub(crate)` to make it public to just its crate: - -```rust -// This type alias is now public -pub type Id = u8; -``` - -## Wildcard Type -Noir can usually infer the type of the variable from the context, so specifying the type of a variable is only required when it cannot be inferred. However, specifying a complex type can be tedious, especially when it has multiple generic arguments. Often some of the generic types can be inferred from the context, and Noir only needs a hint to properly infer the other types. We can partially specify a variable's type by using `_` as a marker, indicating where we still want the compiler to infer the type. - -```rust -let a: [_; 4] = foo(b); -``` - - -### BigInt - -You can achieve BigInt functionality using the [Noir BigInt](https://github.com/shuklaayush/noir-bigint) library. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/integers.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/integers.md deleted file mode 100644 index a1d59bf3166..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/integers.md +++ /dev/null @@ -1,156 +0,0 @@ ---- -title: Integers -description: Explore the Integer data type in Noir. Learn about its methods, see real-world examples, and grasp how to efficiently use Integers in your Noir code. -keywords: [noir, integer types, methods, examples, arithmetic] -sidebar_position: 1 ---- - -An integer type is a range constrained field type. -The Noir frontend supports both unsigned and signed integer types. -The allowed sizes are 1, 8, 16, 32 and 64 bits. - -:::info - -When an integer is defined in Noir without a specific type, it will default to `Field`. - -The one exception is for loop indices which default to `u64` since comparisons on `Field`s are not possible. - -::: - -## Unsigned Integers - -An unsigned integer type is specified first with the letter `u` (indicating its unsigned nature) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: u8 = 1; - let y: u8 = 1; - let z = x + y; - assert (z == 2); -} -``` - -The bit size determines the maximum value the integer type can store. For example, a `u8` variable can store a value in the range of 0 to 255 (i.e. $\\2^{8}-1\\$). - -## Signed Integers - -A signed integer type is specified first with the letter `i` (which stands for integer) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: i8 = -1; - let y: i8 = -1; - let z = x + y; - assert (z == -2); -} -``` - -The bit size determines the maximum and minimum range of value the integer type can store. For example, an `i8` variable can store a value in the range of -128 to 127 (i.e. $\\-2^{7}\\$ to $\\2^{7}-1\\$). - -## 128 bits Unsigned Integers - -The built-in structure `U128` allows you to use 128-bit unsigned integers almost like a native integer type. However, there are some differences to keep in mind: -- You cannot cast between a native integer and `U128` -- There is a higher performance cost when using `U128`, compared to a native type. - -Conversion between unsigned integer types and U128 are done through the use of `from_integer` and `to_integer` functions. `from_integer` also accepts the `Field` type as input. - -```rust -fn main() { - let x = U128::from_integer(23); - let y = U128::from_hex("0x7"); - let z = x + y; - assert(z.to_integer() == 30); -} -``` - -`U128` is implemented with two 64 bits limbs, representing the low and high bits, which explains the performance cost. You should expect `U128` to be twice more costly for addition and four times more costly for multiplication. -You can construct a U128 from its limbs: -```rust -fn main(x: u64, y: u64) { - let x = U128::from_u64s_be(x,y); - assert(z.hi == x as Field); - assert(z.lo == y as Field); -} -``` - -Note that the limbs are stored as Field elements in order to avoid unnecessary conversions. -Apart from this, most operations will work as usual: - -```rust -fn main(x: U128, y: U128) { - // multiplication - let c = x * y; - // addition and subtraction - let c = c - x + y; - // division - let c = x / y; - // bit operation; - let c = x & y | y; - // bit shift - let c = x << y; - // comparisons; - let c = x < y; - let c = x == y; -} -``` - -## Overflows - -Computations that exceed the type boundaries will result in overflow errors. This happens with both signed and unsigned integers. For example, attempting to prove: - -```rust -fn main(x: u8, y: u8) { - let z = x + y; -} -``` - -With: - -```toml -x = "255" -y = "1" -``` - -Would result in: - -``` -$ nargo execute -error: Assertion failed: 'attempt to add with overflow' -┌─ ~/src/main.nr:9:13 -│ -│ let z = x + y; -│ ----- -│ -= Call stack: - ... -``` - -A similar error would happen with signed integers: - -```rust -fn main() { - let x: i8 = -118; - let y: i8 = -11; - let z = x + y; -} -``` - -### Wrapping methods - -Although integer overflow is expected to error, some use-cases rely on wrapping. For these use-cases, the standard library provides `wrapping` variants of certain common operations: - -```rust -fn wrapping_add(x: T, y: T) -> T; -fn wrapping_sub(x: T, y: T) -> T; -fn wrapping_mul(x: T, y: T) -> T; -``` - -Example of how it is used: - -```rust - -fn main(x: u8, y: u8) -> pub u8 { - std::wrapping_add(x, y) -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/references.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/references.md deleted file mode 100644 index a5293d11cfb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/references.md +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: References -sidebar_position: 9 ---- - -Noir supports first-class references. References are a bit like pointers: they point to a specific address that can be followed to access the data stored at that address. You can use Rust-like syntax to use pointers in Noir: the `&` operator references the variable, the `*` operator dereferences it. - -Example: - -```rust -fn main() { - let mut x = 2; - - // you can reference x as &mut and pass it to multiplyBy2 - multiplyBy2(&mut x); -} - -// you can access &mut here -fn multiplyBy2(x: &mut Field) { - // and dereference it with * - *x = *x * 2; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/strings.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/strings.md deleted file mode 100644 index 1fdee42425e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/strings.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Strings -description: - Discover the String data type in Noir. Learn about its methods, see real-world examples, and understand how to effectively manipulate and use Strings in Noir. -keywords: - [ - noir, - string type, - methods, - examples, - concatenation, - ] -sidebar_position: 3 ---- - - -The string type is a fixed length value defined with `str`. - -You can use strings in `assert()` functions or print them with -`println()`. See more about [Logging](../../standard_library/logging.md). - -```rust - -fn main(message : pub str<11>, hex_as_string : str<4>) { - println(message); - assert(message == "hello world"); - assert(hex_as_string == "0x41"); -} -``` - -You can convert a `str` to a byte array by calling `as_bytes()` -or a vector by calling `as_bytes_vec()`. - -```rust -fn main() { - let message = "hello world"; - let message_bytes = message.as_bytes(); - let mut message_vec = message.as_bytes_vec(); - assert(message_bytes.len() == 11); - assert(message_bytes[0] == 104); - assert(message_bytes[0] == message_vec.get(0)); -} -``` - -## Escape characters - -You can use escape characters for your strings: - -| Escape Sequence | Description | -|-----------------|-----------------| -| `\r` | Carriage Return | -| `\n` | Newline | -| `\t` | Tab | -| `\0` | Null Character | -| `\"` | Double Quote | -| `\\` | Backslash | - -Example: - -```rust -let s = "Hello \"world" // prints "Hello "world" -let s = "hey \tyou"; // prints "hey you" -``` - -## Raw strings - -A raw string begins with the letter `r` and is optionally delimited by a number of hashes `#`. - -Escape characters are *not* processed within raw strings. All contents are interpreted literally. - -Example: - -```rust -let s = r"Hello world"; -let s = r#"Simon says "hello world""#; - -// Any number of hashes may be used (>= 1) as long as the string also terminates with the same number of hashes -let s = r#####"One "#, Two "##, Three "###, Four "####, Five will end the string."#####; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/structs.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/structs.md deleted file mode 100644 index e529347f27d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/structs.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: Structs -description: - Explore the Struct data type in Noir. Learn about its methods, see real-world examples, and grasp how to effectively define and use Structs in your Noir programs. -keywords: - [ - noir, - struct type, - methods, - examples, - data structures, - ] -sidebar_position: 8 ---- - -A struct also allows for grouping multiple values of different types. Unlike tuples, we can also -name each field. - -> **Note:** The usage of _field_ here refers to each element of the struct and is unrelated to the -> field type of Noir. - -Defining a struct requires giving it a name and listing each field within as `: ` pairs: - -```rust -struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` - -An instance of a struct can then be created with actual values in `: ` pairs in any -order. Struct fields are accessible using their given names: - -```rust -fn main() { - let legs = 4; - - let dog = Animal { - eyes: 2, - hands: 0, - legs, - }; - - let zero = dog.hands; -} -``` - -Structs can also be destructured in a pattern, binding each field to a new variable: - -```rust -fn main() { - let Animal { hands, legs: feet, eyes } = get_octopus(); - - let ten = hands + feet + eyes as u8; -} - -fn get_octopus() -> Animal { - let octopus = Animal { - hands: 0, - legs: 8, - eyes: 2, - }; - - octopus -} -``` - -The new variables can be bound with names different from the original struct field names, as -showcased in the `legs --> feet` binding in the example above. - -By default, like functions, structs are private to the module the exist in. You can use `pub` -to make the struct public or `pub(crate)` to make it public to just its crate: - -```rust -// This struct is now public -pub struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/tuples.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/tuples.md deleted file mode 100644 index 2ec5c9c4113..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/tuples.md +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Tuples -description: - Dive into the Tuple data type in Noir. Understand its methods, practical examples, and best practices for efficiently using Tuples in your Noir code. -keywords: - [ - noir, - tuple type, - methods, - examples, - multi-value containers, - ] -sidebar_position: 7 ---- - -A tuple collects multiple values like an array, but with the added ability to collect values of -different types: - -```rust -fn main() { - let tup: (u8, u64, Field) = (255, 500, 1000); -} -``` - -One way to access tuple elements is via destructuring using pattern matching: - -```rust -fn main() { - let tup = (1, 2); - - let (one, two) = tup; - - let three = one + two; -} -``` - -Another way to access tuple elements is via direct member access, using a period (`.`) followed by -the index of the element we want to access. Index `0` corresponds to the first tuple element, `1` to -the second and so on: - -```rust -fn main() { - let tup = (5, 6, 7, 8); - - let five = tup.0; - let eight = tup.3; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/functions.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/functions.md deleted file mode 100644 index f656cdfd97a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/functions.md +++ /dev/null @@ -1,226 +0,0 @@ ---- -title: Functions -description: - Learn how to declare functions and methods in Noir, a programming language with Rust semantics. - This guide covers parameter declaration, return types, call expressions, and more. -keywords: [Noir, Rust, functions, methods, parameter declaration, return types, call expressions] -sidebar_position: 1 ---- - -Functions in Noir follow the same semantics of Rust, though Noir does not support early returns. - -To declare a function the `fn` keyword is used. - -```rust -fn foo() {} -``` - -By default, functions are visible only within the package they are defined. To make them visible outside of that package (for example, as part of a [library](../modules_packages_crates/crates_and_packages.md#libraries)), you should mark them as `pub`: - -```rust -pub fn foo() {} -``` - -You can also restrict the visibility of the function to only the crate it was defined in, by specifying `pub(crate)`: - -```rust -pub(crate) fn foo() {} //foo can only be called within its crate -``` - -All parameters in a function must have a type and all types are known at compile time. The parameter -is pre-pended with a colon and the parameter type. Multiple parameters are separated using a comma. - -```rust -fn foo(x : Field, y : Field){} -``` - -The return type of a function can be stated by using the `->` arrow notation. The function below -states that the foo function must return a `Field`. If the function returns no value, then the arrow -is omitted. - -```rust -fn foo(x : Field, y : Field) -> Field { - x + y -} -``` - -Note that a `return` keyword is unneeded in this case - the last expression in a function's body is -returned. - -## Main function - -If you're writing a binary, the `main` function is the starting point of your program. You can pass all types of expressions to it, as long as they have a fixed size at compile time: - -```rust -fn main(x : Field) // this is fine: passing a Field -fn main(x : [Field; 2]) // this is also fine: passing a Field with known size at compile-time -fn main(x : (Field, bool)) // 👌: passing a (Field, bool) tuple means size 2 -fn main(x : str<5>) // this is fine, as long as you pass a string of size 5 - -fn main(x : Vec) // can't compile, has variable size -fn main(x : [Field]) // can't compile, has variable size -fn main(....// i think you got it by now -``` - -Keep in mind [tests](../../tooling/testing.md) don't differentiate between `main` and any other function. The following snippet passes tests, but won't compile or prove: - -```rust -fn main(x : [Field]) { - assert(x[0] == 1); -} - -#[test] -fn test_one() { - main(&[1, 2]); -} -``` - -```bash -$ nargo test -[testing] Running 1 test functions -[testing] Testing test_one... ok -[testing] All tests passed - -$ nargo check -The application panicked (crashed). -Message: Cannot have variable sized arrays as a parameter to main -``` - -## Call Expressions - -Calling a function in Noir is executed by using the function name and passing in the necessary -arguments. - -Below we show how to call the `foo` function from the `main` function using a call expression: - -```rust -fn main(x : Field, y : Field) { - let z = foo(x); -} - -fn foo(x : Field) -> Field { - x + x -} -``` - -## Methods - -You can define methods in Noir on any struct type in scope. - -```rust -struct MyStruct { - foo: Field, - bar: Field, -} - -impl MyStruct { - fn new(foo: Field) -> MyStruct { - MyStruct { - foo, - bar: 2, - } - } - - fn sum(self) -> Field { - self.foo + self.bar - } -} - -fn main() { - let s = MyStruct::new(40); - assert(s.sum() == 42); -} -``` - -Methods are just syntactic sugar for functions, so if we wanted to we could also call `sum` as -follows: - -```rust -assert(MyStruct::sum(s) == 42); -``` - -It is also possible to specialize which method is chosen depending on the [generic](./generics.md) type that is used. In this example, the `foo` function returns different values depending on its type: - -```rust -struct Foo {} - -impl Foo { - fn foo(self) -> Field { 1 } -} - -impl Foo { - fn foo(self) -> Field { 2 } -} - -fn main() { - let f1: Foo = Foo{}; - let f2: Foo = Foo{}; - assert(f1.foo() + f2.foo() == 3); -} -``` - -Also note that impls with the same method name defined in them cannot overlap. For example, if we already have `foo` defined for `Foo` and `Foo` like we do above, we cannot also define `foo` in an `impl Foo` since it would be ambiguous which version of `foo` to choose. - -```rust -// Including this impl in the same project as the above snippet would -// cause an overlapping impls error -impl Foo { - fn foo(self) -> Field { 3 } -} -``` - -## Lambdas - -Lambdas are anonymous functions. They follow the syntax of Rust - `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -See [Lambdas](./lambdas.md) for more details. - -## Attributes - -Attributes are metadata that can be applied to a function, using the following syntax: `#[attribute(value)]`. - -Supported attributes include: - -- **builtin**: the function is implemented by the compiler, for efficiency purposes. -- **deprecated**: mark the function as _deprecated_. Calling the function will generate a warning: `warning: use of deprecated function` -- **field**: Used to enable conditional compilation of code depending on the field size. See below for more details -- **oracle**: mark the function as _oracle_; meaning it is an external unconstrained function, implemented in noir_js. See [Unconstrained](./unconstrained.md) and [NoirJS](../../reference/NoirJS/noir_js/index.md) for more details. -- **test**: mark the function as unit tests. See [Tests](../../tooling/testing.md) for more details - -### Field Attribute - -The field attribute defines which field the function is compatible for. The function is conditionally compiled, under the condition that the field attribute matches the Noir native field. -The field can be defined implicitly, by using the name of the elliptic curve usually associated to it - for instance bn254, bls12_381 - or explicitly by using the field (prime) order, in decimal or hexadecimal form. -As a result, it is possible to define multiple versions of a function with each version specialized for a different field attribute. This can be useful when a function requires different parameters depending on the underlying elliptic curve. - -Example: we define the function `foo()` three times below. Once for the default Noir bn254 curve, once for the field $\mathbb F_{23}$, which will normally never be used by Noir, and once again for the bls12_381 curve. - -```rust -#[field(bn254)] -fn foo() -> u32 { - 1 -} - -#[field(23)] -fn foo() -> u32 { - 2 -} - -// This commented code would not compile as foo would be defined twice because it is the same field as bn254 -// #[field(21888242871839275222246405745257275088548364400416034343698204186575808495617)] -// fn foo() -> u32 { -// 2 -// } - -#[field(bls12_381)] -fn foo() -> u32 { - 3 -} -``` - -If the field name is not known to Noir, it will discard the function. Field names are case insensitive. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/globals.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/globals.md deleted file mode 100644 index 1145c55dfc7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/globals.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: Global Variables -description: - Learn about global variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, globals, global variables, constants] -sidebar_position: 8 ---- - -## Globals - - -Noir supports global variables. The global's type can be inferred by the compiler entirely: - -```rust -global N = 5; // Same as `global N: Field = 5` - -global TUPLE = (3, 2); - -fn main() { - assert(N == 5); - assert(N == TUPLE.0 + TUPLE.1); -} -``` - -:::info - -Globals can be defined as any expression, so long as they don't depend on themselves - otherwise there would be a dependency cycle! For example: - -```rust -global T = foo(T); // dependency error -``` - -::: - - -If they are initialized to a literal integer, globals can be used to specify an array's length: - -```rust -global N: u32 = 2; - -fn main(y : [Field; N]) { - assert(y[0] == y[1]) -} -``` - -A global from another module can be imported or referenced externally like any other name: - -```rust -global N = 20; - -fn main() { - assert(my_submodule::N != N); -} - -mod my_submodule { - global N: Field = 10; -} -``` - -When a global is used, Noir replaces the name with its definition on each occurrence. -This means globals defined using function calls will repeat the call each time they're used: - -```rust -global RESULT = foo(); - -fn foo() -> [Field; 100] { ... } -``` - -This is usually fine since Noir will generally optimize any function call that does not -refer to a program input into a constant. It should be kept in mind however, if the called -function performs side-effects like `println`, as these will still occur on each use. - -### Visibility - -By default, like functions, globals are private to the module the exist in. You can use `pub` -to make the global public or `pub(crate)` to make it public to just its crate: - -```rust -// This global is now public -pub global N = 5; -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/lambdas.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/lambdas.md deleted file mode 100644 index be3c7e0b5ca..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/lambdas.md +++ /dev/null @@ -1,81 +0,0 @@ ---- -title: Lambdas -description: Learn how to use anonymous functions in Noir programming language. -keywords: [Noir programming language, lambda, closure, function, anonymous function] -sidebar_position: 9 ---- - -## Introduction - -Lambdas are anonymous functions. The syntax is `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -A block can be used as the body of a lambda, allowing you to declare local variables inside it: - -```rust -let cool = || { - let x = 100; - let y = 100; - x + y -} - -assert(cool() == 200); -``` - -## Closures - -Inside the body of a lambda, you can use variables defined in the enclosing function. Such lambdas are called **closures**. In this example `x` is defined inside `main` and is accessed from within the lambda: - -```rust -fn main() { - let x = 100; - let closure = || x + 150; - assert(closure() == 250); -} -``` - -## Passing closures to higher-order functions - -It may catch you by surprise that the following code fails to compile: - -```rust -fn foo(f: fn () -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // error :( -} -``` - -The reason is that the closure's capture environment affects its type - we have a closure that captures two Fields and `foo` -expects a regular function as an argument - those are incompatible. -:::note - -Variables contained within the `||` are the closure's parameters, and the expression that follows it is the closure's body. The capture environment is comprised of any variables used in the closure's body that are not parameters. - -E.g. in |x| x + y, y would be a captured variable, but x would not be, since it is a parameter of the closure. - -::: -The syntax for the type of a closure is `fn[env](args) -> ret_type`, where `env` is the capture environment of the closure - -in this example that's `(Field, Field)`. - -The best solution in our case is to make `foo` generic over the environment type of its parameter, so that it can be called -with closures with any environment, as well as with regular functions: - -```rust -fn foo(f: fn[Env]() -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // compiles fine - assert(foo(|| 60) == 60); // compiles fine -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/mutability.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/mutability.md deleted file mode 100644 index fdeef6a87c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/mutability.md +++ /dev/null @@ -1,121 +0,0 @@ ---- -title: Mutability -description: - Learn about mutable variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, mutability in noir, mutable variables] -sidebar_position: 8 ---- - -Variables in noir can be declared mutable via the `mut` keyword. Mutable variables can be reassigned -to via an assignment expression. - -```rust -let x = 2; -x = 3; // error: x must be mutable to be assigned to - -let mut y = 3; -let y = 4; // OK -``` - -The `mut` modifier can also apply to patterns: - -```rust -let (a, mut b) = (1, 2); -a = 11; // error: a must be mutable to be assigned to -b = 12; // OK - -let mut (c, d) = (3, 4); -c = 13; // OK -d = 14; // OK - -// etc. -let MyStruct { x: mut y } = MyStruct { x: a }; -// y is now in scope -``` - -Note that mutability in noir is local and everything is passed by value, so if a called function -mutates its parameters then the parent function will keep the old value of the parameters. - -```rust -fn main() -> pub Field { - let x = 3; - helper(x); - x // x is still 3 -} - -fn helper(mut x: i32) { - x = 4; -} -``` - -## Non-local mutability - -Non-local mutability can be achieved through the mutable reference type `&mut T`: - -```rust -fn set_to_zero(x: &mut Field) { - *x = 0; -} - -fn main() { - let mut y = 42; - set_to_zero(&mut y); - assert(*y == 0); -} -``` - -When creating a mutable reference, the original variable being referred to (`y` in this -example) must also be mutable. Since mutable references are a reference type, they must -be explicitly dereferenced via `*` to retrieve the underlying value. Note that this yields -a copy of the value, so mutating this copy will not change the original value behind the -reference: - -```rust -fn main() { - let mut x = 1; - let x_ref = &mut x; - - let mut y = *x_ref; - let y_ref = &mut y; - - x = 2; - *x_ref = 3; - - y = 4; - *y_ref = 5; - - assert(x == 3); - assert(*x_ref == 3); - assert(y == 5); - assert(*y_ref == 5); -} -``` - -Note that types in Noir are actually deeply immutable so the copy that occurs when -dereferencing is only a conceptual copy - no additional constraints will occur. - -Mutable references can also be stored within structs. Note that there is also -no lifetime parameter on these unlike rust. This is because the allocated memory -always lasts the entire program - as if it were an array of one element. - -```rust -struct Foo { - x: &mut Field -} - -impl Foo { - fn incr(mut self) { - *self.x += 1; - } -} - -fn main() { - let foo = Foo { x: &mut 0 }; - foo.incr(); - assert(*foo.x == 1); -} -``` - -In general, you should avoid non-local & shared mutability unless it is needed. Sticking -to only local mutability will improve readability and potentially improve compiler optimizations as well. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/ops.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/ops.md deleted file mode 100644 index c35c36c38a9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/ops.md +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: Logical Operations -description: - Learn about the supported arithmetic and logical operations in the Noir programming language. - Discover how to perform operations on private input types, integers, and booleans. -keywords: - [ - Noir programming language, - supported operations, - arithmetic operations, - logical operations, - predicate operators, - bitwise operations, - short-circuiting, - backend, - ] -sidebar_position: 3 ---- - -# Operations - -## Table of Supported Operations - -| Operation | Description | Requirements | -| :-------- | :------------------------------------------------------------: | -------------------------------------: | -| + | Adds two private input types together | Types must be private input | -| - | Subtracts two private input types together | Types must be private input | -| \* | Multiplies two private input types together | Types must be private input | -| / | Divides two private input types together | Types must be private input | -| ^ | XOR two private input types together | Types must be integer | -| & | AND two private input types together | Types must be integer | -| \| | OR two private input types together | Types must be integer | -| \<\< | Left shift an integer by another integer amount | Types must be integer, shift must be u8 | -| >> | Right shift an integer by another integer amount | Types must be integer, shift must be u8 | -| ! | Bitwise not of a value | Type must be integer or boolean | -| \< | returns a bool if one value is less than the other | Upper bound must have a known bit size | -| \<= | returns a bool if one value is less than or equal to the other | Upper bound must have a known bit size | -| > | returns a bool if one value is more than the other | Upper bound must have a known bit size | -| >= | returns a bool if one value is more than or equal to the other | Upper bound must have a known bit size | -| == | returns a bool if one value is equal to the other | Both types must not be constants | -| != | returns a bool if one value is not equal to the other | Both types must not be constants | - -### Predicate Operators - -`<,<=, !=, == , >, >=` are known as predicate/comparison operations because they compare two values. -This differs from the operations such as `+` where the operands are used in _computation_. - -### Bitwise Operations Example - -```rust -fn main(x : Field) { - let y = x as u32; - let z = y & y; -} -``` - -`z` is implicitly constrained to be the result of `y & y`. The `&` operand is used to denote bitwise -`&`. - -> `x & x` would not compile as `x` is a `Field` and not an integer type. - -### Logical Operators - -Noir has no support for the logical operators `||` and `&&`. This is because encoding the -short-circuiting that these operators require can be inefficient for Noir's backend. Instead you can -use the bitwise operators `|` and `&` which operate identically for booleans, just without the -short-circuiting. - -```rust -let my_val = 5; - -let mut flag = 1; -if (my_val > 6) | (my_val == 0) { - flag = 0; -} -assert(flag == 1); - -if (my_val != 10) & (my_val < 50) { - flag = 0; -} -assert(flag == 0); -``` - -### Shorthand operators - -Noir shorthand operators for most of the above operators, namely `+=, -=, *=, /=, %=, &=, |=, ^=, <<=`, and `>>=`. These allow for more concise syntax. For example: - -```rust -let mut i = 0; -i = i + 1; -``` - -could be written as: - -```rust -let mut i = 0; -i += 1; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/oracles.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/oracles.mdx deleted file mode 100644 index 77a2ac1550a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/oracles.mdx +++ /dev/null @@ -1,29 +0,0 @@ ---- -title: Oracles -description: Dive into how Noir supports Oracles via RPC calls, and learn how to declare an Oracle in Noir with our comprehensive guide. -keywords: - - Noir - - Oracles - - RPC Calls - - Unconstrained Functions - - Programming - - Blockchain -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -Noir has support for Oracles via RPC calls. This means Noir will make an RPC call and use the return value for proof generation. - -Since Oracles are not resolved by Noir, they are [`unconstrained` functions](./unconstrained.md) - -You can declare an Oracle through the `#[oracle()]` flag. Example: - -```rust -#[oracle(get_number_sequence)] -unconstrained fn get_number_sequence(_size: Field) -> [Field] {} -``` - -The timeout for when using an external RPC oracle resolver can be set with the `NARGO_FOREIGN_CALL_TIMEOUT` environment variable. This timeout is in units of milliseconds. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/shadowing.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/shadowing.md deleted file mode 100644 index 5ce6130d201..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/shadowing.md +++ /dev/null @@ -1,44 +0,0 @@ ---- -title: Shadowing -sidebar_position: 12 ---- - -Noir allows for inheriting variables' values and re-declaring them with the same name similar to Rust, known as shadowing. - -For example, the following function is valid in Noir: - -```rust -fn main() { - let x = 5; - - { - let x = x * 2; - assert (x == 10); - } - - assert (x == 5); -} -``` - -In this example, a variable x is first defined with the value 5. - -The local scope that follows shadows the original x, i.e. creates a local mutable x based on the value of the original x. It is given a value of 2 times the original x. - -When we return to the main scope, x once again refers to just the original x, which stays at the value of 5. - -## Temporal mutability - -One way that shadowing is useful, in addition to ergonomics across scopes, is for temporarily mutating variables. - -```rust -fn main() { - let age = 30; - // age = age + 5; // Would error as `age` is immutable by default. - - let mut age = age + 5; // Temporarily mutates `age` with a new value. - - let age = age; // Locks `age`'s mutability again. - - assert (age == 35); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/traits.md deleted file mode 100644 index 5d07e0c68f0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/traits.md +++ /dev/null @@ -1,475 +0,0 @@ ---- -title: Traits -description: - Traits in Noir can be used to abstract out a common interface for functions across - several data types. -keywords: [noir programming language, traits, interfaces, generic, protocol] -sidebar_position: 14 ---- - -## Overview - -Traits in Noir are a useful abstraction similar to interfaces or protocols in other languages. Each trait defines -the interface of several methods contained within the trait. Types can then implement this trait by providing -implementations for these methods. For example in the program: - -```rust -struct Rectangle { - width: Field, - height: Field, -} - -impl Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -fn log_area(r: Rectangle) { - println(r.area()); -} -``` - -We have a function `log_area` to log the area of a `Rectangle`. Now how should we change the program if we want this -function to work on `Triangle`s as well?: - -```rust -struct Triangle { - width: Field, - height: Field, -} - -impl Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Making `log_area` generic over all types `T` would be invalid since not all types have an `area` method. Instead, we can -introduce a new `Area` trait and make `log_area` generic over all types `T` that implement `Area`: - -```rust -trait Area { - fn area(self) -> Field; -} - -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -We also need to explicitly implement `Area` for `Rectangle` and `Triangle`. We can do that by changing their existing -impls slightly. Note that the parameter types and return type of each of our `area` methods must match those defined -by the `Area` trait. - -```rust -impl Area for Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -impl Area for Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Now we have a working program that is generic over any type of Shape that is used! Others can even use this program -as a library with their own types - such as `Circle` - as long as they also implement `Area` for these types. - -## Where Clauses - -As seen in `log_area` above, when we want to create a function or method that is generic over any type that implements -a trait, we can add a where clause to the generic function. - -```rust -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -It is also possible to apply multiple trait constraints on the same variable at once by combining traits with the `+` -operator. Similarly, we can have multiple trait constraints by separating each with a comma: - -```rust -fn foo(elements: [T], thing: U) where - T: Default + Add + Eq, - U: Bar, -{ - let mut sum = T::default(); - - for element in elements { - sum += element; - } - - if sum == T::default() { - thing.bar(); - } -} -``` - -## Generic Implementations - -You can add generics to a trait implementation by adding the generic list after the `impl` keyword: - -```rust -trait Second { - fn second(self) -> Field; -} - -impl Second for (T, Field) { - fn second(self) -> Field { - self.1 - } -} -``` - -You can also implement a trait for every type this way: - -```rust -trait Debug { - fn debug(self); -} - -impl Debug for T { - fn debug(self) { - println(self); - } -} - -fn main() { - 1.debug(); -} -``` - -### Generic Trait Implementations With Where Clauses - -Where clauses can be placed on trait implementations themselves to restrict generics in a similar way. -For example, while `impl Foo for T` implements the trait `Foo` for every type, `impl Foo for T where T: Bar` -will implement `Foo` only for types that also implement `Bar`. This is often used for implementing generic types. -For example, here is the implementation for array equality: - -```rust -impl Eq for [T; let N: u32] where T: Eq { - // Test if two arrays have the same elements. - // Because both arrays must have length N, we know their lengths already match. - fn eq(self, other: Self) -> bool { - let mut result = true; - - for i in 0 .. self.len() { - // The T: Eq constraint is needed to call == on the array elements here - result &= self[i] == other[i]; - } - - result - } -} -``` - -Where clauses can also be placed on struct implementations. -For example, here is a method utilizing a generic type that implements the equality trait. - -```rust -struct Foo { - a: u32, - b: T, -} - -impl Foo where T: Eq { - fn eq(self, other: Self) -> bool { - (self.a == other.a) & self.b.eq(other.b) - } -} -``` - -## Generic Traits - -Traits themselves can also be generic by placing the generic arguments after the trait name. These generics are in -scope of every item within the trait. - -```rust -trait Into { - // Convert `self` to type `T` - fn into(self) -> T; -} -``` - -When implementing generic traits the generic arguments of the trait must be specified. This is also true anytime -when referencing a generic trait (e.g. in a `where` clause). - -```rust -struct MyStruct { - array: [Field; 2], -} - -impl Into<[Field; 2]> for MyStruct { - fn into(self) -> [Field; 2] { - self.array - } -} - -fn as_array(x: T) -> [Field; 2] - where T: Into<[Field; 2]> -{ - x.into() -} - -fn main() { - let array = [1, 2]; - let my_struct = MyStruct { array }; - - assert_eq(as_array(my_struct), array); -} -``` - -### Associated Types and Constants - -Traits also support associated types and constraints which can be thought of as additional generics that are referred to by name. - -Here's an example of a trait with an associated type `Foo` and a constant `Bar`: - -```rust -trait MyTrait { - type Foo; - - let Bar: u32; -} -``` - -Now when we're implementing `MyTrait` we also have to provide values for `Foo` and `Bar`: - -```rust -impl MyTrait for Field { - type Foo = i32; - - let Bar: u32 = 11; -} -``` - -Since associated constants can also be used in a type position, its values are limited to only other -expression kinds allowed in numeric generics. - -Note that currently all associated types and constants must be explicitly specified in a trait constraint. -If we leave out any, we'll get an error that we're missing one: - -```rust -// Error! Constraint is missing associated constant for `Bar` -fn foo(x: T) where T: MyTrait { - ... -} -``` - -Because all associated types and constants must be explicitly specified, they are essentially named generics, -although this is set to change in the future. Future versions of Noir will allow users to elide associated types -in trait constraints similar to Rust. When this is done, you may still refer to their value with the `::AssociatedType` -syntax: - -```rust -// Only valid in future versions of Noir: -fn foo(x: T) where T: MyTrait { - let _: ::Foo = ...; -} -``` - -The type as trait syntax is possible in Noir today but is less useful when each type must be explicitly specified anyway: - -```rust -fn foo(x: T) where T: MyTrait { - // Works, but could just use F directly - let _: >::Foo = ...; - - let _: F = ...; -} -``` - -## Trait Methods With No `self` - -A trait can contain any number of methods, each of which have access to the `Self` type which represents each type -that eventually implements the trait. Similarly, the `self` variable is available as well but is not required to be used. -For example, we can define a trait to create a default value for a type. This trait will need to return the `Self` type -but doesn't need to take any parameters: - -```rust -trait Default { - fn default() -> Self; -} -``` - -Implementing this trait can be done similarly to any other trait: - -```rust -impl Default for Field { - fn default() -> Field { - 0 - } -} - -struct MyType {} - -impl Default for MyType { - fn default() -> Field { - MyType {} - } -} -``` - -However, since there is no `self` parameter, we cannot call it via the method call syntax `object.method()`. -Instead, we'll need to refer to the function directly. This can be done either by referring to the -specific impl `MyType::default()` or referring to the trait itself `Default::default()`. In the later -case, type inference determines the impl that is selected. - -```rust -let my_struct = MyStruct::default(); - -let x: Field = Default::default(); -let result = x + Default::default(); -``` - -:::warning - -```rust -let _ = Default::default(); -``` - -If type inference cannot select which impl to use because of an ambiguous `Self` type, an impl will be -arbitrarily selected. This occurs most often when the result of a trait function call with no parameters -is unused. To avoid this, when calling a trait function with no `self` or `Self` parameters or return type, -always refer to it via the implementation type's namespace - e.g. `MyType::default()`. -This is set to change to an error in future Noir versions. - -::: - -## Default Method Implementations - -A trait can also have default implementations of its methods by giving a body to the desired functions. -Note that this body must be valid for all types that may implement the trait. As a result, the only -valid operations on `self` will be operations valid for any type or other operations on the trait itself. - -```rust -trait Numeric { - fn add(self, other: Self) -> Self; - - // Default implementation of double is (self + self) - fn double(self) -> Self { - self.add(self) - } -} -``` - -When implementing a trait with default functions, a type may choose to implement only the required functions: - -```rust -impl Numeric for Field { - fn add(self, other: Field) -> Field { - self + other - } -} -``` - -Or it may implement the optional methods as well: - -```rust -impl Numeric for u32 { - fn add(self, other: u32) -> u32 { - self + other - } - - fn double(self) -> u32 { - self * 2 - } -} -``` - -## Impl Specialization - -When implementing traits for a generic type it is possible to implement the trait for only a certain combination -of generics. This can be either as an optimization or because those specific generics are required to implement the trait. - -```rust -trait Sub { - fn sub(self, other: Self) -> Self; -} - -struct NonZero { - value: T, -} - -impl Sub for NonZero { - fn sub(self, other: Self) -> Self { - let value = self.value - other.value; - assert(value != 0); - NonZero { value } - } -} -``` - -## Overlapping Implementations - -Overlapping implementations are disallowed by Noir to ensure Noir's decision on which impl to select is never ambiguous. -This means if a trait `Foo` is already implemented -by a type `Bar` for all `T`, then we cannot also have a separate impl for `Bar` (or any other -type argument). Similarly, if there is an impl for all `T` such as `impl Debug for T`, we cannot create -any more impls to `Debug` for other types since it would be ambiguous which impl to choose for any given -method call. - -```rust -trait Trait {} - -// Previous impl defined here -impl Trait for (A, B) {} - -// error: Impl for type `(Field, Field)` overlaps with existing impl -impl Trait for (Field, Field) {} -``` - -## Trait Coherence - -Another restriction on trait implementations is coherence. This restriction ensures other crates cannot create -impls that may overlap with other impls, even if several unrelated crates are used as dependencies in the same -program. - -The coherence restriction is: to implement a trait, either the trait itself or the object type must be declared -in the crate the impl is in. - -In practice this often comes up when using types provided by libraries. If a library provides a type `Foo` that does -not implement a trait in the standard library such as `Default`, you may not `impl Default for Foo` in your own crate. -While restrictive, this prevents later issues or silent changes in the program if the `Foo` library later added its -own impl for `Default`. If you are a user of the `Foo` library in this scenario and need a trait not implemented by the -library your choices are to either submit a patch to the library or use the newtype pattern. - -### The Newtype Pattern - -The newtype pattern gets around the coherence restriction by creating a new wrapper type around the library type -that we cannot create `impl`s for. Since the new wrapper type is defined in our current crate, we can create -impls for any trait we need on it. - -```rust -struct Wrapper { - foo: some_library::Foo, -} - -impl Default for Wrapper { - fn default() -> Wrapper { - Wrapper { - foo: some_library::Foo::new(), - } - } -} -``` - -Since we have an impl for our own type, the behavior of this code will not change even if `some_library` is updated -to provide its own `impl Default for Foo`. The downside of this pattern is that it requires extra wrapping and -unwrapping of values when converting to and from the `Wrapper` and `Foo` types. - -### Visibility - -By default, like functions, traits are private to the module the exist in. You can use `pub` -to make the trait public or `pub(crate)` to make it public to just its crate: - -```rust -// This trait is now public -pub trait Trait {} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/unconstrained.md deleted file mode 100644 index b5221b8d2dd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/unconstrained.md +++ /dev/null @@ -1,104 +0,0 @@ ---- -title: Unconstrained Functions -description: "Learn about what unconstrained functions in Noir are, how to use them and when you'd want to." - -keywords: [Noir programming language, unconstrained, open] -sidebar_position: 5 ---- - -Unconstrained functions are functions which do not constrain any of the included computation and allow for non-deterministic computation. - -## Why? - -Zero-knowledge (ZK) domain-specific languages (DSL) enable developers to generate ZK proofs from their programs by compiling code down to the constraints of an NP complete language (such as R1CS or PLONKish languages). However, the hard bounds of a constraint system can be very limiting to the functionality of a ZK DSL. - -Enabling a circuit language to perform unconstrained execution is a powerful tool. Said another way, unconstrained execution lets developers generate witnesses from code that does not generate any constraints. Being able to execute logic outside of a circuit is critical for both circuit performance and constructing proofs on information that is external to a circuit. - -Fetching information from somewhere external to a circuit can also be used to enable developers to improve circuit efficiency. - -A ZK DSL does not just prove computation, but proves that some computation was handled correctly. Thus, it is necessary that when we switch from performing some operation directly inside of a circuit to inside of an unconstrained environment that the appropriate constraints are still laid down elsewhere in the circuit. - -## Example - -An in depth example might help drive the point home. This example comes from the excellent [post](https://discord.com/channels/1113924620781883405/1124022445054111926/1128747641853972590) by Tom in the Noir Discord. - -Let's look at how we can optimize a function to turn a `u72` into an array of `u8`s. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u72 & 0xff) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 91 -Backend circuit size: 3619 -``` - -A lot of the operations in this function are optimized away by the compiler (all the bit-shifts turn into divisions by constants). However we can save a bunch of gates by casting to u8 a bit earlier. This automatically truncates the bit-shifted value to fit in a u8 which allows us to remove the AND against 0xff. This saves us ~480 gates in total. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 75 -Backend circuit size: 3143 -``` - -Those are some nice savings already but we can do better. This code is all constrained so we're proving every step of calculating out using num, but we don't actually care about how we calculate this, just that it's correct. This is where brillig comes in. - -It turns out that truncating a u72 into a u8 is hard to do inside a snark, each time we do as u8 we lay down 4 ACIR opcodes which get converted into multiple gates. It's actually much easier to calculate num from out than the other way around. All we need to do is multiply each element of out by a constant and add them all together, both relatively easy operations inside a snark. - -We can then run `u72_to_u8` as unconstrained brillig code in order to calculate out, then use that result in our constrained function and assert that if we were to do the reverse calculation we'd get back num. This looks a little like the below: - -```rust -fn main(num: u72) -> pub [u8; 8] { - let out = unsafe { - u72_to_u8(num) - }; - - let mut reconstructed_num: u72 = 0; - for i in 0..8 { - reconstructed_num += (out[i] as u72 << (56 - (8 * i))); - } - assert(num == reconstructed_num); - out -} - -unconstrained fn u72_to_u8(num: u72) -> [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8))) as u8; - } - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 78 -Backend circuit size: 2902 -``` - -This ends up taking off another ~250 gates from our circuit! We've ended up with more ACIR opcodes than before but they're easier for the backend to prove (resulting in fewer gates). - -Note that in order to invoke unconstrained functions we need to wrap them in an `unsafe` block, -to make it clear that the call is unconstrained. - -Generally we want to use brillig whenever there's something that's easy to verify but hard to compute within the circuit. For example, if you wanted to calculate a square root of a number it'll be a much better idea to calculate this in brillig and then assert that if you square the result you get back your number. - -## Break and Continue - -In addition to loops over runtime bounds, `break` and `continue` are also available in unconstrained code. See [break and continue](../concepts/control_flow.md#break-and-continue) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/_category_.json deleted file mode 100644 index 1debcfe7675..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Modules, Packages and Crates", - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/crates_and_packages.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/crates_and_packages.md deleted file mode 100644 index 95ee9f52ab2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/crates_and_packages.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Crates and Packages -description: Learn how to use Crates and Packages in your Noir project -keywords: [Nargo, dependencies, package management, crates, package] -sidebar_position: 0 ---- - -## Crates - -A crate is the smallest amount of code that the Noir compiler considers at a time. -Crates can contain modules, and the modules may be defined in other files that get compiled with the crate, as we’ll see in the coming sections. - -### Crate Types - -A Noir crate can come in several forms: binaries, libraries or contracts. - -#### Binaries - -_Binary crates_ are programs which you can compile to an ACIR circuit which you can then create proofs against. Each must have a function called `main` that defines the ACIR circuit which is to be proved. - -#### Libraries - -_Library crates_ don't have a `main` function and they don't compile down to ACIR. Instead they define functionality intended to be shared with multiple projects, and eventually included in a binary crate. - -#### Contracts - -Contract crates are similar to binary crates in that they compile to ACIR which you can create proofs against. They are different in that they do not have a single `main` function, but are a collection of functions to be deployed to the [Aztec network](https://aztec.network). You can learn more about the technical details of Aztec in the [monorepo](https://github.com/AztecProtocol/aztec-packages) or contract [examples](https://github.com/AztecProtocol/aztec-packages/tree/master/noir-projects/noir-contracts/contracts). - -### Crate Root - -Every crate has a root, which is the source file that the compiler starts, this is also known as the root module. The Noir compiler does not enforce any conditions on the name of the file which is the crate root, however if you are compiling via Nargo the crate root must be called `lib.nr` or `main.nr` for library or binary crates respectively. - -## Packages - -A Nargo _package_ is a collection of one of more crates that provides a set of functionality. A package must include a Nargo.toml file. - -A package _must_ contain either a library or a binary crate, but not both. - -### Differences from Cargo Packages - -One notable difference between Rust's Cargo and Noir's Nargo is that while Cargo allows a package to contain an unlimited number of binary crates and a single library crate, Nargo currently only allows a package to contain a single crate. - -In future this restriction may be lifted to allow a Nargo package to contain both a binary and library crate or multiple binary crates. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/dependencies.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/dependencies.md deleted file mode 100644 index 24e02de08fe..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/dependencies.md +++ /dev/null @@ -1,124 +0,0 @@ ---- -title: Dependencies -description: - Learn how to specify and manage dependencies in Nargo, allowing you to upload packages to GitHub - and use them easily in your project. -keywords: [Nargo, dependencies, GitHub, package management, versioning] -sidebar_position: 1 ---- - -Nargo allows you to upload packages to GitHub and use them as dependencies. - -## Specifying a dependency - -Specifying a dependency requires a tag to a specific commit and the git url to the url containing -the package. - -Currently, there are no requirements on the tag contents. If requirements are added, it would follow -semver 2.0 guidelines. - -> Note: Without a `tag` , there would be no versioning and dependencies would change each time you -> compile your project. - -For example, to add the [ecrecover-noir library](https://github.com/colinnielsen/ecrecover-noir) to your project, add it to `Nargo.toml`: - -```toml -# Nargo.toml - -[dependencies] -ecrecover = {tag = "v0.8.0", git = "https://github.com/colinnielsen/ecrecover-noir"} -``` - -If the module is in a subdirectory, you can define a subdirectory in your git repository, for example: - -```toml -# Nargo.toml - -[dependencies] -easy_private_token_contract = {tag ="v0.1.0-alpha62", git = "https://github.com/AztecProtocol/aztec-packages", directory = "noir-contracts/contracts/easy_private_token_contract"} -``` - -## Specifying a local dependency - -You can also specify dependencies that are local to your machine. - -For example, this file structure has a library and binary crate - -```tree -├── binary_crate -│   ├── Nargo.toml -│   └── src -│   └── main.nr -└── lib_a - ├── Nargo.toml - └── src - └── lib.nr -``` - -Inside of the binary crate, you can specify: - -```toml -# Nargo.toml - -[dependencies] -lib_a = { path = "../lib_a" } -``` - -## Importing dependencies - -You can import a dependency to a Noir file using the following syntax. For example, to import the -ecrecover-noir library and local lib_a referenced above: - -```rust -use ecrecover; -use lib_a; -``` - -You can also import only the specific parts of dependency that you want to use, like so: - -```rust -use std::hash::sha256; -use std::scalar_mul::fixed_base_embedded_curve; -``` - -Lastly, as demonstrated in the -[elliptic curve example](../standard_library/cryptographic_primitives/ec_primitives.md#examples), you -can import multiple items in the same line by enclosing them in curly braces: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; -``` - -We don't have a way to consume libraries from inside a [workspace](./workspaces.md) as external dependencies right now. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -## Dependencies of Dependencies - -Note that when you import a dependency, you also get access to all of the dependencies of that package. - -For example, the [phy_vector](https://github.com/resurgencelabs/phy_vector) library imports an [fraction](https://github.com/resurgencelabs/fraction) library. If you're importing the phy_vector library, then you can access the functions in fractions library like so: - -```rust -use phy_vector; - -fn main(x : Field, y : pub Field) { - //... - let f = phy_vector::fraction::toFraction(true, 2, 1); - //... -} -``` - -## Available Libraries - -Noir does not currently have an official package manager. You can find a list of available Noir libraries in the [awesome-noir repo here](https://github.com/noir-lang/awesome-noir#libraries). - -Some libraries that are available today include: - -- [Standard Library](https://github.com/noir-lang/noir/tree/master/noir_stdlib) - the Noir Standard Library -- [Ethereum Storage Proof Verification](https://github.com/aragonzkresearch/noir-trie-proofs) - a library that contains the primitives necessary for RLP decoding (in the form of look-up table construction) and Ethereum state and storage proof verification (or verification of any trie proof involving 32-byte long keys) -- [BigInt](https://github.com/shuklaayush/noir-bigint) - a library that provides a custom BigUint56 data type, allowing for computations on large unsigned integers -- [ECrecover](https://github.com/colinnielsen/ecrecover-noir/tree/main) - a library to verify an ECDSA signature and return the source Ethereum address -- [Sparse Merkle Tree Verifier](https://github.com/vocdoni/smtverifier-noir/tree/main) - a library for verification of sparse Merkle trees -- [Signed Int](https://github.com/resurgencelabs/signed_int) - a library for accessing a custom Signed Integer data type, allowing access to negative numbers on Noir -- [Fraction](https://github.com/resurgencelabs/fraction) - a library for accessing fractional number data type in Noir, allowing results that aren't whole numbers diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/modules.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/modules.md deleted file mode 100644 index 05399c38b4c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/modules.md +++ /dev/null @@ -1,221 +0,0 @@ ---- -title: Modules -description: - Learn how to organize your files using modules in Noir, following the same convention as Rust's - module system. Examples included. -keywords: [Noir, Rust, modules, organizing files, sub-modules] -sidebar_position: 2 ---- - -Noir's module system follows the same convention as the _newer_ version of Rust's module system. - -## Purpose of Modules - -Modules are used to organize files. Without modules all of your code would need to live in a single -file. In Noir, the compiler does not automatically scan all of your files to detect modules. This -must be done explicitly by the developer. - -## Examples - -### Importing a module in the crate root - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo.nr` - -```rust -fn from_foo() {} -``` - -In the above snippet, the crate root is the `src/main.nr` file. The compiler sees the module -declaration `mod foo` which prompts it to look for a foo.nr file. - -Visually this module hierarchy looks like the following : - -``` -crate - ├── main - │ - └── foo - └── from_foo - -``` - -The module filename may also be the name of the module as a directory with the contents in a -file named `mod.nr` within that directory. The above example can alternatively be expressed like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -fn from_foo() {} -``` - -Note that it's an error to have both files `src/foo.nr` and `src/foo/mod.nr` in the filesystem. - -### Importing a module throughout the tree - -All modules are accessible from the `crate::` namespace. - -``` -crate - ├── bar - ├── foo - └── main - -``` - -In the above snippet, if `bar` would like to use functions in `foo`, it can do so by `use crate::foo::function_name`. - -### Sub-modules - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -fn from_bar() {} -``` - -In the above snippet, we have added an extra module to the module tree; `bar`. `bar` is a submodule -of `foo` hence we declare bar in `foo.nr` with `mod bar`. Since `foo` is not the crate root, the -compiler looks for the file associated with the `bar` module in `src/foo/bar.nr` - -Visually the module hierarchy looks as follows: - -``` -crate - ├── main - │ - └── foo - ├── from_foo - └── bar - └── from_bar -``` - -Similar to importing a module in the crate root, modules can be placed in a `mod.nr` file, like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar/mod.nr` - -```rust -fn from_bar() {} -``` - -### Referencing a parent module - -Given a submodule, you can refer to its parent module using the `super` keyword. - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; - -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -// Same as bar::from_foo -use super::from_foo; - -fn from_bar() { - from_foo(); // invokes super::from_foo(), which is bar::from_foo() - super::from_foo(); // also invokes bar::from_foo() -} -``` - -### `use` visibility - -`use` declarations are private to the containing module, by default. However, like functions, -they can be marked as `pub` or `pub(crate)`. Such a use declaration serves to _re-export_ a name. -A public `use` declaration can therefore redirect some public name to a different target definition: -even a definition with a private canonical path, inside a different module. - -An example of re-exporting: - -```rust -mod some_module { - pub use foo::{bar, baz}; - mod foo { - pub fn bar() {} - pub fn baz() {} - } -} - -fn main() { - some_module::bar(); - some_module::baz(); -} -``` - -In this example, the module `some_module` re-exports two public names defined in `foo`. - -### Visibility - -By default, like functions, modules are private to the module (or crate) the exist in. You can use `pub` -to make the module public or `pub(crate)` to make it public to just its crate: - -```rust -// This module is now public and can be seen by other crates. -pub mod foo; -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/workspaces.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/workspaces.md deleted file mode 100644 index 513497f12bf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/modules_packages_crates/workspaces.md +++ /dev/null @@ -1,42 +0,0 @@ ---- -title: Workspaces -sidebar_position: 3 ---- - -Workspaces are a feature of nargo that allow you to manage multiple related Noir packages in a single repository. A workspace is essentially a group of related projects that share common build output directories and configurations. - -Each Noir project (with it's own Nargo.toml file) can be thought of as a package. Each package is expected to contain exactly one "named circuit", being the "name" defined in Nargo.toml with the program logic defined in `./src/main.nr`. - -For a project with the following structure: - -```tree -├── crates -│ ├── a -│ │ ├── Nargo.toml -│ │ └── Prover.toml -│ │ └── src -│ │ └── main.nr -│ └── b -│ ├── Nargo.toml -│ └── Prover.toml -│ └── src -│ └── main.nr -│ -└── Nargo.toml -``` - -You can define a workspace in Nargo.toml like so: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -`members` indicates which packages are included in the workspace. As such, all member packages of a workspace will be processed when the `--workspace` flag is used with various commands or if a `default-member` is not specified. - -`default-member` indicates which package various commands process by default. - -Libraries can be defined in a workspace. Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/_category_.json deleted file mode 100644 index af04c0933fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Standard Library", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/bigint.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/bigint.md deleted file mode 100644 index b16fed4ed96..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/bigint.md +++ /dev/null @@ -1,127 +0,0 @@ ---- -title: Big Integers -description: How to use big integers from Noir standard library -keywords: - [ - Big Integer, - Noir programming language, - Noir libraries, - ] ---- - -The BigInt module in the standard library exposes some class of integers which do not fit (well) into a Noir native field. It implements modulo arithmetic, modulo a 'big' prime number. - -:::note - -The module can currently be considered as `Field`s with fixed modulo sizes used by a set of elliptic curves, in addition to just the native curve. [More work](https://github.com/noir-lang/noir/issues/510) is needed to achieve arbitrarily sized big integers. - -:::note - -`nargo` can be built with `--profile release-pedantic` to enable extra overflow checks which may affect `BigInt` results in some cases. -Consider the [`noir-bignum`](https://github.com/noir-lang/noir-bignum) library for an optimized alternative approach. - -::: - -Currently 6 classes of integers (i.e 'big' prime numbers) are available in the module, namely: - -- BN254 Fq: Bn254Fq -- BN254 Fr: Bn254Fr -- Secp256k1 Fq: Secpk1Fq -- Secp256k1 Fr: Secpk1Fr -- Secp256r1 Fr: Secpr1Fr -- Secp256r1 Fq: Secpr1Fq - -Where XXX Fq and XXX Fr denote respectively the order of the base and scalar field of the (usual) elliptic curve XXX. -For instance the big integer 'Secpk1Fq' in the standard library refers to integers modulo $2^{256}-2^{32}-977$. - -Feel free to explore the source code for the other primes: - -```rust title="big_int_definition" showLineNumbers -pub struct BigInt { - pointer: u32, - modulus: u32, -} -``` -> Source code: noir_stdlib/src/bigint.nr#L14-L19 - - -## Example usage - -A common use-case is when constructing a big integer from its bytes representation, and performing arithmetic operations on it: - -```rust title="big_int_example" showLineNumbers -fn big_int_example(x: u8, y: u8) { - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - let b = Secpk1Fq::from_le_bytes(&[y, x, 9]); - let c = (a + b) * b / a; - let d = c.to_le_bytes(); - println(d[0]); -} -``` -> Source code: test_programs/execution_success/bigint/src/main.nr#L72-L80 - - -## Methods - -The available operations for each big integer are: - -### from_le_bytes - -Construct a big integer from its little-endian bytes representation. Example: - -```rust - // Construct a big integer from a slice of bytes - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - // Construct a big integer from an array of 32 bytes - let a = Secpk1Fq::from_le_bytes_32([1;32]); - ``` - -Sure, here's the formatted version of the remaining methods: - -### to_le_bytes - -Return the little-endian bytes representation of a big integer. Example: - -```rust -let bytes = a.to_le_bytes(); -``` - -### add - -Add two big integers. Example: - -```rust -let sum = a + b; -``` - -### sub - -Subtract two big integers. Example: - -```rust -let difference = a - b; -``` - -### mul - -Multiply two big integers. Example: - -```rust -let product = a * b; -``` - -### div - -Divide two big integers. Note that division is field division and not euclidean division. Example: - -```rust -let quotient = a / b; -``` - -### eq - -Compare two big integers. Example: - -```rust -let are_equal = a == b; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/black_box_fns.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/black_box_fns.md deleted file mode 100644 index d6079ab182c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/black_box_fns.md +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Black Box Functions -description: Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. -keywords: [noir, black box functions] ---- - -Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. This makes certain zk-snark unfriendly computations cheaper than if they were implemented in Noir. - -The ACVM spec defines a set of blackbox functions which backends will be expected to implement. This allows backends to use optimized implementations of these constraints if they have them, however they may also fallback to less efficient naive implementations if not. - -## Function list - -Here is a list of the current black box functions: - -- [AES128](./cryptographic_primitives/ciphers.mdx#aes128) -- [SHA256](./cryptographic_primitives/hashes.mdx#sha256) -- [Schnorr signature verification](./cryptographic_primitives/schnorr.mdx) -- [Blake2s](./cryptographic_primitives/hashes.mdx#blake2s) -- [Blake3](./cryptographic_primitives/hashes.mdx#blake3) -- [Pedersen Hash](./cryptographic_primitives/hashes.mdx#pedersen_hash) -- [Pedersen Commitment](./cryptographic_primitives/hashes.mdx#pedersen_commitment) -- [ECDSA signature verification](./cryptographic_primitives/ecdsa_sig_verification.mdx) -- [Embedded curve operations (MSM, addition, ...)](./cryptographic_primitives/embedded_curve_ops.mdx) -- AND -- XOR -- RANGE -- [Keccak256](./cryptographic_primitives/hashes.mdx#keccak256) -- [Recursive proof verification](./recursion.mdx) - -Most black box functions are included as part of the Noir standard library, however `AND`, `XOR` and `RANGE` are used as part of the Noir language syntax. For instance, using the bitwise operator `&` will invoke the `AND` black box function. - -You can view the black box functions defined in the ACVM code [here](https://github.com/noir-lang/noir/blob/master/acvm-repo/acir/src/circuit/black_box_functions.rs). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/bn254.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/bn254.md deleted file mode 100644 index 3294f005dbb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/bn254.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Bn254 Field Library ---- - -Noir provides a module in standard library with some optimized functions for bn254 Fr in `std::field::bn254`. - -## decompose - -```rust -fn decompose(x: Field) -> (Field, Field) {} -``` - -Decomposes a single field into two fields, low and high. The low field contains the lower 16 bytes of the input field and the high field contains the upper 16 bytes of the input field. Both field results are range checked to 128 bits. - - -## assert_gt - -```rust -fn assert_gt(a: Field, b: Field) {} -``` - -Asserts that a > b. This will generate less constraints than using `assert(gt(a, b))`. - -## assert_lt - -```rust -fn assert_lt(a: Field, b: Field) {} -``` - -Asserts that a < b. This will generate less constraints than using `assert(lt(a, b))`. - -## gt - -```rust -fn gt(a: Field, b: Field) -> bool {} -``` - -Returns true if a > b. - -## lt - -```rust -fn lt(a: Field, b: Field) -> bool {} -``` - -Returns true if a < b. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/boundedvec.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/boundedvec.md deleted file mode 100644 index b20b49f863c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/boundedvec.md +++ /dev/null @@ -1,419 +0,0 @@ ---- -title: Bounded Vectors -keywords: [noir, vector, bounded vector, slice] -sidebar_position: 1 ---- - -A `BoundedVec` is a growable storage similar to a `Vec` except that it -is bounded with a maximum possible length. Unlike `Vec`, `BoundedVec` is not implemented -via slices and thus is not subject to the same restrictions slices are (notably, nested -slices - and thus nested vectors as well - are disallowed). - -Since a BoundedVec is backed by a normal array under the hood, growing the BoundedVec by -pushing an additional element is also more efficient - the length only needs to be increased -by one. - -For these reasons `BoundedVec` should generally be preferred over `Vec` when there -is a reasonable maximum bound that can be placed on the vector. - -Example: - -```rust -let mut vector: BoundedVec = BoundedVec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -assert(vector.max_len() == 10); -``` - -## Methods - -### new - -```rust -pub fn new() -> Self -``` - -Creates a new, empty vector of length zero. - -Since this container is backed by an array internally, it still needs an initial value -to give each element. To resolve this, each element is zeroed internally. This value -is guaranteed to be inaccessible unless `get_unchecked` is used. - -Example: - -```rust -let empty_vector: BoundedVec = BoundedVec::new(); -assert(empty_vector.len() == 0); -``` - -Note that whenever calling `new` the maximum length of the vector should always be specified -via a type signature: - -```rust title="new_example" showLineNumbers -fn good() -> BoundedVec { - // Ok! MaxLen is specified with a type annotation - let v1: BoundedVec = BoundedVec::new(); - let v2 = BoundedVec::new(); - - // Ok! MaxLen is known from the type of `good`'s return value - v2 -} - -fn bad() { - // Error: Type annotation needed - // The compiler can't infer `MaxLen` from this code. - let mut v3 = BoundedVec::new(); - v3.push(5); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L11-L27 - - -This defaulting of `MaxLen` (and numeric generics in general) to zero may change in future noir versions -but for now make sure to use type annotations when using bounded vectors. Otherwise, you will receive a constraint failure at runtime when the vec is pushed to. - -### get - -```rust -pub fn get(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this -will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - let last = v.get(v.len() - 1); - assert(first != last); -} -``` - -### get_unchecked - -```rust -pub fn get_unchecked(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero, without -performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, -it is unsafe! Use at your own risk! - -Example: - -```rust title="get_unchecked_example" showLineNumbers -fn sum_of_first_three(v: BoundedVec) -> u32 { - // Always ensure the length is larger than the largest - // index passed to get_unchecked - assert(v.len() > 2); - let first = v.get_unchecked(0); - let second = v.get_unchecked(1); - let third = v.get_unchecked(2); - first + second + third -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L54-L64 - - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - assert(first != 42); - v.set(0, 42); - let new_first = v.get(0); - assert(new_first == 42); -} -``` - -### set_unchecked - -```rust -pub fn set_unchecked(&mut self: Self, index: u64, value: T) -> T { -``` - -Writes an element to the vector at the given index, starting from zero, without performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, it is unsafe! Use at your own risk! - -Example: - -```rust title="set_unchecked_example" showLineNumbers -fn set_unchecked_example() { - let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([1, 2]); - - // Here we're safely writing within the valid range of `vec` - // `vec` now has the value [42, 2] - vec.set_unchecked(0, 42); - - // We can then safely read this value back out of `vec`. - // Notice that we use the checked version of `get` which would prevent reading unsafe values. - assert_eq(vec.get(0), 42); - - // We've now written past the end of `vec`. - // As this index is still within the maximum potential length of `v`, - // it won't cause a constraint failure. - vec.set_unchecked(2, 42); - println(vec); - - // This will write past the end of the maximum potential length of `vec`, - // it will then trigger a constraint failure. - vec.set_unchecked(5, 42); - println(vec); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L67-L91 - - - -### push - -```rust -pub fn push(&mut self, elem: T) { -``` - -Pushes an element to the end of the vector. This increases the length -of the vector by one. - -Panics if the new length of the vector will be greater than the max length. - -Example: - -```rust title="bounded-vec-push-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - v.push(1); - v.push(2); - - // Panics with failed assertion "push out of bounds" - v.push(3); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L95-L103 - - -### pop - -```rust -pub fn pop(&mut self) -> T -``` - -Pops the element at the end of the vector. This will decrease the length -of the vector by one. - -Panics if the vector is empty. - -Example: - -```rust title="bounded-vec-pop-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.push(1); - v.push(2); - - let two = v.pop(); - let one = v.pop(); - - assert(two == 2); - assert(one == 1); - // error: cannot pop from an empty vector - // let _ = v.pop(); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L108-L120 - - -### len - -```rust -pub fn len(self) -> u64 { -``` - -Returns the current length of this vector - -Example: - -```rust title="bounded-vec-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - assert(v.len() == 0); - - v.push(100); - assert(v.len() == 1); - - v.push(200); - v.push(300); - v.push(400); - assert(v.len() == 4); - - let _ = v.pop(); - let _ = v.pop(); - assert(v.len() == 2); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L125-L140 - - -### max_len - -```rust -pub fn max_len(_self: BoundedVec) -> u64 { -``` - -Returns the maximum length of this vector. This is always -equal to the `MaxLen` parameter this vector was initialized with. - -Example: - -```rust title="bounded-vec-max-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.max_len() == 5); - v.push(10); - assert(v.max_len() == 5); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L145-L151 - - -### storage - -```rust -pub fn storage(self) -> [T; MaxLen] { -``` - -Returns the internal array within this vector. -Since arrays in Noir are immutable, mutating the returned storage array will not mutate -the storage held internally by this vector. - -Note that uninitialized elements may be zeroed out! - -Example: - -```rust title="bounded-vec-storage-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.storage() == [0, 0, 0, 0, 0]); - - v.push(57); - assert(v.storage() == [57, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L156-L163 - - -### extend_from_array - -```rust -pub fn extend_from_array(&mut self, array: [T; Len]) -``` - -Pushes each element from the given array to this vector. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-array-example" showLineNumbers -let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([2, 4]); - - assert(vec.len == 2); - assert(vec.get(0) == 2); - assert(vec.get(1) == 4); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L168-L175 - - -### extend_from_bounded_vec - -```rust -pub fn extend_from_bounded_vec(&mut self, vec: BoundedVec) -``` - -Pushes each element from the other vector to this vector. The length of -the other vector is left unchanged. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-bounded-vec-example" showLineNumbers -let mut v1: BoundedVec = BoundedVec::new(); - let mut v2: BoundedVec = BoundedVec::new(); - - v2.extend_from_array([1, 2, 3]); - v1.extend_from_bounded_vec(v2); - - assert(v1.storage() == [1, 2, 3, 0, 0]); - assert(v2.storage() == [1, 2, 3, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L180-L189 - - -### from_array - -```rust -pub fn from_array(array: [T; Len]) -> Self -``` - -Creates a new vector, populating it with values derived from an array input. -The maximum length of the vector is determined based on the type signature. - -Example: -```rust -let bounded_vec: BoundedVec = BoundedVec::from_array([1, 2, 3]) -``` - -### map - -```rust -pub fn map(self, f: fn[Env](T) -> U) -> BoundedVec -``` - -Creates a new vector of equal size by calling a closure on each element in this vector. - -Example: - -```rust title="bounded-vec-map-example" showLineNumbers -let vec: BoundedVec = BoundedVec::from_array([1, 2, 3, 4]); - let result = vec.map(|value| value * 2); -``` -> Source code: noir_stdlib/src/collections/bounded_vec.nr#L493-L496 - - -### any - -```rust -pub fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -Returns true if the given predicate returns true for any element -in this vector. - -Example: - -```rust title="bounded-vec-any-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.extend_from_array([2, 4, 6]); - - let all_even = !v.any(|elem: u32| elem % 2 != 0); - assert(all_even); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L256-L262 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/hashmap.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/hashmap.md deleted file mode 100644 index 39f1ae9b32c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/hashmap.md +++ /dev/null @@ -1,594 +0,0 @@ ---- -title: HashMap -keywords: [noir, map, hash, hashmap] -sidebar_position: 1 ---- - -`HashMap` is used to efficiently store and look up key-value pairs. - -`HashMap` is a bounded type which can store anywhere from zero to `MaxLen` total elements. -Note that due to hash collisions, the actual maximum number of elements stored by any particular -hashmap is likely lower than `MaxLen`. This is true even with cryptographic hash functions since -every hash value will be performed modulo `MaxLen`. - -Example: - -```rust -// Create a mapping from Fields to u32s with a maximum length of 12 -// using a poseidon2 hasher -use std::hash::poseidon2::Poseidon2Hasher; -let mut map: HashMap> = HashMap::default(); - -map.insert(1, 2); -map.insert(3, 4); - -let two = map.get(1).unwrap(); -``` - -## Methods - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default { - /// Constructs an empty HashMap. - /// - /// Example: - /// - /// ```noir - /// let hashmap: HashMap> = HashMap::default(); - /// assert(hashmap.is_empty()); - /// ``` - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L694-L708 - - -Creates a fresh, empty HashMap. - -When using this function, always make sure to specify the maximum size of the hash map. - -This is the same `default` from the `Default` implementation given further below. It is -repeated here for convenience since it is the recommended way to create a hashmap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L204 - - -Because `HashMap` has so many generic arguments that are likely to be the same throughout -your program, it may be helpful to create a type alias: - -```rust title="type_alias" showLineNumbers -type MyMap = HashMap>; -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L195-L197 - - -### with_hasher - -```rust title="with_hasher" showLineNumbers -pub fn with_hasher(_build_hasher: B) -> Self - where - B: BuildHasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L103-L107 - - -Creates a hashmap with an existing `BuildHasher`. This can be used to ensure multiple -hashmaps are created with the same hasher instance. - -Example: - -```rust title="with_hasher_example" showLineNumbers -let my_hasher: BuildHasherDefault = Default::default(); - let hashmap: HashMap> = HashMap::with_hasher(my_hasher); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L206-L210 - - -### get - -```rust title="get" showLineNumbers -pub fn get( - self, - key: K - ) -> Option - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L470-L479 - - -Retrieves a value from the hashmap, returning `Option::none()` if it was not found. - -Example: - -```rust title="get_example" showLineNumbers -fn get_example(map: HashMap>) { - let x = map.get(12); - - if x.is_some() { - assert(x.unwrap() == 42); - } -} -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L298-L306 - - -### insert - -```rust title="insert" showLineNumbers -pub fn insert( - &mut self, - key: K, - value: V - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L514-L524 - - -Inserts a new key-value pair into the map. If the key was already in the map, its -previous value will be overridden with the newly provided one. - -Example: - -```rust title="insert_example" showLineNumbers -let mut map: HashMap> = HashMap::default(); - map.insert(12, 42); - assert(map.len() == 1); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L212-L216 - - -### remove - -```rust title="remove" showLineNumbers -pub fn remove( - &mut self, - key: K - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L573-L582 - - -Removes the given key-value pair from the map. If the key was not already present -in the map, this does nothing. - -Example: - -```rust title="remove_example" showLineNumbers -map.remove(12); - assert(map.is_empty()); - - // If a key was not present in the map, remove does nothing - map.remove(12); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L220-L227 - - -### is_empty - -```rust title="is_empty" showLineNumbers -pub fn is_empty(self) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L168-L170 - - -True if the length of the hash map is empty. - -Example: - -```rust title="is_empty_example" showLineNumbers -assert(map.is_empty()); - - map.insert(1, 2); - assert(!map.is_empty()); - - map.remove(1); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L229-L237 - - -### len - -```rust title="len" showLineNumbers -pub fn len(self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L429-L431 - - -Returns the current length of this hash map. - -Example: - -```rust title="len_example" showLineNumbers -// This is equivalent to checking map.is_empty() - assert(map.len() == 0); - - map.insert(1, 2); - map.insert(3, 4); - map.insert(5, 6); - assert(map.len() == 3); - - // 3 was already present as a key in the hash map, so the length is unchanged - map.insert(3, 7); - assert(map.len() == 3); - - map.remove(1); - assert(map.len() == 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L239-L254 - - -### capacity - -```rust title="capacity" showLineNumbers -pub fn capacity(_self: Self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L451-L453 - - -Returns the maximum capacity of this hashmap. This is always equal to the capacity -specified in the hashmap's type. - -Unlike hashmaps in general purpose programming languages, hashmaps in Noir have a -static capacity that does not increase as the map grows larger. Thus, this capacity -is also the maximum possible element count that can be inserted into the hashmap. -Due to hash collisions (modulo the hashmap length), it is likely the actual maximum -element count will be lower than the full capacity. - -Example: - -```rust title="capacity_example" showLineNumbers -let empty_map: HashMap> = HashMap::default(); - assert(empty_map.len() == 0); - assert(empty_map.capacity() == 42); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L256-L260 - - -### clear - -```rust title="clear" showLineNumbers -pub fn clear(&mut self) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L122-L124 - - -Clears the hashmap, removing all key-value pairs from it. - -Example: - -```rust title="clear_example" showLineNumbers -assert(!map.is_empty()); - map.clear(); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L262-L266 - - -### contains_key - -```rust title="contains_key" showLineNumbers -pub fn contains_key( - self, - key: K - ) -> bool - where - K: Hash + Eq, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L142-L151 - - -True if the hashmap contains the given key. Unlike `get`, this will not also return -the value associated with the key. - -Example: - -```rust title="contains_key_example" showLineNumbers -if map.contains_key(7) { - let value = map.get(7); - assert(value.is_some()); - } else { - println("No value for key 7!"); - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L268-L275 - - -### entries - -```rust title="entries" showLineNumbers -pub fn entries(self) -> BoundedVec<(K, V), N> { -``` -> Source code: noir_stdlib/src/collections/map.nr#L192-L194 - - -Returns a vector of each key-value pair present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="entries_example" showLineNumbers -let entries = map.entries(); - - // The length of a hashmap may not be compile-time known, so we - // need to loop over its capacity instead - for i in 0..map.capacity() { - if i < entries.len() { - let (key, value) = entries.get(i); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L309-L320 - - -### keys - -```rust title="keys" showLineNumbers -pub fn keys(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L228-L230 - - -Returns a vector of each key present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="keys_example" showLineNumbers -let keys = map.keys(); - - for i in 0..keys.max_len() { - if i < keys.len() { - let key = keys.get_unchecked(i); - let value = map.get(key).unwrap_unchecked(); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L322-L332 - - -### values - -```rust title="values" showLineNumbers -pub fn values(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L262-L264 - - -Returns a vector of each value present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="values_example" showLineNumbers -let values = map.values(); - - for i in 0..values.max_len() { - if i < values.len() { - let value = values.get_unchecked(i); - println(f"Found value {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L334-L343 - - -### iter_mut - -```rust title="iter_mut" showLineNumbers -pub fn iter_mut( - &mut self, - f: fn(K, V) -> (K, V) - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L298-L307 - - -Iterates through each key-value pair of the HashMap, setting each key-value pair to the -result returned from the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If this is not desired, use `iter_values_mut` if only values need to be mutated, -or `entries` if neither keys nor values need to be mutated. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_mut_example" showLineNumbers -// Add 1 to each key in the map, and double the value associated with that key. - map.iter_mut(|k, v| (k + 1, v * 2)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L347-L350 - - -### iter_keys_mut - -```rust title="iter_keys_mut" showLineNumbers -pub fn iter_keys_mut( - &mut self, - f: fn(K) -> K - ) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher { -``` -> Source code: noir_stdlib/src/collections/map.nr#L338-L347 - - -Iterates through the HashMap, mutating each key to the result returned from -the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If only iteration is desired and the keys are not intended to be mutated, -prefer using `entries` instead. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_keys_mut_example" showLineNumbers -// Double each key, leaving the value associated with that key untouched - map.iter_keys_mut(|k| k * 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L352-L355 - - -### iter_values_mut - -```rust title="iter_values_mut" showLineNumbers -pub fn iter_values_mut(&mut self, f: fn(V) -> V) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L372-L374 - - -Iterates through the HashMap, applying the given function to each value and mutating the -value to equal the result. This function is more efficient than `iter_mut` and `iter_keys_mut` -because the keys are untouched and the underlying hashmap thus does not need to be reordered. - -Example: - -```rust title="iter_values_mut_example" showLineNumbers -// Halve each value - map.iter_values_mut(|v| v / 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L357-L360 - - -### retain - -```rust title="retain" showLineNumbers -pub fn retain(&mut self, f: fn(K, V) -> bool) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L393-L395 - - -Retains only the key-value pairs for which the given function returns true. -Any key-value pairs for which the function returns false will be removed from the map. - -Example: - -```rust title="retain_example" showLineNumbers -map.retain(|k, v| (k != 0) & (v != 0)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L280-L282 - - -## Trait Implementations - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default { - /// Constructs an empty HashMap. - /// - /// Example: - /// - /// ```noir - /// let hashmap: HashMap> = HashMap::default(); - /// assert(hashmap.is_empty()); - /// ``` - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L694-L708 - - -Constructs an empty HashMap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L204 - - -### eq - -```rust title="eq" showLineNumbers -impl Eq for HashMap -where - K: Eq + Hash, - V: Eq, - B: BuildHasher, - H: Hasher { - /// Checks if two HashMaps are equal. - /// - /// Example: - /// - /// ```noir - /// let mut map1: HashMap> = HashMap::default(); - /// let mut map2: HashMap> = HashMap::default(); - /// - /// map1.insert(1, 2); - /// map1.insert(3, 4); - /// - /// map2.insert(3, 4); - /// map2.insert(1, 2); - /// - /// assert(map1 == map2); - /// ``` - fn eq(self, other: HashMap) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L643-L667 - - -Checks if two HashMaps are equal. - -Example: - -```rust title="eq_example" showLineNumbers -let mut map1: HashMap> = HashMap::default(); - let mut map2: HashMap> = HashMap::default(); - - map1.insert(1, 2); - map1.insert(3, 4); - - map2.insert(3, 4); - map2.insert(1, 2); - - assert(map1 == map2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L284-L295 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/index.md deleted file mode 100644 index ea84c6d5c21..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/index.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -title: Containers -description: Container types provided by Noir's standard library for storing and retrieving data -keywords: [containers, data types, vec, hashmap] ---- diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/vec.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/vec.mdx deleted file mode 100644 index 475011922f8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/containers/vec.mdx +++ /dev/null @@ -1,170 +0,0 @@ ---- -title: Vectors -description: Delve into the Vec data type in Noir. Learn about its methods, practical examples, and best practices for using Vectors in your Noir code. -keywords: [noir, vector type, methods, examples, dynamic arrays] -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A vector is a collection type similar to Rust's `Vec` type. In Noir, it is a convenient way to use slices as mutable arrays. - -Example: - -```rust -let mut vector: Vec = Vec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -``` - -## Methods - -### new - -Creates a new, empty vector. - -```rust -pub fn new() -> Self -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### from_slice - -Creates a vector containing each element from a given slice. Mutations to the resulting vector will not affect the original slice. - -```rust -pub fn from_slice(slice: [T]) -> Self -``` - -Example: - -```rust -let slice: [Field] = &[1, 2, 3]; -let vector_from_slice = Vec::from_slice(slice); -assert(vector_from_slice.len() == 3); -``` - -### len - -Returns the number of elements in the vector. - -```rust -pub fn len(self) -> Field -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### get - -Retrieves an element from the vector at a given index. Panics if the index points beyond the vector's end. - -```rust -pub fn get(self, index: Field) -> T -``` - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -``` - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -Panics if the index points beyond the vector's end. - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -vector.set(1, 42); -assert(vector.get(1) == 42); -``` - -### push - -Adds a new element to the vector's end, returning a new vector with a length one greater than the original unmodified vector. - -```rust -pub fn push(&mut self, elem: T) -``` - -Example: - -```rust -let mut vector: Vec = Vec::new(); -vector.push(10); -assert(vector.len() == 1); -``` - -### pop - -Removes an element from the vector's end, returning a new vector with a length one less than the original vector, along with the removed element. Panics if the vector's length is zero. - -```rust -pub fn pop(&mut self) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20]); -let popped_elem = vector.pop(); -assert(popped_elem == 20); -assert(vector.len() == 1); -``` - -### insert - -Inserts an element at a specified index, shifting subsequent elements to the right. - -```rust -pub fn insert(&mut self, index: Field, elem: T) -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 30]); -vector.insert(1, 20); -assert(vector.get(1) == 20); -``` - -### remove - -Removes an element at a specified index, shifting subsequent elements to the left, and returns the removed element. - -```rust -pub fn remove(&mut self, index: Field) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20, 30]); -let removed_elem = vector.remove(1); -assert(removed_elem == 20); -assert(vector.len() == 2); -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ciphers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ciphers.mdx deleted file mode 100644 index d6a5e1a79eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ciphers.mdx +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Ciphers -description: - Learn about the implemented ciphers ready to use for any Noir project -keywords: - [ciphers, Noir project, aes128, encrypt] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## aes128 - -Given a plaintext as an array of bytes, returns the corresponding aes128 ciphertext (CBC mode). Input padding is automatically performed using PKCS#7, so that the output length is `input.len() + (16 - input.len() % 16)`. - -```rust title="aes128" showLineNumbers -pub fn aes128_encrypt(input: [u8; N], iv: [u8; 16], key: [u8; 16]) -> [u8] {} -``` -> Source code: noir_stdlib/src/aes128.nr#L2-L4 - - -```rust -fn main() { - let input: [u8; 4] = [0, 12, 3, 15] // Random bytes, will be padded to 16 bytes. - let iv: [u8; 16] = [0; 16]; // Initialisation vector - let key: [u8; 16] = [0; 16] // AES key - let ciphertext = std::aes128::aes128_encrypt(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()); // In this case, the output length will be 16 bytes. -} -``` - - - \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ec_primitives.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ec_primitives.md deleted file mode 100644 index f262d8160d6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ec_primitives.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Elliptic Curve Primitives -keywords: [cryptographic primitives, Noir project] -sidebar_position: 4 ---- - -Data structures and methods on them that allow you to carry out computations involving elliptic -curves over the (mathematical) field corresponding to `Field`. For the field currently at our -disposal, applications would involve a curve embedded in BN254, e.g. the -[Baby Jubjub curve](https://eips.ethereum.org/EIPS/eip-2494). - -## Data structures - -### Elliptic curve configurations - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Curve`), i.e. the specific elliptic -curve you want to use, which would be specified using any one of the methods -`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::new` which take the coefficients in the -defining equation together with a generator point as parameters. You can find more detail in the -comments in -[`noir_stdlib/src/ec/mod.nr`](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr), but -the gist of it is that the elliptic curves of interest are usually expressed in one of the standard -forms implemented here (Twisted Edwards, Montgomery and Short Weierstraß), and in addition to that, -you could choose to use `affine` coordinates (Cartesian coordinates - the usual (x,y) - possibly -together with a point at infinity) or `curvegroup` coordinates (some form of projective coordinates -requiring more coordinates but allowing for more efficient implementations of elliptic curve -operations). Conversions between all of these forms are provided, and under the hood these -conversions are done whenever an operation is more efficient in a different representation (or a -mixed coordinate representation is employed). - -### Points - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Point`), i.e. points lying on the -elliptic curve. For a curve configuration `c` and a point `p`, it may be checked that `p` -does indeed lie on `c` by calling `c.contains(p1)`. - -## Methods - -(given a choice of curve representation, e.g. use `std::ec::tecurve::affine::Curve` and use -`std::ec::tecurve::affine::Point`) - -- The **zero element** is given by `Point::zero()`, and we can verify whether a point `p: Point` is - zero by calling `p.is_zero()`. -- **Equality**: Points `p1: Point` and `p2: Point` may be checked for equality by calling - `p1.eq(p2)`. -- **Addition**: For `c: Curve` and points `p1: Point` and `p2: Point` on the curve, adding these two - points is accomplished by calling `c.add(p1,p2)`. -- **Negation**: For a point `p: Point`, `p.negate()` is its negation. -- **Subtraction**: For `c` and `p1`, `p2` as above, subtracting `p2` from `p1` is accomplished by - calling `c.subtract(p1,p2)`. -- **Scalar multiplication**: For `c` as above, `p: Point` a point on the curve and `n: Field`, - scalar multiplication is given by `c.mul(n,p)`. If instead `n :: [u1; N]`, i.e. `n` is a bit - array, the `bit_mul` method may be used instead: `c.bit_mul(n,p)` -- **Multi-scalar multiplication**: For `c` as above and arrays `n: [Field; N]` and `p: [Point; N]`, - multi-scalar multiplication is given by `c.msm(n,p)`. -- **Coordinate representation conversions**: The `into_group` method converts a point or curve - configuration in the affine representation to one in the CurveGroup representation, and - `into_affine` goes in the other direction. -- **Curve representation conversions**: `tecurve` and `montcurve` curves and points are equivalent - and may be converted between one another by calling `into_montcurve` or `into_tecurve` on their - configurations or points. `swcurve` is more general and a curve c of one of the other two types - may be converted to this representation by calling `c.into_swcurve()`, whereas a point `p` lying - on the curve given by `c` may be mapped to its corresponding `swcurve` point by calling - `c.map_into_swcurve(p)`. -- **Map-to-curve methods**: The Elligator 2 method of mapping a field element `n: Field` into a - `tecurve` or `montcurve` with configuration `c` may be called as `c.elligator2_map(n)`. For all of - the curve configurations, the SWU map-to-curve method may be called as `c.swu_map(z,n)`, where - `z: Field` depends on `Field` and `c` and must be chosen by the user (the conditions it needs to - satisfy are specified in the comments - [here](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr)). - -## Examples - -The -[ec_baby_jubjub test](https://github.com/noir-lang/noir/blob/master/test_programs/compile_success_empty/ec_baby_jubjub/src/main.nr) -illustrates all of the above primitives on various forms of the Baby Jubjub curve. A couple of more -interesting examples in Noir would be: - -Public-key cryptography: Given an elliptic curve and a 'base point' on it, determine the public key -from the private key. This is a matter of using scalar multiplication. In the case of Baby Jubjub, -for example, this code would do: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; - -fn bjj_pub_key(priv_key: Field) -> Point -{ - - let bjj = Curve::new(168700, 168696, G::new(995203441582195749578291179787384436505546430278305826713579947235728471134,5472060717959818805561601436314318772137091100104008585924551046643952123905)); - - let base_pt = Point::new(5299619240641551281634865583518297030282874472190772894086521144482721001553, 16950150798460657717958625567821834550301663161624707787222815936182638968203); - - bjj.mul(priv_key,base_pt) -} -``` - -This would come in handy in a Merkle proof. - -- EdDSA signature verification: This is a matter of combining these primitives with a suitable hash - function. See - [feat(stdlib): EdDSA sig verification noir#1136](https://github.com/noir-lang/noir/pull/1136) for - the case of Baby Jubjub and the Poseidon hash function. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx deleted file mode 100644 index 12bf5ae8aaf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: ECDSA Signature Verification -description: Learn about the cryptographic primitives regarding ECDSA over the secp256k1 and secp256r1 curves -keywords: [cryptographic primitives, Noir project, ecdsa, secp256k1, secp256r1, signatures] -sidebar_position: 3 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports ECDSA signatures verification over the secp256k1 and secp256r1 curves. - -## ecdsa_secp256k1::verify_signature - -Verifier for ECDSA Secp256k1 signatures. -See ecdsa_secp256k1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256k1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256k1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256k1::verify_signature_slice - -Verifier for ECDSA Secp256k1 signatures where the message is a slice. - -```rust title="ecdsa_secp256k1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L13-L20 - - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures. -See ecdsa_secp256r1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256r1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256r1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures where the message is a slice. - -```rust title="ecdsa_secp256r1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/eddsa.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/eddsa.mdx deleted file mode 100644 index b283de693c8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/eddsa.mdx +++ /dev/null @@ -1,37 +0,0 @@ ---- -title: EdDSA Verification -description: Learn about the cryptographic primitives regarding EdDSA -keywords: [cryptographic primitives, Noir project, eddsa, signatures] -sidebar_position: 5 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## eddsa::eddsa_poseidon_verify - -Verifier for EdDSA signatures - -```rust -fn eddsa_poseidon_verify(public_key_x : Field, public_key_y : Field, signature_s: Field, signature_r8_x: Field, signature_r8_y: Field, message: Field) -> bool -``` - -It is also possible to specify the hash algorithm used for the signature by using the `eddsa_verify` function by passing a type implementing the Hasher trait with the turbofish operator. -For instance, if you want to use Poseidon2 instead, you can do the following: -```rust -use std::hash::poseidon2::Poseidon2Hasher; - -eddsa_verify::(pub_key_a.x, pub_key_a.y, s_a, r8_a.x, r8_a.y, msg); -``` - - - -## eddsa::eddsa_to_pub - -Private to public key conversion. - -Returns `(pub_key_x, pub_key_y)` - -```rust -fn eddsa_to_pub(secret : Field) -> (Field, Field) -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx deleted file mode 100644 index 9507f16322c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx +++ /dev/null @@ -1,92 +0,0 @@ ---- -title: Scalar multiplication -description: See how you can perform scalar multiplication in Noir -keywords: [cryptographic primitives, Noir project, scalar multiplication] -sidebar_position: 1 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -The following functions perform operations over the embedded curve whose coordinates are defined by the configured noir field. -For the BN254 scalar field, this is BabyJubJub or Grumpkin. - -:::note -Suffixes `_low` and `_high` denote low and high limbs of a scalar. -::: - -## embedded_curve_ops::multi_scalar_mul - -Performs multi scalar multiplication over the embedded curve. -The function accepts arbitrary amount of point-scalar pairs on the input, it multiplies the individual pairs over -the curve and returns a sum of the resulting points. - -Points represented as x and y coordinates [x1, y1, x2, y2, ...], scalars as low and high limbs [low1, high1, low2, high2, ...]. - -```rust title="multi_scalar_mul" showLineNumbers -pub fn multi_scalar_mul( - points: [EmbeddedCurvePoint; N], - scalars: [EmbeddedCurveScalar; N] -) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L103-L108 - - -example - -```rust -fn main(point_x: Field, point_y: Field, scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::multi_scalar_mul([point_x, point_y], [scalar_low, scalar_high]); - println(point); -} -``` - -## embedded_curve_ops::fixed_base_scalar_mul - -Performs fixed base scalar multiplication over the embedded curve (multiplies input scalar with a generator point). -The function accepts a single scalar on the input represented as 2 fields. - -```rust title="fixed_base_scalar_mul" showLineNumbers -pub fn fixed_base_scalar_mul(scalar: EmbeddedCurveScalar) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L120-L122 - - -example - -```rust -fn main(scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::fixed_base_scalar_mul(scalar_low, scalar_high); - println(point); -} -``` - -## embedded_curve_ops::embedded_curve_add - -Adds two points on the embedded curve. -This function takes two `EmbeddedCurvePoint` structures as parameters, representing points on the curve, and returns a new `EmbeddedCurvePoint` structure that represents their sum. - -### Parameters: -- `point1` (`EmbeddedCurvePoint`): The first point to add. -- `point2` (`EmbeddedCurvePoint`): The second point to add. - -### Returns: -- `EmbeddedCurvePoint`: The resulting point after the addition of `point1` and `point2`. - -```rust title="embedded_curve_add" showLineNumbers -pub fn embedded_curve_add(point1: EmbeddedCurvePoint, point2: EmbeddedCurvePoint) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L132-L134 - - -example - -```rust -fn main() { - let point1 = EmbeddedCurvePoint { x: 1, y: 2 }; - let point2 = EmbeddedCurvePoint { x: 3, y: 4 }; - let result = std::embedded_curve_ops::embedded_curve_add(point1, point2); - println!("Resulting Point: ({}, {})", result.x, result.y); -} -``` - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/hashes.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/hashes.mdx deleted file mode 100644 index 33601bae22b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/hashes.mdx +++ /dev/null @@ -1,253 +0,0 @@ ---- -title: Hash methods -description: - Learn about the cryptographic primitives ready to use for any Noir project, including sha256, - blake2s, pedersen, mimc_bn254 and mimc -keywords: - [cryptographic primitives, Noir project, sha256, blake2s, pedersen, mimc_bn254, mimc, hash] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## sha256 - -Given an array of bytes, returns the resulting sha256 hash. -Specify a message_size to hash only the first `message_size` bytes of the input. - -```rust title="sha256" showLineNumbers -pub fn sha256(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/sha256.nr#L7-L9 - - -example: -```rust title="sha256_var" showLineNumbers -let digest = std::hash::sha256_var([x as u8], 1); -``` -> Source code: test_programs/execution_success/sha256/src/main.nr#L16-L18 - - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::sha256::sha256_var(x, 4); -} -``` - - - - -## blake2s - -Given an array of bytes, returns an array with the Blake2 hash - -```rust title="blake2s" showLineNumbers -pub fn blake2s(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L18-L20 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake2s(x); -} -``` - - - -## blake3 - -Given an array of bytes, returns an array with the Blake3 hash - -```rust title="blake3" showLineNumbers -pub fn blake3(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L24-L26 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake3(x); -} -``` - - - -## pedersen_hash - -Given an array of Fields, returns the Pedersen hash. - -```rust title="pedersen_hash" showLineNumbers -pub fn pedersen_hash(input: [Field; N]) -> Field -``` -> Source code: noir_stdlib/src/hash/mod.nr#L77-L79 - - -example: - -```rust title="pedersen-hash" showLineNumbers -fn main(x: Field, y: Field, expected_hash: Field) { - let hash = std::hash::pedersen_hash([x, y]); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/pedersen_hash/src/main.nr#L1-L7 - - - - -## pedersen_commitment - -Given an array of Fields, returns the Pedersen commitment. - -```rust title="pedersen_commitment" showLineNumbers -pub fn pedersen_commitment(input: [Field; N]) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/hash/mod.nr#L29-L31 - - -example: - -```rust title="pedersen-commitment" showLineNumbers -fn main(x: Field, y: Field, expected_commitment: std::embedded_curve_ops::EmbeddedCurvePoint) { - let commitment = std::hash::pedersen_commitment([x, y]); - assert_eq(commitment.x, expected_commitment.x); - assert_eq(commitment.y, expected_commitment.y); -} -``` -> Source code: test_programs/execution_success/pedersen_commitment/src/main.nr#L1-L8 - - - - -## keccak256 - -Given an array of bytes (`u8`), returns the resulting keccak hash as an array of -32 bytes (`[u8; 32]`). Specify a message_size to hash only the first -`message_size` bytes of the input. - -```rust title="keccak256" showLineNumbers -pub fn keccak256(input: [u8; N], message_size: u32) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L125-L127 - - -example: - -```rust title="keccak256" showLineNumbers -fn main(x: Field, result: [u8; 32]) { - // We use the `as` keyword here to denote the fact that we want to take just the first byte from the x Field - // The padding is taken care of by the program - let digest = std::hash::keccak256([x as u8], 1); - assert(digest == result); - - //#1399: variable message size - let message_size = 4; - let hash_a = std::hash::keccak256([1, 2, 3, 4], message_size); - let hash_b = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size); - - assert(hash_a == hash_b); - - let message_size_big = 8; - let hash_c = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size_big); - - assert(hash_a != hash_c); -} -``` -> Source code: test_programs/execution_success/keccak256/src/main.nr#L1-L21 - - - - -## poseidon - -Given an array of Fields, returns a new Field with the Poseidon Hash. Mind that you need to specify -how many inputs are there to your Poseidon function. - -```rust -// example for hash_1, hash_2 accepts an array of length 2, etc -fn hash_1(input: [Field; 1]) -> Field -``` - -example: - -```rust title="poseidon" showLineNumbers -use std::hash::poseidon; - -fn main(x1: [Field; 2], y1: pub Field, x2: [Field; 4], y2: pub Field) { - let hash1 = poseidon::bn254::hash_2(x1); - assert(hash1 == y1); - - let hash2 = poseidon::bn254::hash_4(x2); - assert(hash2 == y2); -} -``` -> Source code: test_programs/execution_success/poseidon_bn254_hash/src/main.nr#L1-L11 - - -## poseidon 2 - -Given an array of Fields, returns a new Field with the Poseidon2 Hash. Contrary to the Poseidon -function, there is only one hash and you can specify a message_size to hash only the first -`message_size` bytes of the input, - -```rust -// example for hashing the first three elements of the input -Poseidon2::hash(input, 3); -``` - -example: - -```rust title="poseidon2" showLineNumbers -use std::hash::poseidon2; - -fn main(inputs: [Field; 4], expected_hash: Field) { - let hash = poseidon2::Poseidon2::hash(inputs, inputs.len()); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/poseidon2/src/main.nr#L1-L8 - - -## mimc_bn254 and mimc - -`mimc_bn254` is `mimc`, but with hardcoded parameters for the BN254 curve. You can use it by -providing an array of Fields, and it returns a Field with the hash. You can use the `mimc` method if -you're willing to input your own constants: - -```rust -fn mimc(x: Field, k: Field, constants: [Field; N], exp : Field) -> Field -``` - -otherwise, use the `mimc_bn254` method: - -```rust -fn mimc_bn254(array: [Field; N]) -> Field -``` - -example: - -```rust - -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::mimc::mimc_bn254(x); -} -``` - -## hash_to_field - -```rust -fn hash_to_field(_input : [Field]) -> Field {} -``` - -Calculates the `blake2s` hash of the inputs and returns the hash modulo the field modulus to return -a value which can be represented as a `Field`. - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/index.md deleted file mode 100644 index 650f30165d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/index.md +++ /dev/null @@ -1,14 +0,0 @@ ---- -title: Cryptographic Primitives -description: - Learn about the cryptographic primitives ready to use for any Noir project -keywords: - [ - cryptographic primitives, - Noir project, - ] ---- - -The Noir team is progressively adding new cryptographic primitives to the standard library. Reach out for news or if you would be interested in adding more of these calculations in Noir. - -Some methods are available thanks to the Aztec backend, not being performed using Noir. When using other backends, these methods may or may not be supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/schnorr.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/schnorr.mdx deleted file mode 100644 index 0058c0c0c98..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/cryptographic_primitives/schnorr.mdx +++ /dev/null @@ -1,64 +0,0 @@ ---- -title: Schnorr Signatures -description: Learn how you can verify Schnorr signatures using Noir -keywords: [cryptographic primitives, Noir project, schnorr, signatures] -sidebar_position: 2 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## schnorr::verify_signature - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin). -See schnorr::verify_signature_slice for a version that works directly on slices. - -```rust title="schnorr_verify" showLineNumbers -pub fn verify_signature( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8; N] -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L4-L11 - - -where `_signature` can be generated like so using the npm package -[@noir-lang/barretenberg](https://www.npmjs.com/package/@noir-lang/barretenberg) - -```js -const { BarretenbergWasm } = require('@noir-lang/barretenberg/dest/wasm'); -const { Schnorr } = require('@noir-lang/barretenberg/dest/crypto/schnorr'); - -... - -const barretenberg = await BarretenbergWasm.new(); -const schnorr = new Schnorr(barretenberg); -const pubKey = schnorr.computePublicKey(privateKey); -const message = ... -const signature = Array.from( - schnorr.constructSignature(hash, privateKey).toBuffer() -); - -... -``` - - - -## schnorr::verify_signature_slice - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin) -where the message is a slice. - -```rust title="schnorr_verify_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8] -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L15-L22 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/fmtstr.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/fmtstr.md deleted file mode 100644 index 65a7da9996d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/fmtstr.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: fmtstr ---- - -`fmtstr` is the type resulting from using format string (`f"..."`). - -## Methods - -### quoted_contents - -```rust title="quoted_contents" showLineNumbers -comptime fn quoted_contents(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/format_string.nr#L3-L5 - - -Returns the format string contents (that is, without the leading and trailing double quotes) as a `Quoted` value. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/is_unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/is_unconstrained.md deleted file mode 100644 index 51bb1bda8f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/is_unconstrained.md +++ /dev/null @@ -1,69 +0,0 @@ ---- -title: Is Unconstrained Function -description: - The is_unconstrained function returns wether the context at that point of the program is unconstrained or not. -keywords: - [ - unconstrained - ] ---- - -It's very common for functions in circuits to take unconstrained hints of an expensive computation and then verify it. This is done by running the hint in an unconstrained context and then verifying the result in a constrained context. - -When a function is marked as unconstrained, any subsequent functions that it calls will also be run in an unconstrained context. However, if we are implementing a library function, other users might call it within an unconstrained context or a constrained one. Generally, in an unconstrained context we prefer just computing the result instead of taking a hint of it and verifying it, since that'd mean doing the same computation twice: - -```rust - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - my_expensive_computation_hint(); - // verify my_expensive_computation: If external_interface is called from unconstrained, this is redundant - ... -} - -``` - -In order to improve the performance in an unconstrained context you can use the function at `std::runtime::is_unconstrained() -> bool`: - - -```rust -use dep::std::runtime::is_unconstrained; - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - if is_unconstrained() { - my_expensive_computation(); - } else { - my_expensive_computation_hint(); - // verify my_expensive_computation - ... - } -} - -``` - -The is_unconstrained result is resolved at compile time, so in unconstrained contexts the compiler removes the else branch, and in constrained contexts the compiler removes the if branch, reducing the amount of compute necessary to run external_interface. - -Note that using `is_unconstrained` in a `comptime` context will also return `true`: - -``` -fn main() { - comptime { - assert(is_unconstrained()); - } -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/logging.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/logging.md deleted file mode 100644 index db75ef9f86f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/logging.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Logging -description: - Learn how to use the println statement for debugging in Noir with this tutorial. Understand the - basics of logging in Noir and how to implement it in your code. -keywords: - [ - noir logging, - println statement, - print statement, - debugging in noir, - noir std library, - logging tutorial, - basic logging in noir, - noir logging implementation, - noir debugging techniques, - rust, - ] ---- - -The standard library provides two familiar statements you can use: `println` and `print`. Despite being a limited implementation of rust's `println!` and `print!` macros, these constructs can be useful for debugging. - -You can print the output of both statements in your Noir code by using the `nargo execute` command or the `--show-output` flag when using `nargo test` (provided there are print statements in your tests). - -It is recommended to use `nargo execute` if you want to debug failing constraints with `println` or `print` statements. This is due to every input in a test being a constant rather than a witness, so we issue an error during compilation while we only print during execution (which comes after compilation). Neither `println`, nor `print` are callable for failed constraints caught at compile time. - -Both `print` and `println` are generic functions which can work on integers, fields, strings, and even structs or expressions. Note however, that slices are currently unsupported. For example: - -```rust -struct Person { - age: Field, - height: Field, -} - -fn main(age: Field, height: Field) { - let person = Person { - age: age, - height: height, - }; - println(person); - println(age + height); - println("Hello world!"); -} -``` - -You can print different types in the same statement (including strings) with a type called `fmtstr`. It can be specified in the same way as a normal string, just prepended with an "f" character: - -```rust - let fmt_str = f"i: {i}, j: {j}"; - println(fmt_str); - - let s = myStruct { y: x, x: y }; - println(s); - - println(f"i: {i}, s: {s}"); - - println(x); - println([x, y]); - - let foo = fooStruct { my_struct: s, foo: 15 }; - println(f"s: {s}, foo: {foo}"); - - println(15); // prints 0x0f, implicit Field - println(-1 as u8); // prints 255 - println(-1 as i8); // prints -1 -``` - -Examples shown above are interchangeable between the two `print` statements: - -```rust -let person = Person { age : age, height : height }; - -println(person); -print(person); - -println("Hello world!"); // Prints with a newline at the end of the input -print("Hello world!"); // Prints the input and keeps cursor on the same line -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/merkle_trees.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/merkle_trees.md deleted file mode 100644 index 6a9ebf72ada..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/merkle_trees.md +++ /dev/null @@ -1,58 +0,0 @@ ---- -title: Merkle Trees -description: Learn about Merkle Trees in Noir with this tutorial. Explore the basics of computing a merkle root using a proof, with examples. -keywords: - [ - Merkle trees in Noir, - Noir programming language, - check membership, - computing root from leaf, - Noir Merkle tree implementation, - Merkle tree tutorial, - Merkle tree code examples, - Noir libraries, - pedersen hash., - ] ---- - -## compute_merkle_root - -Returns the root of the tree from the provided leaf and its hash path, using a [Pedersen hash](./cryptographic_primitives/hashes.mdx#pedersen_hash). - -```rust -fn compute_merkle_root(leaf : Field, index : Field, hash_path: [Field]) -> Field -``` - -example: - -```rust -/** - // these values are for this example only - index = "0" - priv_key = "0x000000000000000000000000000000000000000000000000000000616c696365" - secret = "0x1929ea3ab8d9106a899386883d9428f8256cfedb3c4f6b66bf4aa4d28a79988f" - note_hash_path = [ - "0x1e61bdae0f027b1b2159e1f9d3f8d00fa668a952dddd822fda80dc745d6f65cc", - "0x0e4223f3925f98934393c74975142bd73079ab0621f4ee133cee050a3c194f1a", - "0x2fd7bb412155bf8693a3bd2a3e7581a679c95c68a052f835dddca85fa1569a40" - ] - */ -fn main(index: Field, priv_key: Field, secret: Field, note_hash_path: [Field; 3]) { - - let pubkey = std::scalar_mul::fixed_base_embedded_curve(priv_key); - let pubkey_x = pubkey[0]; - let pubkey_y = pubkey[1]; - let note_commitment = std::hash::pedersen(&[pubkey_x, pubkey_y, secret]); - - let root = std::merkle::compute_merkle_root(note_commitment[0], index, note_hash_path.as_slice()); - println(root); -} -``` - -To check merkle tree membership: - -1. Include a merkle root as a program input. -2. Compute the merkle root of a given leaf, index and hash path. -3. Assert the merkle roots are equal. - -For more info about merkle trees, see the Wikipedia [page](https://en.wikipedia.org/wiki/Merkle_tree). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/ctstring.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/ctstring.md deleted file mode 100644 index 0be9bbced4e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/ctstring.md +++ /dev/null @@ -1,100 +0,0 @@ ---- -title: CtString ---- - -`std::meta::ctstring` contains methods on the built-in `CtString` type which is -a compile-time, dynamically-sized string type. Compared to `str` and `fmtstr`, -`CtString` is useful because its size does not need to be specified in its type. This -can be used for formatting items at compile-time or general string handling in `comptime` -code. - -Since `fmtstr`s can be converted into `CtString`s, you can make use of their formatting -abilities in CtStrings by formatting in `fmtstr`s then converting the result to a CtString -afterward. - -## Traits - -### AsCtString - -```rust title="as-ctstring" showLineNumbers -pub trait AsCtString { - comptime fn as_ctstring(self) -> CtString; -} -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L43-L47 - - -Converts an object into a compile-time string. - -Implementations: - -```rust -impl AsCtString for str { ... } -impl AsCtString for fmtstr { ... } -``` - -## Methods - -### new - -```rust title="new" showLineNumbers -comptime fn new() -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L4-L6 - - -Creates an empty `CtString`. - -### append_str - -```rust title="append_str" showLineNumbers -comptime fn append_str(self, s: str) -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L11-L13 - - -Returns a new CtString with the given str appended onto the end. - -### append_fmtstr - -```rust title="append_fmtstr" showLineNumbers -comptime fn append_fmtstr(self, s: fmtstr) -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L17-L19 - - -Returns a new CtString with the given fmtstr appended onto the end. - -### as_quoted_str - -```rust title="as_quoted_str" showLineNumbers -comptime fn as_quoted_str(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L26-L28 - - -Returns a quoted string literal from this string's contents. - -There is no direct conversion from a `CtString` to a `str` since -the size would not be known. To get around this, this function can -be used in combination with macro insertion (`!`) to insert this string -literal at this function's call site. - -Example: - -```rust title="as_quoted_str_example" showLineNumbers -let my_ctstring = "foo bar".as_ctstring(); - let my_str = my_ctstring.as_quoted_str!(); - - assert_eq(crate::meta::type_of(my_str), quote { str<7> }.as_type()); -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L90-L95 - - -## Trait Implementations - -```rust -impl Eq for CtString -impl Hash for CtString -impl Append for CtString -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/expr.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/expr.md deleted file mode 100644 index 1da32b0bc09..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/expr.md +++ /dev/null @@ -1,378 +0,0 @@ ---- -title: Expr ---- - -`std::meta::expr` contains methods on the built-in `Expr` type for quoted, syntactically valid expressions. - -## Methods - -### as_array - -```rust title="as_array" showLineNumbers -comptime fn as_array(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L10-L12 - - -If this expression is an array, this returns a slice of each element in the array. - -### as_assert - -```rust title="as_assert" showLineNumbers -comptime fn as_assert(self) -> Option<(Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L16-L18 - - -If this expression is an assert, this returns the assert expression and the optional message. - -### as_assert_eq - -```rust title="as_assert_eq" showLineNumbers -comptime fn as_assert_eq(self) -> Option<(Expr, Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L23-L25 - - -If this expression is an assert_eq, this returns the left-hand-side and right-hand-side -expressions, together with the optional message. - -### as_assign - -```rust title="as_assign" showLineNumbers -comptime fn as_assign(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L30-L32 - - -If this expression is an assignment, this returns a tuple with the left hand side -and right hand side in order. - -### as_binary_op - -```rust title="as_binary_op" showLineNumbers -comptime fn as_binary_op(self) -> Option<(Expr, BinaryOp, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L37-L39 - - -If this expression is a binary operator operation ` `, -return the left-hand side, operator, and the right-hand side of the operation. - -### as_block - -```rust title="as_block" showLineNumbers -comptime fn as_block(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L44-L46 - - -If this expression is a block `{ stmt1; stmt2; ...; stmtN }`, return -a slice containing each statement. - -### as_bool - -```rust title="as_bool" showLineNumbers -comptime fn as_bool(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L50-L52 - - -If this expression is a boolean literal, return that literal. - -### as_cast - -```rust title="as_cast" showLineNumbers -#[builtin(expr_as_cast)] - comptime fn as_cast(self) -> Option<(Expr, UnresolvedType)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L56-L59 - - -If this expression is a cast expression (`expr as type`), returns the casted -expression and the type to cast to. - -### as_comptime - -```rust title="as_comptime" showLineNumbers -comptime fn as_comptime(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L64-L66 - - -If this expression is a `comptime { stmt1; stmt2; ...; stmtN }` block, -return each statement in the block. - -### as_constructor - -```rust title="as_constructor" showLineNumbers -comptime fn as_constructor(self) -> Option<(UnresolvedType, [(Quoted, Expr)])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L71-L73 - - -If this expression is a constructor `Type { field1: expr1, ..., fieldN: exprN }`, -return the type and the fields. - -### as_for - -```rust title="as_for" showLineNumbers -comptime fn as_for(self) -> Option<(Quoted, Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L78-L80 - - -If this expression is a for statement over a single expression, return the identifier, -the expression and the for loop body. - -### as_for_range - -```rust title="as_for" showLineNumbers -comptime fn as_for(self) -> Option<(Quoted, Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L78-L80 - - -If this expression is a for statement over a range, return the identifier, -the range start, the range end and the for loop body. - -### as_function_call - -```rust title="as_function_call" showLineNumbers -comptime fn as_function_call(self) -> Option<(Expr, [Expr])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L92-L94 - - -If this expression is a function call `foo(arg1, ..., argN)`, return -the function and a slice of each argument. - -### as_if - -```rust title="as_if" showLineNumbers -comptime fn as_if(self) -> Option<(Expr, Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L100-L102 - - -If this expression is an `if condition { then_branch } else { else_branch }`, -return the condition, then branch, and else branch. If there is no else branch, -`None` is returned for that branch instead. - -### as_index - -```rust title="as_index" showLineNumbers -comptime fn as_index(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L107-L109 - - -If this expression is an index into an array `array[index]`, return the -array and the index. - -### as_integer - -```rust title="as_integer" showLineNumbers -comptime fn as_integer(self) -> Option<(Field, bool)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L114-L116 - - -If this expression is an integer literal, return the integer as a field -as well as whether the integer is negative (true) or not (false). - -### as_lambda - -```rust title="as_lambda" showLineNumbers -comptime fn as_lambda(self) -> Option<([(Expr, Option)], Option, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L120-L122 - - -If this expression is a lambda, returns the parameters, return type and body. - -### as_let - -```rust title="as_let" showLineNumbers -comptime fn as_let(self) -> Option<(Expr, Option, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L127-L129 - - -If this expression is a let statement, returns the let pattern as an `Expr`, -the optional type annotation, and the assigned expression. - -### as_member_access - -```rust title="as_member_access" showLineNumbers -comptime fn as_member_access(self) -> Option<(Expr, Quoted)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L134-L136 - - -If this expression is a member access `foo.bar`, return the struct/tuple -expression and the field. The field will be represented as a quoted value. - -### as_method_call - -```rust title="as_method_call" showLineNumbers -comptime fn as_method_call(self) -> Option<(Expr, Quoted, [UnresolvedType], [Expr])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L141-L143 - - -If this expression is a method call `foo.bar::(arg1, ..., argN)`, return -the receiver, method name, a slice of each generic argument, and a slice of each argument. - -### as_repeated_element_array - -```rust title="as_repeated_element_array" showLineNumbers -comptime fn as_repeated_element_array(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L148-L150 - - -If this expression is a repeated element array `[elem; length]`, return -the repeated element and the length expressions. - -### as_repeated_element_slice - -```rust title="as_repeated_element_slice" showLineNumbers -comptime fn as_repeated_element_slice(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L155-L157 - - -If this expression is a repeated element slice `[elem; length]`, return -the repeated element and the length expressions. - -### as_slice - -```rust title="as_slice" showLineNumbers -comptime fn as_slice(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L162-L164 - - -If this expression is a slice literal `&[elem1, ..., elemN]`, -return each element of the slice. - -### as_tuple - -```rust title="as_tuple" showLineNumbers -comptime fn as_tuple(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L169-L171 - - -If this expression is a tuple `(field1, ..., fieldN)`, -return each element of the tuple. - -### as_unary_op - -```rust title="as_unary_op" showLineNumbers -comptime fn as_unary_op(self) -> Option<(UnaryOp, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L176-L178 - - -If this expression is a unary operation ` `, -return the unary operator as well as the right-hand side expression. - -### as_unsafe - -```rust title="as_unsafe" showLineNumbers -comptime fn as_unsafe(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L183-L185 - - -If this expression is an `unsafe { stmt1; ...; stmtN }` block, -return each statement inside in a slice. - -### has_semicolon - -```rust title="has_semicolon" showLineNumbers -comptime fn has_semicolon(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L204-L206 - - -`true` if this expression is trailed by a semicolon. E.g. - -``` -comptime { - let expr1 = quote { 1 + 2 }.as_expr().unwrap(); - let expr2 = quote { 1 + 2; }.as_expr().unwrap(); - - assert(expr1.as_binary_op().is_some()); - assert(expr2.as_binary_op().is_some()); - - assert(!expr1.has_semicolon()); - assert(expr2.has_semicolon()); -} -``` - -### is_break - -```rust title="is_break" showLineNumbers -comptime fn is_break(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L210-L212 - - -`true` if this expression is `break`. - -### is_continue - -```rust title="is_continue" showLineNumbers -comptime fn is_continue(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L216-L218 - - -`true` if this expression is `continue`. - -### modify - -```rust title="modify" showLineNumbers -comptime fn modify(self, f: fn[Env](Expr) -> Option) -> Expr { -``` -> Source code: noir_stdlib/src/meta/expr.nr#L227-L229 - - -Applies a mapping function to this expression and to all of its sub-expressions. -`f` will be applied to each sub-expression first, then applied to the expression itself. - -This happens recursively for every expression within `self`. - -For example, calling `modify` on `(&[1], &[2, 3])` with an `f` that returns `Option::some` -for expressions that are integers, doubling them, would return `(&[2], &[4, 6])`. - -### quoted - -```rust title="quoted" showLineNumbers -comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/expr.nr#L264-L266 - - -Returns this expression as a `Quoted` value. It's the same as `quote { $self }`. - -### resolve - -```rust title="resolve" showLineNumbers -comptime fn resolve(self, in_function: Option) -> TypedExpr {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L280-L282 - - -Resolves and type-checks this expression and returns the result as a `TypedExpr`. - -The `in_function` argument specifies where the expression is resolved: -- If it's `none`, the expression is resolved in the function where `resolve` was called -- If it's `some`, the expression is resolved in the given function - -If any names used by this expression are not in scope or if there are any type errors, -this will give compiler errors as if the expression was written directly into -the current `comptime` function. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/function_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/function_def.md deleted file mode 100644 index e490af68f7f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/function_def.md +++ /dev/null @@ -1,166 +0,0 @@ ---- -title: FunctionDefinition ---- - -`std::meta::function_def` contains methods on the built-in `FunctionDefinition` type representing -a function definition in the source program. - -## Methods - -### add_attribute - -```rust title="add_attribute" showLineNumbers -comptime fn add_attribute(self, attribute: str) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L3-L5 - - -Adds an attribute to the function. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### body - -```rust title="body" showLineNumbers -comptime fn body(self) -> Expr {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L8-L10 - - -Returns the body of the function as an expression. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L13-L15 - - -Returns true if this function has a custom attribute with the given name. - -### is_unconstrained - -```rust title="is_unconstrained" showLineNumbers -comptime fn is_unconstrained(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L18-L20 - - -Returns true if this function is unconstrained. - -### module - -```rust title="module" showLineNumbers -comptime fn module(self) -> Module {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L23-L25 - - -Returns the module where the function is defined. - -### name - -```rust title="name" showLineNumbers -comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L28-L30 - - -Returns the name of the function. - -### parameters - -```rust title="parameters" showLineNumbers -comptime fn parameters(self) -> [(Quoted, Type)] {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L33-L35 - - -Returns each parameter of the function as a tuple of (parameter pattern, parameter type). - -### return_type - -```rust title="return_type" showLineNumbers -comptime fn return_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L38-L40 - - -The return type of the function. - -### set_body - -```rust title="set_body" showLineNumbers -comptime fn set_body(self, body: Expr) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L43-L45 - - -Mutate the function body to a new expression. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_parameters - -```rust title="set_parameters" showLineNumbers -comptime fn set_parameters(self, parameters: [(Quoted, Type)]) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L48-L50 - - -Mutates the function's parameters to a new set of parameters. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -Expects a slice of (parameter pattern, parameter type) for each parameter. Requires -each parameter pattern to be a syntactically valid parameter. - -### set_return_type - -```rust title="set_return_type" showLineNumbers -comptime fn set_return_type(self, return_type: Type) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L53-L55 - - -Mutates the function's return type to a new type. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_return_public - -```rust title="set_return_public" showLineNumbers -comptime fn set_return_public(self, public: bool) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L58-L60 - - -Mutates the function's return visibility to public (if `true` is given) or private (if `false` is given). -This is only valid on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_unconstrained - -```rust title="set_unconstrained" showLineNumbers -comptime fn set_unconstrained(self, value: bool) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L63-L65 - - -Mutates the function to be unconstrained (if `true` is given) or not (if `false` is given). -This is only valid on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -## Trait Implementations - -```rust -impl Eq for FunctionDefinition -impl Hash for FunctionDefinition -``` - -Note that each function is assigned a unique ID internally and this is what is used for -equality and hashing. So even functions with identical signatures and bodies may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/index.md deleted file mode 100644 index f81c89e05ae..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/index.md +++ /dev/null @@ -1,210 +0,0 @@ ---- -title: Metaprogramming -description: Noir's Metaprogramming API -keywords: [metaprogramming, comptime, macros, macro, quote, unquote] ---- - -`std::meta` is the entry point for Noir's metaprogramming API. This consists of `comptime` functions -and types used for inspecting and modifying Noir programs. - -## Functions - -### type_of - -```rust title="type_of" showLineNumbers -pub comptime fn type_of(x: T) -> Type {} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L27-L29 - - -Returns the type of a variable at compile-time. - -Example: -```rust -comptime { - let x: i32 = 1; - let x_type: Type = std::meta::type_of(x); - - assert_eq(x_type, quote { i32 }.as_type()); -} -``` - -### unquote - -```rust title="unquote" showLineNumbers -pub comptime fn unquote(code: Quoted) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L19-L21 - - -Unquotes the passed-in token stream where this function was called. - -Example: -```rust -comptime { - let code = quote { 1 + 2 }; - - // let x = 1 + 2; - let x = unquote!(code); -} -``` - -### derive - -```rust title="derive" showLineNumbers -#[varargs] -pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L47-L50 - - -Attribute placed on struct definitions. - -Creates a trait impl for each trait passed in as an argument. -To do this, the trait must have a derive handler registered -with `derive_via` beforehand. The traits in the stdlib that -can be derived this way are `Eq`, `Ord`, `Default`, and `Hash`. - -Example: -```rust -#[derive(Eq, Default)] -struct Foo { - x: i32, - y: T, -} - -fn main() { - let foo1 = Foo::default(); - let foo2 = Foo { x: 0, y: &[0] }; - assert_eq(foo1, foo2); -} -``` - -### derive_via - -```rust title="derive_via_signature" showLineNumbers -pub comptime fn derive_via(t: TraitDefinition, f: DeriveFunction) { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L69-L71 - - -Attribute placed on trait definitions. - -Registers a function to create impls for the given trait -when the trait is used in a `derive` call. Users may use -this to register their own functions to enable their traits -to be derived by `derive`. - -Because this function requires a function as an argument which -should produce a trait impl for any given struct, users may find -it helpful to use a function like `std::meta::make_trait_impl` to -help creating these impls. - -Example: -```rust -#[derive_via(derive_do_nothing)] -trait DoNothing { - fn do_nothing(self); -} - -comptime fn derive_do_nothing(s: StructDefinition) -> Quoted { - let typ = s.as_type(); - quote { - impl DoNothing for $typ { - fn do_nothing(self) { - println("Nothing"); - } - } - } -} -``` - -As another example, `derive_eq` in the stdlib is used to derive the `Eq` -trait for any struct. It makes use of `make_trait_impl` to do this: - -```rust title="derive_eq" showLineNumbers -comptime fn derive_eq(s: StructDefinition) -> Quoted { - let signature = quote { fn eq(_self: Self, _other: Self) -> bool }; - let for_each_field = |name| quote { (_self.$name == _other.$name) }; - let body = |fields| { - if s.fields().len() == 0 { - quote { true } - } else { - fields - } - }; - crate::meta::make_trait_impl(s, quote { Eq }, signature, for_each_field, quote { & }, body) -} -``` -> Source code: noir_stdlib/src/cmp.nr#L10-L23 - - -### make_trait_impl - -```rust title="make_trait_impl" showLineNumbers -pub comptime fn make_trait_impl( - s: StructDefinition, - trait_name: Quoted, - function_signature: Quoted, - for_each_field: fn[Env1](Quoted) -> Quoted, - join_fields_with: Quoted, - body: fn[Env2](Quoted) -> Quoted -) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L88-L97 - - -A helper function to more easily create trait impls while deriving traits. - -Note that this function only works for traits which: -1. Have only one method -2. Have no generics on the trait itself. - - E.g. Using this on a trait such as `trait Foo { ... }` will result in the - generated impl incorrectly missing the `T` generic. - -If your trait fits these criteria then `make_trait_impl` is likely the easiest -way to write your derive handler. The arguments are as follows: - -- `s`: The struct to make the impl for -- `trait_name`: The name of the trait to derive. E.g. `quote { Eq }`. -- `function_signature`: The signature of the trait method to derive. E.g. `fn eq(self, other: Self) -> bool`. -- `for_each_field`: An operation to be performed on each field. E.g. `|name| quote { (self.$name == other.$name) }`. -- `join_fields_with`: A separator to join each result of `for_each_field` with. - E.g. `quote { & }`. You can also use an empty `quote {}` for no separator. -- `body`: The result of the field operations are passed into this function for any final processing. - This is the place to insert any setup/teardown code the trait requires. If the trait doesn't require - any such code, you can return the body as-is: `|body| body`. - -Example deriving `Hash`: - -```rust title="derive_hash" showLineNumbers -comptime fn derive_hash(s: StructDefinition) -> Quoted { - let name = quote { Hash }; - let signature = quote { fn hash(_self: Self, _state: &mut H) where H: std::hash::Hasher }; - let for_each_field = |name| quote { _self.$name.hash(_state); }; - crate::meta::make_trait_impl(s, name, signature, for_each_field, quote {}, |fields| fields) -} -``` -> Source code: noir_stdlib/src/hash/mod.nr#L144-L151 - - -Example deriving `Ord`: - -```rust title="derive_ord" showLineNumbers -comptime fn derive_ord(s: StructDefinition) -> Quoted { - let signature = quote { fn cmp(_self: Self, _other: Self) -> std::cmp::Ordering }; - let for_each_field = |name| quote { - if result == std::cmp::Ordering::equal() { - result = _self.$name.cmp(_other.$name); - } - }; - let body = |fields| quote { - let mut result = std::cmp::Ordering::equal(); - $fields - result - }; - crate::meta::make_trait_impl(s, quote { Ord }, signature, for_each_field, quote {}, body) -} -``` -> Source code: noir_stdlib/src/cmp.nr#L181-L196 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/module.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/module.md deleted file mode 100644 index efd3e61e125..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/module.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: Module ---- - -`std::meta::module` contains methods on the built-in `Module` type which represents a module in the source program. -Note that this type represents a module generally, it isn't limited to only `mod my_submodule { ... }` -declarations in the source program. - -## Methods - -### add_item - -```rust title="add_item" showLineNumbers -comptime fn add_item(self, item: Quoted) {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L3-L5 - - -Adds a top-level item (a function, a struct, a global, etc.) to the module. -Adding multiple items in one go is also valid if the `Quoted` value has multiple items in it. -Note that the items are type-checked as if they are inside the module they are being added to. - -### functions - -```rust title="functions" showLineNumbers -comptime fn functions(self) -> [FunctionDefinition] {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L18-L20 - - -Returns each function defined in the module. - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L8-L10 - - -Returns true if this module has a custom attribute with the given name. - -### is_contract - -```rust title="is_contract" showLineNumbers -comptime fn is_contract(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L13-L15 - - -`true` if this module is a contract module (was declared via `contract foo { ... }`). - -### name - -```rust title="name" showLineNumbers -comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L28-L30 - - -Returns the name of the module. - -### structs - -```rust title="structs" showLineNumbers -comptime fn structs(self) -> [StructDefinition] {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L23-L25 - - -Returns each struct defined in the module. - -## Trait Implementations - -```rust -impl Eq for Module -impl Hash for Module -``` - -Note that each module is assigned a unique ID internally and this is what is used for -equality and hashing. So even modules with identical names and contents may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/op.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/op.md deleted file mode 100644 index 18d1f0768fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/op.md +++ /dev/null @@ -1,244 +0,0 @@ ---- -title: UnaryOp and BinaryOp ---- - -`std::meta::op` contains the `UnaryOp` and `BinaryOp` types as well as methods on them. -These types are used to represent a unary or binary operator respectively in Noir source code. - -## Types - -### UnaryOp - -Represents a unary operator. One of `-`, `!`, `&mut`, or `*`. - -### Methods - -#### is_minus - -```rust title="is_minus" showLineNumbers -pub fn is_minus(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L7-L9 - - -Returns `true` if this operator is `-`. - -#### is_not - -```rust title="is_not" showLineNumbers -pub fn is_not(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L13-L15 - - -`true` if this operator is `!` - -#### is_mutable_reference - -```rust title="is_mutable_reference" showLineNumbers -pub fn is_mutable_reference(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L19-L21 - - -`true` if this operator is `&mut` - -#### is_dereference - -```rust title="is_dereference" showLineNumbers -pub fn is_dereference(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L25-L27 - - -`true` if this operator is `*` - -#### quoted - -```rust title="unary_quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/op.nr#L31-L33 - - -Returns this operator as a `Quoted` value. - -### Trait Implementations - -```rust -impl Eq for UnaryOp -impl Hash for UnaryOp -``` - -### BinaryOp - -Represents a binary operator. One of `+`, `-`, `*`, `/`, `%`, `==`, `!=`, `<`, `<=`, `>`, `>=`, `&`, `|`, `^`, `>>`, or `<<`. - -### Methods - -#### is_add - -```rust title="is_add" showLineNumbers -pub fn is_add(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L55-L57 - - -`true` if this operator is `+` - -#### is_subtract - -```rust title="is_subtract" showLineNumbers -pub fn is_subtract(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L61-L63 - - -`true` if this operator is `-` - -#### is_multiply - -```rust title="is_multiply" showLineNumbers -pub fn is_multiply(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L67-L69 - - -`true` if this operator is `*` - -#### is_divide - -```rust title="is_divide" showLineNumbers -pub fn is_divide(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L73-L75 - - -`true` if this operator is `/` - -#### is_modulo - -```rust title="is_modulo" showLineNumbers -pub fn is_modulo(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L145-L147 - - -`true` if this operator is `%` - -#### is_equal - -```rust title="is_equal" showLineNumbers -pub fn is_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L79-L81 - - -`true` if this operator is `==` - -#### is_not_equal - -```rust title="is_not_equal" showLineNumbers -pub fn is_not_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L85-L87 - - -`true` if this operator is `!=` - -#### is_less_than - -```rust title="is_less_than" showLineNumbers -pub fn is_less_than(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L91-L93 - - -`true` if this operator is `<` - -#### is_less_than_or_equal - -```rust title="is_less_than_or_equal" showLineNumbers -pub fn is_less_than_or_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L97-L99 - - -`true` if this operator is `<=` - -#### is_greater_than - -```rust title="is_greater_than" showLineNumbers -pub fn is_greater_than(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L103-L105 - - -`true` if this operator is `>` - -#### is_greater_than_or_equal - -```rust title="is_greater_than_or_equal" showLineNumbers -pub fn is_greater_than_or_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L109-L111 - - -`true` if this operator is `>=` - -#### is_and - -```rust title="is_and" showLineNumbers -pub fn is_and(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L115-L117 - - -`true` if this operator is `&` - -#### is_or - -```rust title="is_or" showLineNumbers -pub fn is_or(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L121-L123 - - -`true` if this operator is `|` - -#### is_shift_right - -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L133-L135 - - -`true` if this operator is `>>` - -#### is_shift_left - -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L133-L135 - - -`true` if this operator is `<<` - -#### quoted - -```rust title="binary_quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/op.nr#L151-L153 - - -Returns this operator as a `Quoted` value. - -### Trait Implementations - -```rust -impl Eq for BinaryOp -impl Hash for BinaryOp -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/quoted.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/quoted.md deleted file mode 100644 index cb779bca98f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/quoted.md +++ /dev/null @@ -1,138 +0,0 @@ ---- -title: Quoted ---- - -`std::meta::quoted` contains methods on the built-in `Quoted` type which represents -quoted token streams and is the result of the `quote { ... }` expression. - -## Methods - -### as_expr - -```rust title="as_expr" showLineNumbers -comptime fn as_expr(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L6-L8 - - -Parses the quoted token stream as an expression. Returns `Option::none()` if -the expression failed to parse. - -Example: - -```rust title="as_expr_example" showLineNumbers -#[test] - fn test_expr_as_function_call() { - comptime - { - let expr = quote { foo(42) }.as_expr().unwrap(); - let (_function, args) = expr.as_function_call().unwrap(); - assert_eq(args.len(), 1); - assert_eq(args[0].as_integer().unwrap(), (42, false)); - } - } -``` -> Source code: test_programs/noir_test_success/comptime_expr/src/main.nr#L360-L371 - - -### as_module - -```rust title="as_module" showLineNumbers -comptime fn as_module(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L11-L13 - - -Interprets this token stream as a module path leading to the name of a module. -Returns `Option::none()` if the module isn't found or this token stream cannot be parsed as a path. - -Example: - -```rust title="as_module_example" showLineNumbers -mod baz { - pub mod qux {} -} - -#[test] -fn as_module_test() { - comptime - { - let my_mod = quote { baz::qux }.as_module().unwrap(); - assert_eq(my_mod.name(), quote { qux }); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_module/src/main.nr#L116-L129 - - -### as_trait_constraint - -```rust title="as_trait_constraint" showLineNumbers -comptime fn as_trait_constraint(self) -> TraitConstraint {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L16-L18 - - -Interprets this token stream as a trait constraint (without an object type). -Note that this function panics instead of returning `Option::none()` if the token -stream does not parse and resolve to a valid trait constraint. - -Example: - -```rust title="implements_example" showLineNumbers -fn function_with_where(_x: T) where T: SomeTrait { - comptime - { - let t = quote { T }.as_type(); - let some_trait_i32 = quote { SomeTrait }.as_trait_constraint(); - assert(t.implements(some_trait_i32)); - - assert(t.get_trait_impl(some_trait_i32).is_none()); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L154-L165 - - -### as_type - -```rust title="as_type" showLineNumbers -comptime fn as_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L21-L23 - - -Interprets this token stream as a resolved type. Panics if the token -stream doesn't parse to a type or if the type isn't a valid type in scope. - -```rust title="implements_example" showLineNumbers -fn function_with_where(_x: T) where T: SomeTrait { - comptime - { - let t = quote { T }.as_type(); - let some_trait_i32 = quote { SomeTrait }.as_trait_constraint(); - assert(t.implements(some_trait_i32)); - - assert(t.get_trait_impl(some_trait_i32).is_none()); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L154-L165 - - -### tokens - -```rust title="tokens" showLineNumbers -comptime fn tokens(self) -> [Quoted] {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L26-L28 - - -Returns a slice of the individual tokens that form this token stream. - -## Trait Implementations - -```rust -impl Eq for Quoted -impl Hash for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/struct_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/struct_def.md deleted file mode 100644 index 212c636d12a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/struct_def.md +++ /dev/null @@ -1,177 +0,0 @@ ---- -title: StructDefinition ---- - -`std::meta::struct_def` contains methods on the built-in `StructDefinition` type. -This type corresponds to `struct Name { field1: Type1, ... }` items in the source program. - -## Methods - -### add_attribute - -```rust title="add_attribute" showLineNumbers -comptime fn add_attribute(self, attribute: str) {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L3-L5 - - -Adds an attribute to the struct. - -### add_generic - -```rust title="add_generic" showLineNumbers -comptime fn add_generic(self, generic_name: str) -> Type {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L8-L10 - - -Adds an generic to the struct. Returns the new generic type. -Errors if the given generic name isn't a single identifier or if -the struct already has a generic with the same name. - -This method should be used carefully, if there is existing code referring -to the struct type it may be checked before this function is called and -see the struct with the original number of generics. This method should -thus be preferred to use on code generated from other macros and structs -that are not used in function signatures. - -Example: - -```rust title="add-generic-example" showLineNumbers -comptime fn add_generic(s: StructDefinition) { - assert_eq(s.generics().len(), 0); - let new_generic = s.add_generic("T"); - - let generics = s.generics(); - assert_eq(generics.len(), 1); - assert_eq(generics[0], new_generic); - } -``` -> Source code: test_programs/compile_success_empty/comptime_struct_definition/src/main.nr#L38-L47 - - -### as_type - -```rust title="as_type" showLineNumbers -comptime fn as_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L15-L17 - - -Returns this struct as a type in the source program. If this struct has -any generics, the generics are also included as-is. - -### generics - -```rust title="generics" showLineNumbers -comptime fn generics(self) -> [Type] {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L26-L28 - - -Returns each generic on this struct. - -Example: - -``` -#[example] -struct Foo { - bar: [T; 2], - baz: Baz, -} - -comptime fn example(foo: StructDefinition) { - assert_eq(foo.generics().len(), 2); - - // Fails because `T` isn't in scope - // let t = quote { T }.as_type(); - // assert_eq(foo.generics()[0], t); -} -``` - -### fields - -```rust title="fields" showLineNumbers -comptime fn fields(self) -> [(Quoted, Type)] {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L33-L35 - - -Returns each field of this struct as a pair of (field name, field type). - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L20-L22 - - -Returns true if this struct has a custom attribute with the given name. - -### module - -```rust title="module" showLineNumbers -comptime fn module(self) -> Module {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L38-L40 - - -Returns the module where the struct is defined. - -### name - -```rust title="name" showLineNumbers -comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L43-L45 - - -Returns the name of this struct - -Note that the returned quoted value will be just the struct name, it will -not be the full path to the struct, nor will it include any generics. - -### set_fields - -```rust title="set_fields" showLineNumbers -comptime fn set_fields(self, new_fields: [(Quoted, Type)]) {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L52-L54 - - -Sets the fields of this struct to the given fields list where each element -is a pair of the field's name and the field's type. Expects each field name -to be a single identifier. Note that this will override any previous fields -on this struct. If those should be preserved, use `.fields()` to retrieve the -current fields on the struct type and append the new fields from there. - -Example: - -```rust -// Change this struct to: -// struct Foo { -// a: u32, -// b: i8, -// } -#[mangle_fields] -struct Foo { x: Field } - -comptime fn mangle_fields(s: StructDefinition) { - s.set_fields(&[ - (quote { a }, quote { u32 }.as_type()), - (quote { b }, quote { i8 }.as_type()), - ]); -} -``` - -## Trait Implementations - -```rust -impl Eq for StructDefinition -impl Hash for StructDefinition -``` - -Note that each struct is assigned a unique ID internally and this is what is used for -equality and hashing. So even structs with identical generics and fields may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_constraint.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_constraint.md deleted file mode 100644 index 3106f732b5a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_constraint.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: TraitConstraint ---- - -`std::meta::trait_constraint` contains methods on the built-in `TraitConstraint` type which represents -a trait constraint that can be used to search for a trait implementation. This is similar -syntactically to just the trait itself, but can also contain generic arguments. E.g. `Eq`, `Default`, -`BuildHasher`. - -This type currently has no public methods but it can be used alongside `Type` in `implements` or `get_trait_impl`. - -## Trait Implementations - -```rust -impl Eq for TraitConstraint -impl Hash for TraitConstraint -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_def.md deleted file mode 100644 index a1f363d46ff..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_def.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: TraitDefinition ---- - -`std::meta::trait_def` contains methods on the built-in `TraitDefinition` type. This type -represents trait definitions such as `trait Foo { .. }` at the top-level of a program. - -## Methods - -### as_trait_constraint - -```rust title="as_trait_constraint" showLineNumbers -comptime fn as_trait_constraint(_self: Self) -> TraitConstraint {} -``` -> Source code: noir_stdlib/src/meta/trait_def.nr#L6-L8 - - -Converts this trait into a trait constraint. If there are any generics on this -trait, they will be kept as-is without instantiating or replacing them. - -## Trait Implementations - -```rust -impl Eq for TraitDefinition -impl Hash for TraitDefinition -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_impl.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_impl.md deleted file mode 100644 index 66d31ed2560..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/trait_impl.md +++ /dev/null @@ -1,60 +0,0 @@ ---- -title: TraitImpl ---- - -`std::meta::trait_impl` contains methods on the built-in `TraitImpl` type which represents a trait -implementation such as `impl Foo for Bar { ... }`. - -## Methods - -### trait_generic_args - -```rust title="trait_generic_args" showLineNumbers -comptime fn trait_generic_args(self) -> [Type] {} -``` -> Source code: noir_stdlib/src/meta/trait_impl.nr#L3-L5 - - -Returns any generic arguments on the trait of this trait implementation, if any. - -```rs -impl Foo for Bar { ... } - -comptime { - let bar_type = quote { Bar }.as_type(); - let foo = quote { Foo }.as_trait_constraint(); - - let my_impl: TraitImpl = bar_type.get_trait_impl(foo).unwrap(); - - let generics = my_impl.trait_generic_args(); - assert_eq(generics.len(), 2); - - assert_eq(generics[0], quote { i32 }.as_type()); - assert_eq(generics[1], quote { Field }.as_type()); -} -``` - -### methods - -```rust title="methods" showLineNumbers -comptime fn methods(self) -> [FunctionDefinition] {} -``` -> Source code: noir_stdlib/src/meta/trait_impl.nr#L8-L10 - - -Returns each method in this trait impl. - -Example: - -```rs -comptime { - let i32_type = quote { i32 }.as_type(); - let eq = quote { Eq }.as_trait_constraint(); - - let impl_eq_for_i32: TraitImpl = i32_type.get_trait_impl(eq).unwrap(); - let methods = impl_eq_for_i32.methods(); - - assert_eq(methods.len(), 1); - assert_eq(methods[0].name(), quote { eq }); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/typ.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/typ.md deleted file mode 100644 index 6c9f4b8d087..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/typ.md +++ /dev/null @@ -1,239 +0,0 @@ ---- -title: Type ---- - -`std::meta::typ` contains methods on the built-in `Type` type used for representing -a type in the source program. - -## Functions - -```rust title="fresh_type_variable" showLineNumbers -pub comptime fn fresh_type_variable() -> Type {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L5-L7 - - -Creates and returns an unbound type variable. This is a special kind of type internal -to type checking which will type check with any other type. When it is type checked -against another type it will also be set to that type. For example, if `a` is a type -variable and we have the type equality `(a, i32) = (u8, i32)`, the compiler will set -`a` equal to `u8`. - -Unbound type variables will often be rendered as `_` while printing them. Bound type -variables will appear as the type they are bound to. - -This can be used in conjunction with functions which internally perform type checks -such as `Type::implements` or `Type::get_trait_impl` to potentially grab some of the types used. - -Note that calling `Type::implements` or `Type::get_trait_impl` on a type variable will always -fail. - -Example: - -```rust title="serialize-setup" showLineNumbers -trait Serialize {} - -impl Serialize<1> for Field {} - -impl Serialize for [T; N] - where T: Serialize {} - -impl Serialize for (T, U) - where T: Serialize, U: Serialize {} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L20-L30 - -```rust title="fresh-type-variable-example" showLineNumbers -let typevar1 = std::meta::typ::fresh_type_variable(); - let constraint = quote { Serialize<$typevar1> }.as_trait_constraint(); - let field_type = quote { Field }.as_type(); - - // Search for a trait impl (binding typevar1 to 1 when the impl is found): - assert(field_type.implements(constraint)); - - // typevar1 should be bound to the "1" generic now: - assert_eq(typevar1.as_constant().unwrap(), 1); - - // If we want to do the same with a different type, we need to - // create a new type variable now that `typevar1` is bound - let typevar2 = std::meta::typ::fresh_type_variable(); - let constraint = quote { Serialize<$typevar2> }.as_trait_constraint(); - let array_type = quote { [(Field, Field); 5] }.as_type(); - assert(array_type.implements(constraint)); - - // Now typevar2 should be bound to the serialized pair size 2 times the array length 5 - assert_eq(typevar2.as_constant().unwrap(), 10); -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L130-L150 - - -## Methods - -### as_array - -```rust title="as_array" showLineNumbers -comptime fn as_array(self) -> Option<(Type, Type)> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L11-L13 - - -If this type is an array, return a pair of (element type, size type). - -Example: - -```rust -comptime { - let array_type = quote { [Field; 3] }.as_type(); - let (field_type, three_type) = array_type.as_array().unwrap(); - - assert(field_type.is_field()); - assert_eq(three_type.as_constant().unwrap(), 3); -} -``` - -### as_constant - -```rust title="as_constant" showLineNumbers -comptime fn as_constant(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L16-L18 - - -If this type is a constant integer (such as the `3` in the array type `[Field; 3]`), -return the numeric constant. - -### as_integer - -```rust title="as_integer" showLineNumbers -comptime fn as_integer(self) -> Option<(bool, u8)> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L21-L23 - - -If this is an integer type, return a boolean which is `true` -if the type is signed, as well as the number of bits of this integer type. - -### as_slice - -```rust title="as_slice" showLineNumbers -comptime fn as_slice(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L26-L28 - - -If this is a slice type, return the element type of the slice. - -### as_str - -```rust title="as_str" showLineNumbers -comptime fn as_str(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L31-L33 - - -If this is a `str` type, returns the length `N` as a type. - -### as_struct - -```rust title="as_struct" showLineNumbers -comptime fn as_struct(self) -> Option<(StructDefinition, [Type])> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L36-L38 - - -If this is a struct type, returns the struct in addition to -any generic arguments on this type. - -### as_tuple - -```rust title="as_tuple" showLineNumbers -comptime fn as_tuple(self) -> Option<[Type]> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L41-L43 - - -If this is a tuple type, returns each element type of the tuple. - -### get_trait_impl - -```rust title="get_trait_impl" showLineNumbers -comptime fn get_trait_impl(self, constraint: TraitConstraint) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L46-L48 - - -Retrieves the trait implementation that implements the given -trait constraint for this type. If the trait constraint is not -found, `None` is returned. Note that since the concrete trait implementation -for a trait constraint specified from a `where` clause is unknown, -this function will return `None` in these cases. If you only want to know -whether a type implements a trait, use `implements` instead. - -Example: - -```rust -comptime { - let field_type = quote { Field }.as_type(); - let default = quote { Default }.as_trait_constraint(); - - let the_impl: TraitImpl = field_type.get_trait_impl(default).unwrap(); - assert(the_impl.methods().len(), 1); -} -``` - -### implements - -```rust title="implements" showLineNumbers -comptime fn implements(self, constraint: TraitConstraint) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L51-L53 - - -`true` if this type implements the given trait. Note that unlike -`get_trait_impl` this will also return true for any `where` constraints -in scope. - -Example: - -```rust -fn foo() where T: Default { - comptime { - let field_type = quote { Field }.as_type(); - let default = quote { Default }.as_trait_constraint(); - assert(field_type.implements(default)); - - let t = quote { T }.as_type(); - assert(t.implements(default)); - } -} -``` - -### is_bool - -```rust title="is_bool" showLineNumbers -comptime fn is_bool(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L56-L58 - - -`true` if this type is `bool`. - -### is_field - -```rust title="is_field" showLineNumbers -comptime fn is_field(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L61-L63 - - -`true` if this type is `Field`. - -## Trait Implementations - -```rust -impl Eq for Type -impl Hash for Type -``` -Note that this is syntactic equality, this is not the same as whether two types will type check -to be the same type. Unless type inference or generics are being used however, users should not -typically have to worry about this distinction unless `std::meta::typ::fresh_type_variable` is used. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/typed_expr.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/typed_expr.md deleted file mode 100644 index 1ee71c8b064..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/typed_expr.md +++ /dev/null @@ -1,27 +0,0 @@ ---- -title: TypedExpr ---- - -`std::meta::typed_expr` contains methods on the built-in `TypedExpr` type for resolved and type-checked expressions. - -## Methods - -### get_type - -```rust title="as_function_definition" showLineNumbers -comptime fn as_function_definition(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typed_expr.nr#L7-L9 - - -If this expression refers to a function definitions, returns it. Otherwise returns `Option::none()`. - -### get_type - -```rust title="get_type" showLineNumbers -comptime fn get_type(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typed_expr.nr#L13-L15 - - -Returns the type of the expression, or `Option::none()` if there were errors when the expression was previously resolved. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/unresolved_type.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/unresolved_type.md deleted file mode 100644 index d6f2b1494bb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/meta/unresolved_type.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: UnresolvedType ---- - -`std::meta::unresolved_type` contains methods on the built-in `UnresolvedType` type for the syntax of types. - -## Methods - -### is_field - -```rust title="is_field" showLineNumbers -comptime fn is_field(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L3-L5 - - -Returns true if this type refers to the Field type. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/options.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/options.md deleted file mode 100644 index a1bd4e1de5f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/options.md +++ /dev/null @@ -1,101 +0,0 @@ ---- -title: Option Type ---- - -The `Option` type is a way to express that a value might be present (`Some(T))` or absent (`None`). It's a safer way to handle potential absence of values, compared to using nulls in many other languages. - -```rust -struct Option { - None, - Some(T), -} -``` - -The `Option` type, already imported into your Noir program, can be used directly: - -```rust -fn main() { - let none = Option::none(); - let some = Option::some(3); -} -``` - -See [this test](https://github.com/noir-lang/noir/blob/5cbfb9c4a06c8865c98ff2b594464b037d821a5c/crates/nargo_cli/tests/test_data/option/src/main.nr) for a more comprehensive set of examples of each of the methods described below. - -## Methods - -### none - -Constructs a none value. - -### some - -Constructs a some wrapper around a given value. - -### is_none - -Returns true if the Option is None. - -### is_some - -Returns true of the Option is Some. - -### unwrap - -Asserts `self.is_some()` and returns the wrapped value. - -### unwrap_unchecked - -Returns the inner value without asserting `self.is_some()`. This method can be useful within an if condition when we already know that `option.is_some()`. If the option is None, there is no guarantee what value will be returned, only that it will be of type T for an `Option`. - -### unwrap_or - -Returns the wrapped value if `self.is_some()`. Otherwise, returns the given default value. - -### unwrap_or_else - -Returns the wrapped value if `self.is_some()`. Otherwise, calls the given function to return a default value. - -### expect - -Asserts `self.is_some()` with a provided custom message and returns the contained `Some` value. The custom message is expected to be a format string. - -### map - -If self is `Some(x)`, this returns `Some(f(x))`. Otherwise, this returns `None`. - -### map_or - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns the given default value. - -### map_or_else - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns `default()`. - -### and - -Returns None if self is None. Otherwise, this returns `other`. - -### and_then - -If self is None, this returns None. Otherwise, this calls the given function with the Some value contained within self, and returns the result of that call. In some languages this function is called `flat_map` or `bind`. - -### or - -If self is Some, return self. Otherwise, return `other`. - -### or_else - -If self is Some, return self. Otherwise, return `default()`. - -### xor - -If only one of the two Options is Some, return that option. Otherwise, if both options are Some or both are None, None is returned. - -### filter - -Returns `Some(x)` if self is `Some(x)` and `predicate(x)` is true. Otherwise, this returns `None`. - -### flatten - -Flattens an `Option>` into a `Option`. This returns `None` if the outer Option is None. Otherwise, this returns the inner Option. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/recursion.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/recursion.mdx deleted file mode 100644 index 60414a2fa51..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/recursion.mdx +++ /dev/null @@ -1,85 +0,0 @@ ---- -title: Recursive Proofs -description: Learn about how to write recursive proofs in Noir. -keywords: [recursion, recursive proofs, verification_key, verify_proof] ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports recursively verifying proofs, meaning you verify the proof of a Noir program in another Noir program. This enables creating proofs of arbitrary size by doing step-wise verification of smaller components of a large proof. - -Read [the explainer on recursion](../../explainers/explainer-recursion.md) to know more about this function and the [guide on how to use it.](../../how_to/how-to-recursion.md) - -## The `#[recursive]` Attribute - -In Noir, the `#[recursive]` attribute is used to indicate that a circuit is designed for recursive proof generation. When applied, it informs the compiler and the tooling that the circuit should be compiled in a way that makes its proofs suitable for recursive verification. This attribute eliminates the need for manual flagging of recursion at the tooling level, streamlining the proof generation process for recursive circuits. - -### Example usage with `#[recursive]` - -```rust -#[recursive] -fn main(x: Field, y: pub Field) { - assert(x == y, "x and y are not equal"); -} - -// This marks the circuit as recursion-friendly and indicates that proofs generated from this circuit -// are intended for recursive verification. -``` - -By incorporating this attribute directly in the circuit's definition, tooling like Nargo and NoirJS can automatically execute recursive-specific duties for Noir programs (e.g. recursive-friendly proof artifact generation) without additional flags or configurations. - -## Verifying Recursive Proofs - -```rust -#[foreign(recursive_aggregation)] -pub fn verify_proof(verification_key: [Field], proof: [Field], public_inputs: [Field], key_hash: Field) {} -``` - - - -## Example usage - -```rust - -fn main( - verification_key : [Field; 114], - proof : [Field; 93], - public_inputs : [Field; 1], - key_hash : Field, - proof_b : [Field; 93], -) { - std::verify_proof( - verification_key, - proof, - public_inputs, - key_hash - ); - - std::verify_proof( - verification_key, - proof_b, - public_inputs, - key_hash - ); -} -``` - -You can see a full example of recursive proofs in [this example recursion demo repo](https://github.com/noir-lang/noir-examples/tree/master/recursion). - -## Parameters - -### `verification_key` - -The verification key for the zk program that is being verified. - -### `proof` - -The proof for the zk program that is being verified. - -### `public_inputs` - -These represent the public inputs of the proof we are verifying. - -### `key_hash` - -A key hash is used to check the validity of the verification key. The circuit implementing this opcode can use this hash to ensure that the key provided to the circuit matches the key produced by the circuit creator. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/traits.md deleted file mode 100644 index 74ed6276234..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/traits.md +++ /dev/null @@ -1,625 +0,0 @@ ---- -title: Traits -description: Noir's stdlib provides a few commonly used traits. -keywords: [traits, trait, interface, protocol, default, add, eq] ---- - -## `std::default` - -### `std::default::Default` - -```rust title="default-trait" showLineNumbers -pub trait Default { - fn default() -> Self; -} -``` -> Source code: noir_stdlib/src/default.nr#L4-L8 - - -Constructs a default value of a type. - -Implementations: -```rust -impl Default for Field { .. } - -impl Default for i8 { .. } -impl Default for i16 { .. } -impl Default for i32 { .. } -impl Default for i64 { .. } - -impl Default for u8 { .. } -impl Default for u16 { .. } -impl Default for u32 { .. } -impl Default for u64 { .. } - -impl Default for () { .. } -impl Default for bool { .. } - -impl Default for [T; N] - where T: Default { .. } - -impl Default for [T] { .. } - -impl Default for (A, B) - where A: Default, B: Default { .. } - -impl Default for (A, B, C) - where A: Default, B: Default, C: Default { .. } - -impl Default for (A, B, C, D) - where A: Default, B: Default, C: Default, D: Default { .. } - -impl Default for (A, B, C, D, E) - where A: Default, B: Default, C: Default, D: Default, E: Default { .. } -``` - -For primitive integer types, the return value of `default` is `0`. Container -types such as arrays are filled with default values of their element type, -except slices whose length is unknown and thus defaulted to zero. - ---- - -## `std::convert` - -### `std::convert::From` - -```rust title="from-trait" showLineNumbers -pub trait From { - fn from(input: T) -> Self; -} -``` -> Source code: noir_stdlib/src/convert.nr#L1-L5 - - -The `From` trait defines how to convert from a given type `T` to the type on which the trait is implemented. - -The Noir standard library provides a number of implementations of `From` between primitive types. -```rust title="from-impls" showLineNumbers -// Unsigned integers - -impl From for u32 { - fn from(value: u8) -> u32 { - value as u32 - } -} - -impl From for u64 { - fn from(value: u8) -> u64 { - value as u64 - } -} -impl From for u64 { - fn from(value: u32) -> u64 { - value as u64 - } -} - -impl From for Field { - fn from(value: u8) -> Field { - value as Field - } -} -impl From for Field { - fn from(value: u32) -> Field { - value as Field - } -} -impl From for Field { - fn from(value: u64) -> Field { - value as Field - } -} - -// Signed integers - -impl From for i32 { - fn from(value: i8) -> i32 { - value as i32 - } -} - -impl From for i64 { - fn from(value: i8) -> i64 { - value as i64 - } -} -impl From for i64 { - fn from(value: i32) -> i64 { - value as i64 - } -} - -// Booleans -impl From for u8 { - fn from(value: bool) -> u8 { - value as u8 - } -} -impl From for u32 { - fn from(value: bool) -> u32 { - value as u32 - } -} -impl From for u64 { - fn from(value: bool) -> u64 { - value as u64 - } -} -impl From for i8 { - fn from(value: bool) -> i8 { - value as i8 - } -} -impl From for i32 { - fn from(value: bool) -> i32 { - value as i32 - } -} -impl From for i64 { - fn from(value: bool) -> i64 { - value as i64 - } -} -impl From for Field { - fn from(value: bool) -> Field { - value as Field - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L25-L116 - - -#### When to implement `From` - -As a general rule of thumb, `From` may be implemented in the [situations where it would be suitable in Rust](https://doc.rust-lang.org/std/convert/trait.From.html#when-to-implement-from): - -- The conversion is *infallible*: Noir does not provide an equivalent to Rust's `TryFrom`, if the conversion can fail then provide a named method instead. -- The conversion is *lossless*: semantically, it should not lose or discard information. For example, `u32: From` can losslessly convert any `u16` into a valid `u32` such that the original `u16` can be recovered. On the other hand, `u16: From` should not be implemented as `2**16` is a `u32` which cannot be losslessly converted into a `u16`. -- The conversion is *value-preserving*: the conceptual kind and meaning of the resulting value is the same, even though the Noir type and technical representation might be different. While it's possible to infallibly and losslessly convert a `u8` into a `str<2>` hex representation, `4u8` and `"04"` are too different for `str<2>: From` to be implemented. -- The conversion is *obvious*: it's the only reasonable conversion between the two types. If there's ambiguity on how to convert between them such that the same input could potentially map to two different values then a named method should be used. For instance rather than implementing `U128: From<[u8; 16]>`, the methods `U128::from_le_bytes` and `U128::from_be_bytes` are used as otherwise the endianness of the array would be ambiguous, resulting in two potential values of `U128` from the same byte array. - -One additional recommendation specific to Noir is: -- The conversion is *efficient*: it's relatively cheap to convert between the two types. Due to being a ZK DSL, it's more important to avoid unnecessary computation compared to Rust. If the implementation of `From` would encourage users to perform unnecessary conversion, resulting in additional proving time, then it may be preferable to expose functionality such that this conversion may be avoided. - -### `std::convert::Into` - -The `Into` trait is defined as the reciprocal of `From`. It should be easy to convince yourself that if we can convert to type `A` from type `B`, then it's possible to convert type `B` into type `A`. - -For this reason, implementing `From` on a type will automatically generate a matching `Into` implementation. One should always prefer implementing `From` over `Into` as implementing `Into` will not generate a matching `From` implementation. - -```rust title="into-trait" showLineNumbers -pub trait Into { - fn into(self) -> T; -} - -impl Into for U where T: From { - fn into(self) -> T { - T::from(self) - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L13-L23 - - -`Into` is most useful when passing function arguments where the types don't quite match up with what the function expects. In this case, the compiler has enough type information to perform the necessary conversion by just appending `.into()` onto the arguments in question. - ---- - -## `std::cmp` - -### `std::cmp::Eq` - -```rust title="eq-trait" showLineNumbers -pub trait Eq { - fn eq(self, other: Self) -> bool; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L4-L8 - - -Returns `true` if `self` is equal to `other`. Implementing this trait on a type -allows the type to be used with `==` and `!=`. - -Implementations: -```rust -impl Eq for Field { .. } - -impl Eq for i8 { .. } -impl Eq for i16 { .. } -impl Eq for i32 { .. } -impl Eq for i64 { .. } - -impl Eq for u8 { .. } -impl Eq for u16 { .. } -impl Eq for u32 { .. } -impl Eq for u64 { .. } - -impl Eq for () { .. } -impl Eq for bool { .. } - -impl Eq for [T; N] - where T: Eq { .. } - -impl Eq for [T] - where T: Eq { .. } - -impl Eq for (A, B) - where A: Eq, B: Eq { .. } - -impl Eq for (A, B, C) - where A: Eq, B: Eq, C: Eq { .. } - -impl Eq for (A, B, C, D) - where A: Eq, B: Eq, C: Eq, D: Eq { .. } - -impl Eq for (A, B, C, D, E) - where A: Eq, B: Eq, C: Eq, D: Eq, E: Eq { .. } -``` - -### `std::cmp::Ord` - -```rust title="ord-trait" showLineNumbers -pub trait Ord { - fn cmp(self, other: Self) -> Ordering; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L175-L179 - - -`a.cmp(b)` compares two values returning `Ordering::less()` if `a < b`, -`Ordering::equal()` if `a == b`, or `Ordering::greater()` if `a > b`. -Implementing this trait on a type allows `<`, `<=`, `>`, and `>=` to be -used on values of the type. - -`std::cmp` also provides `max` and `min` functions for any type which implements the `Ord` trait. - -Implementations: - -```rust -impl Ord for u8 { .. } -impl Ord for u16 { .. } -impl Ord for u32 { .. } -impl Ord for u64 { .. } - -impl Ord for i8 { .. } -impl Ord for i16 { .. } -impl Ord for i32 { .. } - -impl Ord for i64 { .. } - -impl Ord for () { .. } -impl Ord for bool { .. } - -impl Ord for [T; N] - where T: Ord { .. } - -impl Ord for [T] - where T: Ord { .. } - -impl Ord for (A, B) - where A: Ord, B: Ord { .. } - -impl Ord for (A, B, C) - where A: Ord, B: Ord, C: Ord { .. } - -impl Ord for (A, B, C, D) - where A: Ord, B: Ord, C: Ord, D: Ord { .. } - -impl Ord for (A, B, C, D, E) - where A: Ord, B: Ord, C: Ord, D: Ord, E: Ord { .. } -``` - ---- - -## `std::ops` - -### `std::ops::Add`, `std::ops::Sub`, `std::ops::Mul`, and `std::ops::Div` - -These traits abstract over addition, subtraction, multiplication, and division respectively. -Implementing these traits for a given type will also allow that type to be used with the corresponding operator -for that trait (`+` for Add, etc) in addition to the normal method names. - -```rust title="add-trait" showLineNumbers -pub trait Add { - fn add(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L1-L5 - -```rust title="sub-trait" showLineNumbers -pub trait Sub { - fn sub(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L60-L64 - -```rust title="mul-trait" showLineNumbers -pub trait Mul { - fn mul(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L119-L123 - -```rust title="div-trait" showLineNumbers -pub trait Div { - fn div(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L178-L182 - - -The implementations block below is given for the `Add` trait, but the same types that implement -`Add` also implement `Sub`, `Mul`, and `Div`. - -Implementations: -```rust -impl Add for Field { .. } - -impl Add for i8 { .. } -impl Add for i16 { .. } -impl Add for i32 { .. } -impl Add for i64 { .. } - -impl Add for u8 { .. } -impl Add for u16 { .. } -impl Add for u32 { .. } -impl Add for u64 { .. } -``` - -### `std::ops::Rem` - -```rust title="rem-trait" showLineNumbers -pub trait Rem { - fn rem(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L237-L241 - - -`Rem::rem(a, b)` is the remainder function returning the result of what is -left after dividing `a` and `b`. Implementing `Rem` allows the `%` operator -to be used with the implementation type. - -Unlike other numeric traits, `Rem` is not implemented for `Field`. - -Implementations: -```rust -impl Rem for u8 { fn rem(self, other: u8) -> u8 { self % other } } -impl Rem for u16 { fn rem(self, other: u16) -> u16 { self % other } } -impl Rem for u32 { fn rem(self, other: u32) -> u32 { self % other } } -impl Rem for u64 { fn rem(self, other: u64) -> u64 { self % other } } - -impl Rem for i8 { fn rem(self, other: i8) -> i8 { self % other } } -impl Rem for i16 { fn rem(self, other: i16) -> i16 { self % other } } -impl Rem for i32 { fn rem(self, other: i32) -> i32 { self % other } } -impl Rem for i64 { fn rem(self, other: i64) -> i64 { self % other } } -``` - -### `std::ops::Neg` - -```rust title="neg-trait" showLineNumbers -pub trait Neg { - fn neg(self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L290-L294 - - -`Neg::neg` is equivalent to the unary negation operator `-`. - -Implementations: -```rust title="neg-trait-impls" showLineNumbers -impl Neg for Field { - fn neg(self) -> Field { - -self - } -} - -impl Neg for i8 { - fn neg(self) -> i8 { - -self - } -} -impl Neg for i16 { - fn neg(self) -> i16 { - -self - } -} -impl Neg for i32 { - fn neg(self) -> i32 { - -self - } -} -impl Neg for i64 { - fn neg(self) -> i64 { - -self - } -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L296-L323 - - -### `std::ops::Not` - -```rust title="not-trait" showLineNumbers -pub trait Not { - fn not(self: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L1-L5 - - -`Not::not` is equivalent to the unary bitwise NOT operator `!`. - -Implementations: -```rust title="not-trait-impls" showLineNumbers -impl Not for bool { - fn not(self) -> bool { - !self - } -} - -impl Not for u64 { - fn not(self) -> u64 { - !self - } -} -impl Not for u32 { - fn not(self) -> u32 { - !self - } -} -impl Not for u16 { - fn not(self) -> u16 { - !self - } -} -impl Not for u8 { - fn not(self) -> u8 { - !self - } -} -impl Not for u1 { - fn not(self) -> u1 { - !self - } -} - -impl Not for i8 { - fn not(self) -> i8 { - !self - } -} -impl Not for i16 { - fn not(self) -> i16 { - !self - } -} -impl Not for i32 { - fn not(self) -> i32 { - !self - } -} -impl Not for i64 { - fn not(self) -> i64 { - !self - } -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L7-L60 - - -### `std::ops::{ BitOr, BitAnd, BitXor }` - -```rust title="bitor-trait" showLineNumbers -pub trait BitOr { - fn bitor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L62-L66 - -```rust title="bitand-trait" showLineNumbers -pub trait BitAnd { - fn bitand(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L121-L125 - -```rust title="bitxor-trait" showLineNumbers -pub trait BitXor { - fn bitxor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L180-L184 - - -Traits for the bitwise operations `|`, `&`, and `^`. - -Implementing `BitOr`, `BitAnd` or `BitXor` for a type allows the `|`, `&`, or `^` operator respectively -to be used with the type. - -The implementations block below is given for the `BitOr` trait, but the same types that implement -`BitOr` also implement `BitAnd` and `BitXor`. - -Implementations: -```rust -impl BitOr for bool { fn bitor(self, other: bool) -> bool { self | other } } - -impl BitOr for u8 { fn bitor(self, other: u8) -> u8 { self | other } } -impl BitOr for u16 { fn bitor(self, other: u16) -> u16 { self | other } } -impl BitOr for u32 { fn bitor(self, other: u32) -> u32 { self | other } } -impl BitOr for u64 { fn bitor(self, other: u64) -> u64 { self | other } } - -impl BitOr for i8 { fn bitor(self, other: i8) -> i8 { self | other } } -impl BitOr for i16 { fn bitor(self, other: i16) -> i16 { self | other } } -impl BitOr for i32 { fn bitor(self, other: i32) -> i32 { self | other } } -impl BitOr for i64 { fn bitor(self, other: i64) -> i64 { self | other } } -``` - -### `std::ops::{ Shl, Shr }` - -```rust title="shl-trait" showLineNumbers -pub trait Shl { - fn shl(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L239-L243 - -```rust title="shr-trait" showLineNumbers -pub trait Shr { - fn shr(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L292-L296 - - -Traits for a bit shift left and bit shift right. - -Implementing `Shl` for a type allows the left shift operator (`<<`) to be used with the implementation type. -Similarly, implementing `Shr` allows the right shift operator (`>>`) to be used with the type. - -Note that bit shifting is not currently implemented for signed types. - -The implementations block below is given for the `Shl` trait, but the same types that implement -`Shl` also implement `Shr`. - -Implementations: -```rust -impl Shl for u8 { fn shl(self, other: u8) -> u8 { self << other } } -impl Shl for u16 { fn shl(self, other: u16) -> u16 { self << other } } -impl Shl for u32 { fn shl(self, other: u32) -> u32 { self << other } } -impl Shl for u64 { fn shl(self, other: u64) -> u64 { self << other } } -``` - ---- - -## `std::append` - -### `std::append::Append` - -`Append` can abstract over types that can be appended to - usually container types: - -```rust title="append-trait" showLineNumbers -pub trait Append { - fn empty() -> Self; - fn append(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/append.nr#L9-L14 - - -`Append` requires two methods: - -- `empty`: Constructs an empty value of `Self`. -- `append`: Append two values together, returning the result. - -Additionally, it is expected that for any implementation: - -- `T::empty().append(x) == x` -- `x.append(T::empty()) == x` - -Implementations: -```rust -impl Append for [T] -impl Append for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/zeroed.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/zeroed.md deleted file mode 100644 index f450fecdd36..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/standard_library/zeroed.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Zeroed Function -description: - The zeroed function returns a zeroed value of any type. -keywords: - [ - zeroed - ] ---- - -Implements `fn zeroed() -> T` to return a zeroed value of any type. This function is generally unsafe to use as the zeroed bit pattern is not guaranteed to be valid for all types. It can however, be useful in cases when the value is guaranteed not to be used such as in a BoundedVec library implementing a growable vector, up to a certain length, backed by an array. The array can be initialized with zeroed values which are guaranteed to be inaccessible until the vector is pushed to. Similarly, enumerations in noir can be implemented using this method by providing zeroed values for the unused variants. - -You can access the function at `std::unsafe::zeroed`. - -This function currently supports the following types: - -- Field -- Bool -- Uint -- Array -- Slice -- String -- Tuple -- Function - -Using it on other types could result in unexpected behavior. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md deleted file mode 100644 index 252d72a0b71..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend.md +++ /dev/null @@ -1,138 +0,0 @@ -# BarretenbergBackend - -## Implements - -- [`Backend`](../index.md#backend) -- [`Backend`](../index.md#backend) - -## Constructors - -### new BarretenbergBackend(acirCircuit, options) - -```ts -new BarretenbergBackend(acirCircuit, options): BarretenbergBackend -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `acirCircuit` | `CompiledCircuit` | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`BarretenbergBackend`](BarretenbergBackend.md) - -## Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `backend` | `UltraPlonkBackend` | - | - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### generateProof() - -```ts -generateProof(compressedWitness): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `compressedWitness` | `Uint8Array` | - -#### Returns - -`Promise`\<`ProofData`\> - -#### Description - -Generates a proof - -*** - -### generateRecursiveProofArtifacts() - -```ts -generateRecursiveProofArtifacts(proofData, numOfPublicInputs): Promise -``` - -Generates artifacts that will be passed to a circuit that will verify this proof. - -Instead of passing the proof and verification key as a byte array, we pass them -as fields which makes it cheaper to verify in a circuit. - -The proof that is passed here will have been created using a circuit -that has the #[recursive] attribute on its `main` method. - -The number of public inputs denotes how many public inputs are in the inner proof. - -#### Parameters - -| Parameter | Type | Default value | -| :------ | :------ | :------ | -| `proofData` | `ProofData` | `undefined` | -| `numOfPublicInputs` | `number` | `0` | - -#### Returns - -`Promise`\<`object`\> - -#### Example - -```typescript -const artifacts = await backend.generateRecursiveProofArtifacts(proof, numOfPublicInputs); -``` - -*** - -### getVerificationKey() - -```ts -getVerificationKey(): Promise -``` - -#### Returns - -`Promise`\<`Uint8Array`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md deleted file mode 100644 index 500276ea748..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier.md +++ /dev/null @@ -1,58 +0,0 @@ -# BarretenbergVerifier - -## Constructors - -### new BarretenbergVerifier(options) - -```ts -new BarretenbergVerifier(options): BarretenbergVerifier -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`BarretenbergVerifier`](BarretenbergVerifier.md) - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData, verificationKey): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | -| `verificationKey` | `Uint8Array` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md deleted file mode 100644 index 204aaa18db6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend.md +++ /dev/null @@ -1,114 +0,0 @@ -# UltraHonkBackend - -## Implements - -- [`Backend`](../index.md#backend) -- [`Backend`](../index.md#backend) - -## Constructors - -### new UltraHonkBackend(acirCircuit, options) - -```ts -new UltraHonkBackend(acirCircuit, options): UltraHonkBackend -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `acirCircuit` | `CompiledCircuit` | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`UltraHonkBackend`](UltraHonkBackend.md) - -## Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `backend` | `UltraHonkBackend` | - | - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### generateProof() - -```ts -generateProof(compressedWitness): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `compressedWitness` | `Uint8Array` | - -#### Returns - -`Promise`\<`ProofData`\> - -*** - -### generateRecursiveProofArtifacts() - -```ts -generateRecursiveProofArtifacts(proofData, numOfPublicInputs): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | -| `numOfPublicInputs` | `number` | - -#### Returns - -`Promise`\<`object`\> - -*** - -### getVerificationKey() - -```ts -getVerificationKey(): Promise -``` - -#### Returns - -`Promise`\<`Uint8Array`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | - -#### Returns - -`Promise`\<`boolean`\> - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md deleted file mode 100644 index aee9460153f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier.md +++ /dev/null @@ -1,58 +0,0 @@ -# UltraHonkVerifier - -## Constructors - -### new UltraHonkVerifier(options) - -```ts -new UltraHonkVerifier(options): UltraHonkVerifier -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `options` | [`BackendOptions`](../type-aliases/BackendOptions.md) | - -#### Returns - -[`UltraHonkVerifier`](UltraHonkVerifier.md) - -## Methods - -### destroy() - -```ts -destroy(): Promise -``` - -#### Returns - -`Promise`\<`void`\> - -*** - -### verifyProof() - -```ts -verifyProof(proofData, verificationKey): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `proofData` | `ProofData` | -| `verificationKey` | `Uint8Array` | - -#### Returns - -`Promise`\<`boolean`\> - -#### Description - -Verifies a proof - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/index.md deleted file mode 100644 index 4699e16dee6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/index.md +++ /dev/null @@ -1,42 +0,0 @@ -# backend_barretenberg - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [BarretenbergBackend](classes/BarretenbergBackend.md) | - | -| [BarretenbergVerifier](classes/BarretenbergVerifier.md) | - | -| [UltraHonkBackend](classes/UltraHonkBackend.md) | - | -| [UltraHonkVerifier](classes/UltraHonkVerifier.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [BackendOptions](type-aliases/BackendOptions.md) | - | - -## References - -### CompiledCircuit - -Renames and re-exports [Backend](index.md#backend) - -*** - -### ProofData - -Renames and re-exports [Backend](index.md#backend) - -## Variables - -### Backend - -```ts -Backend: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md deleted file mode 100644 index 99ade2ce6f7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions.md +++ /dev/null @@ -1,18 +0,0 @@ -# BackendOptions - -```ts -type BackendOptions: object; -``` - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `memory` | `object` | - | -| `memory.initial` | `number` | - | -| `memory.maximum` | `number` | - | -| `threads` | `number` | - | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs deleted file mode 100644 index 8ecf05c0163..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/backend_barretenberg/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/BarretenbergBackend","label":"BarretenbergBackend"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/BarretenbergVerifier","label":"BarretenbergVerifier"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/UltraHonkBackend","label":"UltraHonkBackend"},{"type":"doc","id":"reference/NoirJS/backend_barretenberg/classes/UltraHonkVerifier","label":"UltraHonkVerifier"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/backend_barretenberg/type-aliases/BackendOptions","label":"BackendOptions"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/classes/Noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/classes/Noir.md deleted file mode 100644 index ead255bc504..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/classes/Noir.md +++ /dev/null @@ -1,52 +0,0 @@ -# Noir - -## Constructors - -### new Noir(circuit) - -```ts -new Noir(circuit): Noir -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `circuit` | `CompiledCircuit` | - -#### Returns - -[`Noir`](Noir.md) - -## Methods - -### execute() - -```ts -execute(inputs, foreignCallHandler?): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `inputs` | `InputMap` | -| `foreignCallHandler`? | [`ForeignCallHandler`](../type-aliases/ForeignCallHandler.md) | - -#### Returns - -`Promise`\<`object`\> - -#### Description - -Allows to execute a circuit to get its witness and return value. - -#### Example - -```typescript -async execute(inputs) -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/and.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/and.md deleted file mode 100644 index c783283e396..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/and.md +++ /dev/null @@ -1,22 +0,0 @@ -# and() - -```ts -and(lhs, rhs): string -``` - -Performs a bitwise AND operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/blake2s256.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/blake2s256.md deleted file mode 100644 index 7882d0da8d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/blake2s256.md +++ /dev/null @@ -1,21 +0,0 @@ -# blake2s256() - -```ts -blake2s256(inputs): Uint8Array -``` - -Calculates the Blake2s256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md deleted file mode 100644 index 5e3cd53e9d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256k1\_verify() - -```ts -ecdsa_secp256k1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256k1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md deleted file mode 100644 index 0b20ff68957..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256r1\_verify() - -```ts -ecdsa_secp256r1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256r1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/keccak256.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/keccak256.md deleted file mode 100644 index d10f155ce86..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/keccak256.md +++ /dev/null @@ -1,21 +0,0 @@ -# keccak256() - -```ts -keccak256(inputs): Uint8Array -``` - -Calculates the Keccak256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/xor.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/xor.md deleted file mode 100644 index 8d762b895d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/functions/xor.md +++ /dev/null @@ -1,22 +0,0 @@ -# xor() - -```ts -xor(lhs, rhs): string -``` - -Performs a bitwise XOR operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/index.md deleted file mode 100644 index b65a44865ee..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/index.md +++ /dev/null @@ -1,48 +0,0 @@ -# noir_js - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [Noir](classes/Noir.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [ErrorWithPayload](type-aliases/ErrorWithPayload.md) | - | -| [ForeignCallHandler](type-aliases/ForeignCallHandler.md) | A callback which performs an foreign call and returns the response. | -| [ForeignCallInput](type-aliases/ForeignCallInput.md) | - | -| [ForeignCallOutput](type-aliases/ForeignCallOutput.md) | - | -| [WitnessMap](type-aliases/WitnessMap.md) | - | - -### Functions - -| Function | Description | -| :------ | :------ | -| [and](functions/and.md) | Performs a bitwise AND operation between `lhs` and `rhs` | -| [blake2s256](functions/blake2s256.md) | Calculates the Blake2s256 hash of the input bytes | -| [ecdsa\_secp256k1\_verify](functions/ecdsa_secp256k1_verify.md) | Verifies a ECDSA signature over the secp256k1 curve. | -| [ecdsa\_secp256r1\_verify](functions/ecdsa_secp256r1_verify.md) | Verifies a ECDSA signature over the secp256r1 curve. | -| [keccak256](functions/keccak256.md) | Calculates the Keccak256 hash of the input bytes | -| [xor](functions/xor.md) | Performs a bitwise XOR operation between `lhs` and `rhs` | - -## References - -### CompiledCircuit - -Renames and re-exports [InputMap](index.md#inputmap) - -## Variables - -### InputMap - -```ts -InputMap: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md deleted file mode 100644 index e8c2f4aef3d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md +++ /dev/null @@ -1,15 +0,0 @@ -# ErrorWithPayload - -```ts -type ErrorWithPayload: ExecutionError & object; -``` - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `decodedAssertionPayload` | `any` | - | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md deleted file mode 100644 index 812b8b16481..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md +++ /dev/null @@ -1,24 +0,0 @@ -# ForeignCallHandler - -```ts -type ForeignCallHandler: (name, inputs) => Promise; -``` - -A callback which performs an foreign call and returns the response. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | The identifier for the type of foreign call being performed. | -| `inputs` | [`ForeignCallInput`](ForeignCallInput.md)[] | An array of hex encoded inputs to the foreign call. | - -## Returns - -`Promise`\<[`ForeignCallOutput`](ForeignCallOutput.md)[]\> - -outputs - An array of hex encoded outputs containing the results of the foreign call. - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md deleted file mode 100644 index dd95809186a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallInput - -```ts -type ForeignCallInput: string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md deleted file mode 100644 index b71fb78a946..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallOutput - -```ts -type ForeignCallOutput: string | string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md deleted file mode 100644 index 258c46f9d0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md +++ /dev/null @@ -1,9 +0,0 @@ -# WitnessMap - -```ts -type WitnessMap: Map; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs deleted file mode 100644 index 9c7fc4a353f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/noir_js/classes/Noir","label":"Noir"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ErrorWithPayload","label":"ErrorWithPayload"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallHandler","label":"ForeignCallHandler"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallInput","label":"ForeignCallInput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallOutput","label":"ForeignCallOutput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/WitnessMap","label":"WitnessMap"}]},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_js/functions/and","label":"and"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/blake2s256","label":"blake2s256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify","label":"ecdsa_secp256k1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify","label":"ecdsa_secp256r1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/keccak256","label":"keccak256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/xor","label":"xor"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/compile.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/compile.md deleted file mode 100644 index 6faf763b37f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/compile.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile() - -```ts -compile( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ProgramCompilationArtifacts`](../index.md#programcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_program(fm); -``` - -```typescript -// Browser - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_program(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/compile_contract.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/compile_contract.md deleted file mode 100644 index 7d0b39a43ef..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/compile_contract.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile\_contract() - -```ts -compile_contract( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ContractCompilationArtifacts`](../index.md#contractcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_contract(fm); -``` - -```typescript -// Browser - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_contract(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/createFileManager.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/createFileManager.md deleted file mode 100644 index 7e65c1d69c7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/createFileManager.md +++ /dev/null @@ -1,21 +0,0 @@ -# createFileManager() - -```ts -createFileManager(dataDir): FileManager -``` - -Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `dataDir` | `string` | root of the file system | - -## Returns - -`FileManager` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md deleted file mode 100644 index fcea9275341..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md +++ /dev/null @@ -1,21 +0,0 @@ -# inflateDebugSymbols() - -```ts -inflateDebugSymbols(debugSymbols): any -``` - -Decompresses and decodes the debug symbols - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `debugSymbols` | `string` | The base64 encoded debug symbols | - -## Returns - -`any` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/index.md deleted file mode 100644 index b6e0f9d1bc0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_wasm - -## Exports - -### Functions - -| Function | Description | -| :------ | :------ | -| [compile](functions/compile.md) | Compiles a Noir project | -| [compile\_contract](functions/compile_contract.md) | Compiles a Noir project | -| [createFileManager](functions/createFileManager.md) | Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) | -| [inflateDebugSymbols](functions/inflateDebugSymbols.md) | Decompresses and decodes the debug symbols | - -## References - -### compile\_program - -Renames and re-exports [compile](functions/compile.md) - -## Interfaces - -### ContractCompilationArtifacts - -The compilation artifacts of a given contract. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `contract` | `ContractArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -### ProgramCompilationArtifacts - -The compilation artifacts of a given program. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | not part of the compilation output, injected later | -| `program` | `ProgramArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs deleted file mode 100644 index e0870710349..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"doc","id":"reference/NoirJS/noir_wasm/index","label":"API"},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile","label":"compile"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile_contract","label":"compile_contract"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/createFileManager","label":"createFileManager"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/inflateDebugSymbols","label":"inflateDebugSymbols"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/_category_.json deleted file mode 100644 index 5b6a20a609a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 4, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/_category_.json deleted file mode 100644 index 27869205ad3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugger", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_known_limitations.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_known_limitations.md deleted file mode 100644 index 936d416ac4b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_known_limitations.md +++ /dev/null @@ -1,59 +0,0 @@ ---- -title: Known limitations -description: - An overview of known limitations of the current version of the Noir debugger -keywords: - [ - Nargo, - Noir Debugger, - VS Code, - ] -sidebar_position: 2 ---- - -# Debugger Known Limitations - -There are currently some limits to what the debugger can observe. - -## Mutable references - -The debugger is currently blind to any state mutated via a mutable reference. For example, in: - -``` -let mut x = 1; -let y = &mut x; -*y = 2; -``` - -The update on `x` will not be observed by the debugger. That means, when running `vars` from the debugger REPL, or inspecting the _local variables_ pane in the VS Code debugger, `x` will appear with value 1 despite having executed `*y = 2;`. - -## Variables of type function or mutable references are opaque - -When inspecting variables, any variable of type `Function` or `MutableReference` will render its value as `<>` or `<>`. - -## Debugger instrumentation affects resulting ACIR - -In order to make the state of local variables observable, the debugger compiles Noir circuits interleaving foreign calls that track any mutations to them. While this works (except in the cases described above) and doesn't introduce any behavior changes, it does as a side effect produce bigger bytecode. In particular, when running the command `opcodes` on the REPL debugger, you will notice Unconstrained VM blocks that look like this: - -``` -... -5 BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [], q_c: 2 }), Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(2))], q_c: 0 })] - | outputs=[] - 5.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 5.1 | Mov { destination: RegisterIndex(3), source: RegisterIndex(1) } - 5.2 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 5.3 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 5.4 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 5.5 | Mov { destination: RegisterIndex(3), source: RegisterIndex(3) } - 5.6 | Call { location: 8 } - 5.7 | Stop - 5.8 | ForeignCall { function: "__debug_var_assign", destinations: [], inputs: [RegisterIndex(RegisterIndex(2)), RegisterIndex(RegisterIndex(3))] } -... -``` - -If you are interested in debugging/inspecting compiled ACIR without these synthetic changes, you can invoke the REPL debugger with the `--skip-instrumentation` flag or launch the VS Code debugger with the `skipConfiguration` property set to true in its launch configuration. You can find more details about those in the [Debugger REPL reference](debugger_repl.md) and the [VS Code Debugger reference](debugger_vscode.md). - -:::note -Skipping debugger instrumentation means you won't be able to inspect values of local variables. -::: - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_repl.md deleted file mode 100644 index 46e2011304e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_repl.md +++ /dev/null @@ -1,360 +0,0 @@ ---- -title: REPL Debugger -description: - Noir Debugger REPL options and commands. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -## Running the REPL debugger - -`nargo debug [OPTIONS] [WITNESS_NAME]` - -Runs the Noir REPL debugger. If a `WITNESS_NAME` is provided the debugger writes the resulting execution witness to a `WITNESS_NAME` file. - -### Options - -| Option | Description | -| --------------------- | ------------------------------------------------------------ | -| `-p, --prover-name ` | The name of the toml file which contains the inputs for the prover [default: Prover]| -| `--package ` | The name of the package to debug | -| `--print-acir` | Display the ACIR for compiled circuit | -| `--deny-warnings` | Treat all warnings as errors | -| `--silence-warnings` | Suppress warnings | -| `-h, --help` | Print help | - -None of these options are required. - -:::note -Since the debugger starts by compiling the target package, all Noir compiler options are also available. Check out the [compiler reference](../nargo_commands.md#nargo-compile) to learn more about the compiler options. -::: - -## REPL commands - -Once the debugger is running, it accepts the following commands. - -#### `help` (h) - -Displays the menu of available commands. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) value - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -### Stepping through programs - -#### `next` (n) - -Step until the next Noir source code location. While other commands, such as [`into`](#into-i) and [`step`](#step-s), allow for finer grained control of the program's execution at the opcode level, `next` is source code centric. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `next` here would cause the debugger to jump to the definition of `deep_entry_point` (if available). - -If you want to step over `deep_entry_point` and go straight to line 8, use [the `over` command](#over) instead. - -#### `over` - -Step until the next source code location, without diving into function calls. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `over` here would cause the debugger to execute until line 8 (`multiple_values_entry_point(x);`). - -If you want to step into `deep_entry_point` instead, use [the `next` command](#next-n). - -#### `out` - -Step until the end of the current function call. For example: - -``` - 3 ... - 4 fn main(x: u32) { - 5 assert(entry_point(x) == 2); - 6 swap_entry_point(x, x + 1); - 7 -> assert(deep_entry_point(x) == 4); - 8 multiple_values_entry_point(x); - 9 } - 10 - 11 unconstrained fn returns_multiple_values(x: u32) -> (u32, u32, u32, u32) { - 12 ... - ... - 55 - 56 unconstrained fn deep_entry_point(x: u32) -> u32 { - 57 -> level_1(x + 1) - 58 } - -``` - -Running `out` here will resume execution until line 8. - -#### `step` (s) - -Skips to the next ACIR code. A compiled Noir program is a sequence of ACIR opcodes. However, an unconstrained VM opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `step` command at this point would result in the debugger stopping at ACIR opcode 2, `EXPR`, skipping unconstrained computation steps. - -Use [the `into` command](#into-i) instead if you want to follow unconstrained computation step by step. - -#### `into` (i) - -Steps into the next opcode. A compiled Noir program is a sequence of ACIR opcodes. However, a BRILLIG opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `into` command at this point would result in the debugger stopping at opcode 1.0, `Mov ...`, allowing the debugger user to follow unconstrained computation step by step. - -Use [the `step` command](#step-s) instead if you want to skip to the next ACIR code directly. - -#### `continue` (c) - -Continues execution until the next breakpoint, or the end of the program. - -#### `restart` (res) - -Interrupts execution, and restarts a new debugging session from scratch. - -#### `opcodes` (o) - -Display the program's ACIR opcode sequence. For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -### Breakpoints - -#### `break [Opcode]` (or shorthand `b [Opcode]`) - -Sets a breakpoint on the specified opcode index. To get a list of the program opcode numbers, see [the `opcode` command](#opcodes-o). For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -In this example, issuing a `break 1.2` command adds break on opcode 1.2, as denoted by the `*` character: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | * Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -Running [the `continue` command](#continue-c) at this point would cause the debugger to execute the program until opcode 1.2. - -#### `delete [Opcode]` (or shorthand `d [Opcode]`) - -Deletes a breakpoint at an opcode location. Usage is analogous to [the `break` command](#). - -### Variable inspection - -#### vars - -Show variable values available at this point in execution. - -:::note -The ability to inspect variable values from the debugger depends on compilation to be run in a special debug instrumentation mode. This instrumentation weaves variable tracing code with the original source code. - -So variable value inspection comes at the expense of making the resulting ACIR bytecode bigger and harder to understand and optimize. - -If you find this compromise unacceptable, you can run the debugger with the flag `--skip-debug-instrumentation`. This will compile your circuit without any additional debug information, so the resulting ACIR bytecode will be identical to the one produced by standard Noir compilation. However, if you opt for this, the `vars` command will not be available while debugging. -::: - - -### Stacktrace - -#### `stacktrace` - -Displays the current stack trace. - - -### Witness map - -#### `witness` (w) - -Show witness map. For example: - -``` -_0 = 0 -_1 = 2 -_2 = 1 -``` - -#### `witness [Witness Index]` - -Display a single witness from the witness map. For example: - -``` -> witness 1 -_1 = 2 -``` - -#### `witness [Witness Index] [New value]` - -Overwrite the given index with a new value. For example: - -``` -> witness 1 3 -_1 = 3 -``` - - -### Unconstrained VM memory - -#### `memory` - -Show unconstrained VM memory state. For example: - -``` -> memory -At opcode 1.13: Store { destination_pointer: RegisterIndex(0), source: RegisterIndex(3) } -... -> registers -0 = 0 -1 = 10 -2 = 0 -3 = 1 -4 = 1 -5 = 2³² -6 = 1 -> into -At opcode 1.14: Const { destination: RegisterIndex(5), value: Value { inner: 1 } } -... -> memory -0 = 1 -> -``` - -In the example above: we start with clean memory, then step through a `Store` opcode which stores the value of register 3 (1) into the memory address stored in register 0 (0). Thus now `memory` shows memory address 0 contains value 1. - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: - -#### `memset [Memory address] [New value]` - -Update a memory cell with the given value. For example: - -``` -> memory -0 = 1 -> memset 0 2 -> memory -0 = 2 -> memset 1 4 -> memory -0 = 2 -1 = 4 -> -``` - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_vscode.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_vscode.md deleted file mode 100644 index c027332b3b0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/debugger/debugger_vscode.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: VS Code Debugger -description: - VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -# VS Code Noir Debugger Reference - -The Noir debugger enabled by the vscode-noir extension ships with default settings such that the most common scenario should run without any additional configuration steps. - -These defaults can nevertheless be overridden by defining a launch configuration file. This page provides a reference for the properties you can override via a launch configuration file, as well as documenting the Nargo `dap` command, which is a dependency of the VS Code Noir debugger. - - -## Creating and editing launch configuration files - -To create a launch configuration file from VS Code, open the _debug pane_, and click on _create a launch.json file_. - -![Creating a launch configuration file](@site/static/img/debugger/ref1-create-launch.png) - -A `launch.json` file will be created, populated with basic defaults. - -### Noir Debugger launch.json properties - -#### projectFolder - -_String, optional._ - -Absolute path to the Nargo project to debug. By default, it is dynamically determined by looking for the nearest `Nargo.toml` file to the active file at the moment of launching the debugger. - -#### proverName - -_String, optional._ - -Name of the prover input to use. Defaults to `Prover`, which looks for a file named `Prover.toml` at the `projectFolder`. - -#### generateAcir - -_Boolean, optional._ - -If true, generate ACIR opcodes instead of unconstrained opcodes which will be closer to release binaries but less convenient for debugging. Defaults to `false`. - -#### skipInstrumentation - -_Boolean, optional._ - -Skips variables debugging instrumentation of code, making debugging less convenient but the resulting binary smaller and closer to production. Defaults to `false`. - -:::note -Skipping instrumentation causes the debugger to be unable to inspect local variables. -::: - -## `nargo dap [OPTIONS]` - -When run without any option flags, it starts the Nargo Debug Adapter Protocol server, which acts as the debugging backend for the VS Code Noir Debugger. - -All option flags are related to preflight checks. The Debug Adapter Protocol specifies how errors are to be informed from a running DAP server, but it doesn't specify mechanisms to communicate server initialization errors between the DAP server and its client IDE. - -Thus `nargo dap` ships with a _preflight check_ mode. If flag `--preflight-check` and the rest of the `--preflight-*` flags are provided, Nargo will run the same initialization routine except it will not start the DAP server. - -`vscode-noir` will then run `nargo dap` in preflight check mode first before a debugging session starts. If the preflight check ends in error, vscode-noir will present stderr and stdout output from this process through its own Output pane in VS Code. This makes it possible for users to diagnose what pieces of configuration might be wrong or missing in case of initialization errors. - -If the preflight check succeeds, `vscode-noir` proceeds to start the DAP server normally but running `nargo dap` without any additional flags. - -### Options - -| Option | Description | -| --------------------------------------------------------- | --------------------------------------------------------------------------------------------------------- | -| `--preflight-check` | If present, dap runs in preflight check mode. | -| `--preflight-project-folder ` | Absolute path to the project to debug for preflight check. | -| `--preflight-prover-name ` | Name of prover file to use for preflight check | -| `--preflight-generate-acir` | Optional. If present, compile in ACIR mode while running preflight check. | -| `--preflight-skip-instrumentation` | Optional. If present, compile without introducing debug instrumentation while running preflight check. | -| `-h, --help` | Print help. | diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/nargo_commands.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/nargo_commands.md deleted file mode 100644 index db1884afee2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/nargo_commands.md +++ /dev/null @@ -1,297 +0,0 @@ ---- -title: Nargo -description: - Noir CLI Commands for Noir Prover and Verifier to create, execute, prove and verify programs, - generate Solidity verifier smart contract and compile into JSON file containing ACIR - representation and ABI of circuit. -keywords: - [ - Nargo, - Noir CLI, - Noir Prover, - Noir Verifier, - generate Solidity verifier, - compile JSON file, - ACIR representation, - ABI of circuit, - TypeScript, - ] -sidebar_position: 0 ---- - -# Command-Line Help for `nargo` - -This document contains the help content for the `nargo` command-line program. - -**Command Overview:** - -* [`nargo`↴](#nargo) -* [`nargo check`↴](#nargo-check) -* [`nargo fmt`↴](#nargo-fmt) -* [`nargo compile`↴](#nargo-compile) -* [`nargo new`↴](#nargo-new) -* [`nargo init`↴](#nargo-init) -* [`nargo execute`↴](#nargo-execute) -* [`nargo debug`↴](#nargo-debug) -* [`nargo test`↴](#nargo-test) -* [`nargo info`↴](#nargo-info) -* [`nargo lsp`↴](#nargo-lsp) - -## `nargo` - -Noir's package manager - -**Usage:** `nargo ` - -###### **Subcommands:** - -* `check` — Checks the constraint system for errors -* `fmt` — Format the Noir files in a workspace -* `compile` — Compile the program and its secret execution trace into ACIR format -* `new` — Create a Noir project in a new directory -* `init` — Create a Noir project in the current directory -* `execute` — Executes a circuit to calculate its return value -* `debug` — Executes a circuit in debug mode -* `test` — Run the tests for this program -* `info` — Provides detailed information on each of a program's function (represented by a single circuit) -* `lsp` — Starts the Noir LSP server - -###### **Options:** - - - - -## `nargo check` - -Checks the constraint system for errors - -**Usage:** `nargo check [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to check -* `--workspace` — Check all packages in the workspace -* `--overwrite` — Force overwrite of existing files -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo fmt` - -Format the Noir files in a workspace - -**Usage:** `nargo fmt [OPTIONS]` - -###### **Options:** - -* `--check` — Run noirfmt in check mode - - - -## `nargo compile` - -Compile the program and its secret execution trace into ACIR format - -**Usage:** `nargo compile [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to compile -* `--workspace` — Compile all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo new` - -Create a Noir project in a new directory - -**Usage:** `nargo new [OPTIONS] ` - -###### **Arguments:** - -* `` — The path to save the new project - -###### **Options:** - -* `--name ` — Name of the package [default: package directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo init` - -Create a Noir project in the current directory - -**Usage:** `nargo init [OPTIONS]` - -###### **Options:** - -* `--name ` — Name of the package [default: current directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo execute` - -Executes a circuit to calculate its return value - -**Usage:** `nargo execute [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -Defaults to the name of the package being executed. - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--workspace` — Execute all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo debug` - -Executes a circuit in debug mode - -**Usage:** `nargo debug [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--acir-mode` — Force ACIR output (disabling instrumentation) -* `--skip-instrumentation ` — Disable vars debug instrumentation (enabled by default) - - Possible values: `true`, `false` - - - - -## `nargo test` - -Run the tests for this program - -**Usage:** `nargo test [OPTIONS] [TEST_NAME]` - -###### **Arguments:** - -* `` — If given, only tests with names containing this string will be run - -###### **Options:** - -* `--show-output` — Display output of `println` statements -* `--exact` — Only run tests that match exactly -* `--package ` — The name of the package to test -* `--workspace` — Test all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo info` - -Provides detailed information on each of a program's function (represented by a single circuit) - -Current information provided per circuit: 1. The number of ACIR opcodes 2. Counts the final number gates in the circuit used by a backend - -**Usage:** `nargo info [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to detail -* `--workspace` — Detail all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo lsp` - -Starts the Noir LSP server - -Starts an LSP server which allows IDEs such as VS Code to display diagnostics in Noir source. - -VS Code Noir Language Support: https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir - -**Usage:** `nargo lsp` - - - -
- - - This document was generated automatically by - clap-markdown. - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/noir_codegen.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/noir_codegen.md deleted file mode 100644 index db8f07dc22e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/reference/noir_codegen.md +++ /dev/null @@ -1,114 +0,0 @@ ---- -title: Noir Codegen for TypeScript -description: Learn how to use Noir codegen to generate TypeScript bindings -keywords: [Nargo, Noir, compile, TypeScript] -sidebar_position: 3 ---- - -When using TypeScript, it is extra work to interpret Noir program outputs in a type-safe way. Third party libraries may exist for popular Noir programs, but they are either hard to find or unmaintained. - -Now you can generate TypeScript bindings for your Noir programs in two steps: -1. Exporting Noir functions using `nargo export` -2. Using the TypeScript module `noir_codegen` to generate TypeScript binding - -**Note:** you can only export functions from a Noir *library* (not binary or contract program types). - -## Installation - -### Your TypeScript project - -If you don't already have a TypeScript project you can add the module with `yarn` (or `npm`), then initialize it: - -```bash -yarn add typescript -D -npx tsc --init -``` - -### Add TypeScript module - `noir_codegen` - -The following command will add the module to your project's devDependencies: - -```bash -yarn add @noir-lang/noir_codegen -D -``` - -### Nargo library -Make sure you have Nargo, v0.25.0 or greater, installed. If you don't, follow the [installation guide](../getting_started/installation/index.md). - -If you're in a new project, make a `circuits` folder and create a new Noir library: - -```bash -mkdir circuits && cd circuits -nargo new --lib myNoirLib -``` - -## Usage - -### Export ABI of specified functions - -First go to the `.nr` files in your Noir library, and add the `#[export]` macro to each function that you want to use in TypeScript. - -```rust -#[export] -fn your_function(... -``` - -From your Noir library (where `Nargo.toml` is), run the following command: - -```bash -nargo export -``` - -You will now have an `export` directory with a .json file per exported function. - -You can also specify the directory of Noir programs using `--program-dir`, for example: - -```bash -nargo export --program-dir=./circuits/myNoirLib -``` - -### Generate TypeScript bindings from exported functions - -To use the `noir-codegen` package we added to the TypeScript project: - -```bash -yarn noir-codegen ./export/your_function.json -``` - -This creates an `exports` directory with an `index.ts` file containing all exported functions. - -**Note:** adding `--out-dir` allows you to specify an output dir for your TypeScript bindings to go. Eg: - -```bash -yarn noir-codegen ./export/*.json --out-dir ./path/to/output/dir -``` - -## Example .nr function to .ts output - -Consider a Noir library with this function: - -```rust -#[export] -fn not_equal(x: Field, y: Field) -> bool { - x != y -} -``` - -After the export and codegen steps, you should have an `index.ts` like: - -```typescript -export type Field = string; - - -export const is_equal_circuit: CompiledCircuit = -{"abi":{"parameters":[{"name":"x","type":{"kind":"field"},"visibility":"private"},{"name":"y","type":{"kind":"field"},"visibility":"private"}],"return_type":{"abi_type":{"kind":"boolean"},"visibility":"private"}},"bytecode":"H4sIAAAAAAAA/7WUMQ7DIAxFQ0Krrr2JjSGYLVcpKrn/CaqqDQN12WK+hPBgmWd/wEyHbF1SS923uhOs3pfoChI+wKXMAXzIKyNj4PB0TFTYc0w5RUjoqeAeEu1wqK0F54RGkWvW44LPzExnlkbMEs4JNZmN8PxS42uHv82T8a3Jeyn2Ks+VLPcO558HmyLMCDOXAXXtpPt4R/Rt9T36ss6dS9HGPx/eG17nGegKBQAA"}; - -export async function is_equal(x: Field, y: Field, foreignCallHandler?: ForeignCallHandler): Promise { - const program = new Noir(is_equal_circuit); - const args: InputMap = { x, y }; - const { returnValue } = await program.execute(args, foreignCallHandler); - return returnValue as boolean; -} -``` - -Now the `is_equal()` function and relevant types are readily available for use in TypeScript. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/debugger.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/debugger.md deleted file mode 100644 index 9b7565ba9ff..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/debugger.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Debugger -description: Learn about the Noir Debugger, in its REPL or VS Code versions. -keywords: [Nargo, VSCode, Visual Studio Code, REPL, Debugger] -sidebar_position: 2 ---- - -# Noir Debugger - -There are currently two ways of debugging Noir programs: - -1. From VS Code, via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). -2. Via the REPL debugger, which ships with Nargo. - -In order to use either version of the debugger, you will need to install recent enough versions of Noir, [Nargo](../getting_started/installation/index.md) and vscode-noir: - -- Noir & Nargo ≥0.28.0 -- Noir's VS Code extension ≥0.0.11 - -:::info -At the moment, the debugger supports debugging binary projects, but not contracts. -::: - -We cover the VS Code Noir debugger more in depth in [its VS Code debugger how-to guide](../how_to/debugger/debugging_with_vs_code.md) and [the reference](../reference/debugger/debugger_vscode.md). - -The REPL debugger is discussed at length in [the REPL debugger how-to guide](../how_to/debugger/debugging_with_the_repl.md) and [the reference](../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/language_server.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/language_server.md deleted file mode 100644 index 81e0356ef8a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/language_server.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Language Server -description: Learn about the Noir Language Server, how to install the components, and configuration that may be required. -keywords: [Nargo, Language Server, LSP, VSCode, Visual Studio Code] -sidebar_position: 0 ---- - -This section helps you install and configure the Noir Language Server. - -The Language Server Protocol (LSP) has two components, the [Server](#language-server) and the [Client](#language-client). Below we describe each in the context of Noir. - -## Language Server - -The Server component is provided by the Nargo command line tool that you installed at the beginning of this guide. -As long as Nargo is installed and you've used it to run other commands in this guide, it should be good to go! - -If you'd like to verify that the `nargo lsp` command is available, you can run `nargo --help` and look for `lsp` in the list of commands. If you see it, you're using a version of Noir with LSP support. - -## Language Client - -The Client component is usually an editor plugin that launches the Server. It communicates LSP messages between the editor and the Server. For example, when you save a file, the Client will alert the Server, so it can try to compile the project and report any errors. - -Currently, Noir provides a Language Client for Visual Studio Code via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). - -> **Note:** Noir's Language Server Protocol support currently assumes users' VSCode workspace root to be the same as users' Noir project root (i.e. where Nargo.toml lies). -> -> If LSP features seem to be missing / malfunctioning, make sure you are opening your Noir project directly (instead of as a sub-folder) in your VSCode instance. - -When your language server is running correctly and the VSCode plugin is installed, you should see handy codelens buttons for compilation, measuring circuit size, execution, and tests: - -![Compile and Execute](@site/static/img/codelens_compile_execute.png) -![Run test](@site/static/img/codelens_run_test.png) - -You should also see your tests in the `testing` panel: - -![Testing panel](@site/static/img/codelens_testing_panel.png) - -### Configuration - -- **Noir: Enable LSP** - If checked, the extension will launch the Language Server via `nargo lsp` and communicate with it. -- **Noir: Nargo Flags** - Additional flags may be specified if you require them to be added when the extension calls `nargo lsp`. -- **Noir: Nargo Path** - An absolute path to a Nargo binary with the `lsp` command. This may be useful if Nargo is not within the `PATH` of your editor. -- **Noir > Trace: Server** - Setting this to `"messages"` or `"verbose"` will log LSP messages between the Client and Server. Useful for debugging. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/testing.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/testing.md deleted file mode 100644 index 866677da567..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tooling/testing.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Testing in Noir -description: Learn how to use Nargo to test your Noir program in a quick and easy way -keywords: [Nargo, testing, Noir, compile, test] -sidebar_position: 1 ---- - -You can test your Noir programs using Noir circuits. - -Nargo will automatically compile and run any functions which have the decorator `#[test]` on them if -you run `nargo test`. - -For example if you have a program like: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test] -fn test_add() { - assert(add(2,2) == 4); - assert(add(0,1) == 1); - assert(add(1,0) == 1); -} -``` - -Running `nargo test` will test that the `test_add` function can be executed while satisfying all -the constraints which allows you to test that add returns the expected values. Test functions can't -have any arguments currently. - -### Test fail - -You can write tests that are expected to fail by using the decorator `#[test(should_fail)]`. For example: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test(should_fail)] -fn test_add() { - assert(add(2,2) == 5); -} -``` - -You can be more specific and make it fail with a specific reason by using `should_fail_with = ""`: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "What is the airspeed velocity of an unladen swallow")] -fn test_bridgekeeper() { - main(32); -} -``` - -The string given to `should_fail_with` doesn't need to exactly match the failure reason, it just needs to be a substring of it: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "airspeed velocity")] -fn test_bridgekeeper() { - main(32); -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tutorials/noirjs_app.md deleted file mode 100644 index eac28168445..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/tutorials/noirjs_app.md +++ /dev/null @@ -1,362 +0,0 @@ ---- -title: Building a web app with NoirJS -description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. -keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] -sidebar_position: 0 -pagination_next: noir/concepts/data_types/index ---- - -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! - -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/installation/index.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! - -### Nargo - -Run: - -```bash -nargo new circuit -``` - -And... That's about it. Your program is ready to be compiled and run. - -To compile, let's `cd` into the `circuit` folder to enter our project, and call: - -```bash -nargo compile -``` - -This compiles our circuit into `json` format and add it to a new `target` folder. - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` - -::: - -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/hello_noir/index.md) guide. However, we want our app to run on the browser, so we need Vite. - -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. - -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". - -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: - -```bash -cd vite-project -``` - -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; - -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); -``` - -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: - -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev -``` - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... -``` - -::: - -#### Some cleanup - -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. - -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) - -## HTML - -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: - -```html - - - - - - -

Noir app

-
- - -
-
-

Logs

-

Proof

-
- - -``` - -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). - -Start by pasting in this boilerplate code: - -```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} - -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } -}); -``` - -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html -``` - -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - -::: - -## Some NoirJS - -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: - -```ts -import circuit from '../circuit/target/circuit.json'; -``` - -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: - -```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -import { Noir } from '@noir-lang/noir_js'; -``` - -And instantiate them inside our try-catch block: - -```ts -// try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); -// } -``` - -:::note - -For the remainder of the tutorial, everything will be happening inside the `try` block - -::: - -## Our app - -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: - -```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; -``` - -Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: - -```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); -const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); -``` - -You're probably eager to see stuff happening, so go and run your app now! - -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. - -![Getting Started 0](@site/static/img/noir_getting_started_1.png) - -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! - -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. - -## Verifying - -Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: - -```js -display('logs', 'Verifying proof... ⌛'); -const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); -``` - -You have successfully generated a client-side Noir web app! - -![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) - -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. - -## UltraHonk Backend - -Barretenberg has recently exposed a new UltraHonk backend. We can use UltraHonk in NoirJS after version 0.33.0. Everything will be the same as the tutorial above, except that the class we need to import will change: -```js -import { UltraHonkBackend, UltraHonkVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -``` -The backend will then be instantiated as such: -```js -const backend = new UltraHonkBackend(circuit); -``` -Then all the commands to prove and verify your circuit will be same. - -The only feature currently unsupported with UltraHonk are [recursive proofs](../explainers/explainer-recursion.md). \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/cspell.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/cspell.json deleted file mode 100644 index c60b0a597b1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/cspell.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "words": [ - "Cryptdoku" - ] -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/explainer-oracle.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/explainer-oracle.md deleted file mode 100644 index 821e1f95c04..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/explainer-oracle.md +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: Oracles -description: This guide provides an in-depth understanding of how Oracles work in Noir programming. Learn how to use outside calculations in your programs, constrain oracles, and understand their uses and limitations. -keywords: - - Noir Programming - - Oracles - - JSON-RPC - - Foreign Call Handlers - - Constrained Functions - - Blockchain Programming -sidebar_position: 1 ---- - -If you've seen "The Matrix" you may recall "The Oracle" as Gloria Foster smoking cigarettes and baking cookies. While she appears to "know things", she is actually providing a calculation of a pre-determined future. Noir Oracles are similar, in a way. They don't calculate the future (yet), but they allow you to use outside calculations in your programs. - -![matrix oracle prediction](@site/static/img/memes/matrix_oracle.jpeg) - -A Noir program is usually self-contained. You can pass certain inputs to it, and it will generate a deterministic output for those inputs. But what if you wanted to defer some calculation to an outside process or source? - -Oracles are functions that provide this feature. - -## Use cases - -An example usage for Oracles is proving something on-chain. For example, proving that the ETH-USDC quote was below a certain target at a certain block time. Or even making more complex proofs like proving the ownership of an NFT as an anonymous login method. - -Another interesting use case is to defer expensive calculations to be made outside of the Noir program, and then constraining the result; similar to the use of [unconstrained functions](../noir/concepts//unconstrained.md). - -In short, anything that can be constrained in a Noir program but needs to be fetched from an external source is a great candidate to be used in oracles. - -## Constraining oracles - -Just like in The Matrix, Oracles are powerful. But with great power, comes great responsibility. Just because you're using them in a Noir program doesn't mean they're true. Noir has no superpowers. If you want to prove that Portugal won the Euro Cup 2016, you're still relying on potentially untrusted information. - -To give a concrete example, Alice wants to login to the [NounsDAO](https://nouns.wtf/) forum with her username "noir_nouner" by proving she owns a noun without revealing her ethereum address. Her Noir program could have an oracle call like this: - -```rust -#[oracle(getNoun)] -unconstrained fn get_noun(address: Field) -> Field -``` - -This oracle could naively resolve with the number of Nouns she possesses. However, it is useless as a trusted source, as the oracle could resolve to anything Alice wants. In order to make this oracle call actually useful, Alice would need to constrain the response from the oracle, by proving her address and the noun count belongs to the state tree of the contract. - -In short, **Oracles don't prove anything. Your Noir program does.** - -:::danger - -If you don't constrain the return of your oracle, you could be clearly opening an attack vector on your Noir program. Make double-triple sure that the return of an oracle call is constrained! - -::: - -## How to use Oracles - -On CLI, Nargo resolves oracles by making JSON RPC calls, which means it would require an RPC node to be running. - -In JavaScript, NoirJS accepts and resolves arbitrary call handlers (that is, not limited to JSON) as long as they match the expected types the developer defines. Refer to [Foreign Call Handler](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) to learn more about NoirJS's call handling. - -If you want to build using oracles, follow through to the [oracle guide](../how_to/how-to-oracles.md) for a simple example on how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/explainer-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/explainer-recursion.md deleted file mode 100644 index df8529ef4e0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/explainer-recursion.md +++ /dev/null @@ -1,176 +0,0 @@ ---- -title: Recursive proofs -description: Explore the concept of recursive proofs in Zero-Knowledge programming. Understand how recursion works in Noir, a language for writing smart contracts on the EVM blockchain. Learn through practical examples like Alice and Bob's guessing game, Charlie's recursive merkle tree, and Daniel's reusable components. Discover how to use recursive proofs to optimize computational resources and improve efficiency. - -keywords: - [ - "Recursive Proofs", - "Zero-Knowledge Programming", - "Noir", - "EVM Blockchain", - "Smart Contracts", - "Recursion in Noir", - "Alice and Bob Guessing Game", - "Recursive Merkle Tree", - "Reusable Components", - "Optimizing Computational Resources", - "Improving Efficiency", - "Verification Key", - "Aggregation", - "Recursive zkSNARK schemes", - "PLONK", - "Proving and Verification Keys" - ] -sidebar_position: 1 -pagination_next: how_to/how-to-recursion ---- - -In programming, we tend to think of recursion as something calling itself. A classic example would be the calculation of the factorial of a number: - -```js -function factorial(n) { - if (n === 0 || n === 1) { - return 1; - } else { - return n * factorial(n - 1); - } -} -``` - -In this case, while `n` is not `1`, this function will keep calling itself until it hits the base case, bubbling up the result on the call stack: - -```md - Is `n` 1? <--------- - /\ / - / \ n = n -1 - / \ / - Yes No -------- -``` - -In Zero-Knowledge, recursion has some similarities. - -It is not a Noir function calling itself, but a proof being used as an input to another circuit. In short, you verify one proof *inside* another proof, returning the proof that both proofs are valid. - -This means that, given enough computational resources, you can prove the correctness of any arbitrary number of proofs in a single proof. This could be useful to design state channels (for which a common example would be [Bitcoin's Lightning Network](https://en.wikipedia.org/wiki/Lightning_Network)), to save on gas costs by settling one proof on-chain, or simply to make business logic less dependent on a consensus mechanism. - -## Examples - -Let us look at some of these examples - -### Alice and Bob - Guessing game - -Alice and Bob are friends, and they like guessing games. They want to play a guessing game online, but for that, they need a trusted third-party that knows both of their secrets and finishes the game once someone wins. - -So, they use zero-knowledge proofs. Alice tries to guess Bob's number, and Bob will generate a ZK proof stating whether she succeeded or failed. - -This ZK proof can go on a smart contract, revealing the winner and even giving prizes. However, this means every turn needs to be verified on-chain. This incurs some cost and waiting time that may simply make the game too expensive or time-consuming to be worth it. - -As a solution, Alice proposes the following: "what if Bob generates his proof, and instead of sending it on-chain, I verify it *within* my own proof before playing my own turn?". - -She can then generate a proof that she verified his proof, and so on. - -```md - Did you fail? <-------------------------- - / \ / - / \ n = n -1 - / \ / - Yes No / - | | / - | | / - | You win / - | / - | / -Generate proof of that / - + / - my own guess ---------------- -``` - -### Charlie - Recursive merkle tree - -Charlie is a concerned citizen, and wants to be sure his vote in an election is accounted for. He votes with a ZK proof, but he has no way of knowing that his ZK proof was included in the total vote count! - -If the vote collector puts all of the votes into a [Merkle tree](https://en.wikipedia.org/wiki/Merkle_tree), everyone can prove the verification of two proofs within one proof, as such: - -```md - abcd - __________|______________ - | | - ab cd - _____|_____ ______|______ - | | | | - alice bob charlie daniel -``` - -Doing this recursively allows us to arrive on a final proof `abcd` which if true, verifies the correctness of all the votes. - -### Daniel - Reusable components - -Daniel has a big circuit and a big headache. A part of his circuit is a setup phase that finishes with some assertions that need to be made. But that section alone takes most of the proving time, and is largely independent of the rest of the circuit. - -He might find it more efficient to generate a proof for that setup phase separately, and verify that proof recursively in the actual business logic section of his circuit. This will allow for parallelization of both proofs, which results in a considerable speedup. - -## What params do I need - -As you can see in the [recursion reference](noir/standard_library/recursion.mdx), a simple recursive proof requires: - -- The proof to verify -- The Verification Key of the circuit that generated the proof -- A hash of this verification key, as it's needed for some backends -- The public inputs for the proof - -:::info - -Recursive zkSNARK schemes do not necessarily "verify a proof" in the sense that you expect a true or false to be spit out by the verifier. Rather an aggregation object is built over the public inputs. - -So, taking the example of Alice and Bob and their guessing game: - -- Alice makes her guess. Her proof is *not* recursive: it doesn't verify any proof within it! It's just a standard `assert(x != y)` circuit -- Bob verifies Alice's proof and makes his own guess. In this circuit, he doesn't exactly *prove* the verification of Alice's proof. Instead, he *aggregates* his proof to Alice's proof. The actual verification is done when the full proof is verified, for example when using `nargo verify` or through the verifier smart contract. - -We can imagine recursive proofs a [relay race](https://en.wikipedia.org/wiki/Relay_race). The first runner doesn't have to receive the baton from anyone else, as he/she already starts with it. But when his/her turn is over, the next runner needs to receive it, run a bit more, and pass it along. Even though every runner could theoretically verify the baton mid-run (why not? 🏃🔍), only at the end of the race does the referee verify that the whole race is valid. - -::: - -## Some architecture - -As with everything in computer science, there's no one-size-fits all. But there are some patterns that could help understanding and implementing them. To give three examples: - -### Adding some logic to a proof verification - -This would be an approach for something like our guessing game, where proofs are sent back and forth and are verified by each opponent. This circuit would be divided in two sections: - -- A `recursive verification` section, which would be just the call to `std::verify_proof`, and that would be skipped on the first move (since there's no proof to verify) -- A `guessing` section, which is basically the logic part where the actual guessing happens - -In such a situation, and assuming Alice is first, she would skip the first part and try to guess Bob's number. Bob would then verify her proof on the first section of his run, and try to guess Alice's number on the second part, and so on. - -### Aggregating proofs - -In some one-way interaction situations, recursion would allow for aggregation of simple proofs that don't need to be immediately verified on-chain or elsewhere. - -To give a practical example, a barman wouldn't need to verify a "proof-of-age" on-chain every time he serves alcohol to a customer. Instead, the architecture would comprise two circuits: - -- A `main`, non-recursive circuit with some logic -- A `recursive` circuit meant to verify two proofs in one proof - -The customer's proofs would be intermediate, and made on their phones, and the barman could just verify them locally. He would then aggregate them into a final proof sent on-chain (or elsewhere) at the end of the day. - -### Recursively verifying different circuits - -Nothing prevents you from verifying different circuits in a recursive proof, for example: - -- A `circuit1` circuit -- A `circuit2` circuit -- A `recursive` circuit - -In this example, a regulator could verify that taxes were paid for a specific purchase by aggregating both a `payer` circuit (proving that a purchase was made and taxes were paid), and a `receipt` circuit (proving that the payment was received) - -## How fast is it - -At the time of writing, verifying recursive proofs is surprisingly fast. This is because most of the time is spent on generating the verification key that will be used to generate the next proof. So you are able to cache the verification key and reuse it later. - -Currently, Noir JS packages don't expose the functionality of loading proving and verification keys, but that feature exists in the underlying `bb.js` package. - -## How can I try it - -Learn more about using recursion in Nargo and NoirJS in the [how-to guide](../how_to/how-to-recursion.md) and see a full example in [noir-examples](https://github.com/noir-lang/noir-examples). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/getting_started/noir_installation.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/getting_started/noir_installation.md deleted file mode 100644 index f92fd8dea38..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/getting_started/noir_installation.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Standalone Noir Installation -description: There are different ways to install Nargo, the one-stop shop and command-line tool for developing Noir programs. This guide explains how to specify which version to install when using noirup, and using WSL for windows. -keywords: [ - Installation - Nargo - Noirup - Binaries - Compiling from Source - WSL for Windows - macOS - Linux - Nix - Direnv - Uninstalling Nargo - ] -sidebar_position: 2 ---- - -Noirup is the endorsed method for installing Nargo, streamlining the process of fetching binaries or compiling from source. It supports a range of options to cater to your specific needs, from nightly builds and specific versions to compiling from various sources. - -### Installing Noirup - -First, ensure you have `noirup` installed: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -### Fetching Binaries - -With `noirup`, you can easily switch between different Nargo versions, including nightly builds: - -- **Nightly Version**: Install the latest nightly build. - - ```sh - noirup --version nightly - ``` - -- **Specific Version**: Install a specific version of Nargo. - - ```sh - noirup --version - ``` - -### Compiling from Source - -`noirup` also enables compiling Nargo from various sources: - -- **From a Specific Branch**: Install from the latest commit on a branch. - - ```sh - noirup --branch - ``` - -- **From a Fork**: Install from the main branch of a fork. - - ```sh - noirup --repo - ``` - -- **From a Specific Branch in a Fork**: Install from a specific branch in a fork. - - ```sh - noirup --repo --branch - ``` - -- **From a Specific Pull Request**: Install from a specific PR. - - ```sh - noirup --pr - ``` - -- **From a Specific Commit**: Install from a specific commit. - - ```sh - noirup -C - ``` - -- **From Local Source**: Compile and install from a local directory. - - ```sh - noirup --path ./path/to/local/source - ``` - -## Installation on Windows - -The default backend for Noir (Barretenberg) doesn't provide Windows binaries at this time. For that reason, Noir cannot be installed natively. However, it is available by using Windows Subsystem for Linux (WSL). - -Step 1: Follow the instructions [here](https://learn.microsoft.com/en-us/windows/wsl/install) to install and run WSL. - -step 2: Follow the [Noirup instructions](#installing-noirup). - -## Uninstalling Nargo - -If you installed Nargo with `noirup`, you can uninstall Nargo by removing the files in `~/.nargo`, `~/nargo`, and `~/noir_cache`. This ensures that all installed binaries, configurations, and cache related to Nargo are fully removed from your system. - -```bash -rm -r ~/.nargo -rm -r ~/nargo -rm -r ~/noir_cache -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/getting_started/quick_start.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/getting_started/quick_start.md deleted file mode 100644 index 73ef62f6523..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/getting_started/quick_start.md +++ /dev/null @@ -1,124 +0,0 @@ ---- -title: Quick Start -tags: [] -sidebar_position: 0 ---- - -## Installation - -### Noir - -The easiest way to develop with Noir is using Nargo the CLI tool. It provides you the ability to start new projects, compile, execute and test Noir programs from the terminal. - -You can use `noirup` the installation script to quickly install and update Nargo: - -```bash -curl -L https://raw.githubusercontent.com/noir-lang/noirup/refs/heads/main/install | bash -noirup -``` - -### Proving backend - -After installing Noir, we install a proving backend to work with our Noir programs. - -Proving backends provide you the abilities to generate proofs, verify proofs, generate smart contracts and more for your Noir programs. - -Different proving backends provide different tools for working with Noir programs, here we will use the [Barretenberg proving backend](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg) developed by Aztec Labs as an example. - -You can use the `bbup` installation script to quickly install and update BB, Barretenberg's CLI tool: - -You can find the full list of proving backends compatible with Noir in Awesome Noir. - -```bash -curl -L https://raw.githubusercontent.com/AztecProtocol/aztec-packages/refs/heads/master/barretenberg/bbup/install | bash -bbup -``` - -For the full list of proving backends compatible with Noir, visit [Awesome Noir](https://github.com/noir-lang/awesome-noir/?tab=readme-ov-file#proving-backends). - -## Nargo - -Nargo provides the ability to initiate and execute Noir projects. Let's initialize the traditional `hello_world`: - -```sh -nargo new hello_world -``` - -Two files will be created. - -- `src/main.nr` contains a simple boilerplate circuit -- `Nargo.toml` contains environmental options, such as name, author, dependencies, and others. - -Glancing at _main.nr_ , we can see that inputs in Noir are private by default, but can be labeled public using the keyword `pub`. This means that we will _assert_ that we know a value `x` which is different from `y` without revealing `x`: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -To learn more about private and public values, check the [Data Types](../noir/concepts/data_types/index.md) section. - -### Compiling and executing - -We can now use `nargo` to generate a _Prover.toml_ file, where our input values will be specified: - -```sh -cd hello_world -nargo check - -Let's feed some valid values into this file: - -```toml -x = "1" -y = "2" -``` - -We're now ready to compile and execute our Noir program. By default the `nargo execute` command will do both, and generate the `witness` that we need to feed to our proving backend: - -```sh -nargo execute -``` - -The witness corresponding to this execution will then be written to the file _./target/witness-name.gz_. - -The command also automatically compiles your Noir program if it was not already / was edited, which you may notice the compiled artifacts being written to the file _./target/hello_world.json_. - -With circuit compiled and witness generated, we're ready to prove. - -## Proving backend - -Different proving backends may provide different tools and commands to work with Noir programs. Here Barretenberg's `bb` CLI tool is used as an example: - -```sh -bb prove -b ./target/hello_world.json -w ./target/hello_world.gz -o ./target/proof -``` - -:::tip - -Naming can be confusing, specially as you pass them to the `bb` commands. If unsure, it won't hurt to delete the target folder and start anew to make sure you're using the most recent versions of the compiled circuit and witness. - -::: - -The proof is now generated in the `target` folder. To verify it we first need to compute the verification key from the compiled circuit, and use it to verify: - -```sh -bb write_vk -b ./target/hello_world.json -o ./target/vk -bb verify -k ./target/vk -p ./target/proof -``` - -:::info - -Notice that in order to verify a proof, the verifier knows nothing but the circuit, which is compiled and used to generate the verification key. This is obviously quite important: private inputs remain private. - -As for the public inputs, you may have noticed they haven't been specified. This behavior varies with each particular backend, but barretenberg typically attaches them to the proof. You can see them by parsing and splitting it. For example for if your public inputs are 32 bytes: - -```bash -head -c 32 ./target/proof | od -An -v -t x1 | tr -d $' \n' -``` - -::: - -Congratulations, you have now created and verified a proof for your very first Noir program! - -In the [next section](./project_breakdown.md), we will go into more detail on each step performed. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/_category_.json deleted file mode 100644 index 23b560f610b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/_category_.json deleted file mode 100644 index cc2cbb1c253..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugging", - "position": 5, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/debugging_with_the_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/debugging_with_the_repl.md deleted file mode 100644 index 1d64dae3f37..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/debugging_with_the_repl.md +++ /dev/null @@ -1,164 +0,0 @@ ---- -title: Using the REPL Debugger -description: - Step-by-step guide on how to debug your Noir circuits with the REPL Debugger. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -#### Pre-requisites - -In order to use the REPL debugger, first you need to install recent enough versions of Nargo and vscode-noir. - -## Debugging a simple circuit - -Let's debug a simple circuit: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -To start the REPL debugger, using a terminal, go to a Noir circuit's home directory. Then: - -`$ nargo debug` - -You should be seeing this in your terminal: - -``` -[main] Starting debugger -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:9 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> -``` - -The debugger displays the current Noir code location, and it is now waiting for us to drive it. - -Let's first take a look at the available commands. For that we'll use the `help` command. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -Some commands operate only for unconstrained functions, such as `memory` and `memset`. If you try to use them while execution is paused at an ACIR opcode, the debugger will simply inform you that you are not executing unconstrained code: - -``` -> memory -Unconstrained VM memory not available -> -``` - -Before continuing, we can take a look at the initial witness map: - -``` -> witness -_0 = 1 -_1 = 2 -> -``` - -Cool, since `x==1`, `y==2`, and we want to check that `x != y`, our circuit should succeed. At this point we could intervene and use the witness setter command to change one of the witnesses. Let's set `y=3`, then back to 2, so we don't affect the expected result: - -``` -> witness -_0 = 1 -_1 = 2 -> witness 1 3 -_1 = 3 -> witness -_0 = 1 -_1 = 3 -> witness 1 2 -_1 = 2 -> witness -_0 = 1 -_1 = 2 -> -``` - -Now we can inspect the current state of local variables. For that we use the `vars` command. - -``` -> vars -> -``` - -We currently have no vars in context, since we are at the entry point of the program. Let's use `next` to execute until the next point in the program. - -``` -> vars -> next -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:20 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> vars -x:Field = 0x01 -``` - -As a result of stepping, the variable `x`, whose initial value comes from the witness map, is now in context and returned by `vars`. - -``` -> next - 1 fn main(x : Field, y : pub Field) { - 2 -> assert(x != y); - 3 } -> vars -y:Field = 0x02 -x:Field = 0x01 -``` - -Stepping again we can finally see both variables and their values. And now we can see that the next assertion should succeed. - -Let's continue to the end: - -``` -> continue -(Continuing execution...) -Finished execution -> q -[main] Circuit witness successfully solved -``` - -Upon quitting the debugger after a solved circuit, the resulting circuit witness gets saved, equivalent to what would happen if we had run the same circuit with `nargo execute`. - -We just went through the basics of debugging using Noir REPL debugger. For a comprehensive reference, check out [the reference page](../../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/debugging_with_vs_code.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/debugging_with_vs_code.md deleted file mode 100644 index a5858c1a5eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/debugger/debugging_with_vs_code.md +++ /dev/null @@ -1,68 +0,0 @@ ---- -title: Using the VS Code Debugger -description: - Step by step guide on how to debug your Noir circuits with the VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -This guide will show you how to use VS Code with the vscode-noir extension to debug a Noir project. - -#### Pre-requisites - -- Nargo -- vscode-noir -- A Noir project with a `Nargo.toml`, `Prover.toml` and at least one Noir (`.nr`) containing an entry point function (typically `main`). - -## Running the debugger - -The easiest way to start debugging is to open the file you want to debug, and press `F5`. This will cause the debugger to launch, using your `Prover.toml` file as input. - -You should see something like this: - -![Debugger launched](@site/static/img/debugger/1-started.png) - -Let's inspect the state of the program. For that, we open VS Code's _Debug pane_. Look for this icon: - -![Debug pane icon](@site/static/img/debugger/2-icon.png) - -You will now see two categories of variables: Locals and Witness Map. - -![Debug pane expanded](@site/static/img/debugger/3-debug-pane.png) - -1. **Locals**: variables of your program. At this point in execution this section is empty, but as we step through the code it will get populated by `x`, `result`, `digest`, etc. - -2. **Witness map**: these are initially populated from your project's `Prover.toml` file. In this example, they will be used to populate `x` and `result` at the beginning of the `main` function. - -Most of the time you will probably be focusing mostly on locals, as they represent the high level state of your program. - -You might be interested in inspecting the witness map in case you are trying to solve a really low level issue in the compiler or runtime itself, so this concerns mostly advanced or niche users. - -Let's step through the program, by using the debugger buttons or their corresponding keyboard shortcuts. - -![Debugger buttons](@site/static/img/debugger/4-debugger-buttons.png) - -Now we can see in the variables pane that there's values for `digest`, `result` and `x`. - -![Inspecting locals](@site/static/img/debugger/5-assert.png) - -We can also inspect the values of variables by directly hovering on them on the code. - -![Hover locals](@site/static/img/debugger/6-hover.png) - -Let's set a break point at the `keccak256` function, so we can continue execution up to the point when it's first invoked without having to go one step at a time. - -We just need to click the to the right of the line number 18. Once the breakpoint appears, we can click the `continue` button or use its corresponding keyboard shortcut (`F5` by default). - -![Breakpoint](@site/static/img/debugger/7-break.png) - -Now we are debugging the `keccak256` function, notice the _Call Stack pane_ at the lower right. This lets us inspect the current call stack of our process. - -That covers most of the current debugger functionalities. Check out [the reference](../../reference/debugger/debugger_vscode.md) for more details on how to configure the debugger. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-oracles.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-oracles.md deleted file mode 100644 index 08ce6ee5476..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-oracles.md +++ /dev/null @@ -1,275 +0,0 @@ ---- -title: How to use Oracles -description: Learn how to use oracles in your Noir program with examples in both Nargo and NoirJS. This guide also covers writing a JSON RPC server and providing custom foreign call handlers for NoirJS. -keywords: - - Noir Programming - - Oracles - - Nargo - - NoirJS - - JSON RPC Server - - Foreign Call Handlers -sidebar_position: 1 ---- - -This guide shows you how to use oracles in your Noir program. For the sake of clarity, it assumes that: - -- You have read the [explainer on Oracles](../explainers/explainer-oracle.md) and are comfortable with the concept. -- You have a Noir program to add oracles to. You can create one using the [vite-hardhat starter](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) as a boilerplate. -- You understand the concept of a JSON-RPC server. Visit the [JSON-RPC website](https://www.jsonrpc.org/) if you need a refresher. -- You are comfortable with server-side JavaScript (e.g. Node.js, managing packages, etc.). - -## Rundown - -This guide has 3 major steps: - -1. How to modify our Noir program to make use of oracle calls as unconstrained functions -2. How to write a JSON RPC Server to resolve these oracle calls with Nargo -3. How to use them in Nargo and how to provide a custom resolver in NoirJS - -## Step 1 - Modify your Noir program - -An oracle is defined in a Noir program by defining two methods: - -- An unconstrained method - This tells the compiler that it is executing an [unconstrained functions](../noir/concepts//unconstrained.md). -- A decorated oracle method - This tells the compiler that this method is an RPC call. - -An example of an oracle that returns a `Field` would be: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(number: Field) -> Field { } - -unconstrained fn get_sqrt(number: Field) -> Field { - sqrt(number) -} -``` - -In this example, we're wrapping our oracle function in an unconstrained method, and decorating it with `oracle(getSqrt)`. We can then call the unconstrained function as we would call any other function: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); -} -``` - -In the next section, we will make this `getSqrt` (defined on the `sqrt` decorator) be a method of the RPC server Noir will use. - -:::danger - -As explained in the [Oracle Explainer](../explainers/explainer-oracle.md), this `main` function is unsafe unless you constrain its return value. For example: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); - assert(sqrt.pow_32(2) as u64 == input as u64); // <---- constrain the return of an oracle! -} -``` - -::: - -:::info - -Currently, oracles only work with single params or array params. For example: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt([Field; 2]) -> [Field; 2] { } -``` - -::: - -## Step 2 - Write an RPC server - -Brillig will call *one* RPC server. Most likely you will have to write your own, and you can do it in whatever language you prefer. In this guide, we will do it in Javascript. - -Let's use the above example of an oracle that consumes an array with two `Field` and returns their square roots: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(input: [Field; 2]) -> [Field; 2] { } - -unconstrained fn get_sqrt(input: [Field; 2]) -> [Field; 2] { - sqrt(input) -} - -fn main(input: [Field; 2]) { - let sqrt = get_sqrt(input); - assert(sqrt[0].pow_32(2) as u64 == input[0] as u64); - assert(sqrt[1].pow_32(2) as u64 == input[1] as u64); -} - -#[test] -fn test() { - let input = [4, 16]; - main(input); -} -``` - -:::info - -Why square root? - -In general, computing square roots is computationally more expensive than multiplications, which takes a toll when speaking about ZK applications. In this case, instead of calculating the square root in Noir, we are using our oracle to offload that computation to be made in plain. In our circuit we can simply multiply the two values. - -::: - -Now, we should write the correspondent RPC server, starting with the [default JSON-RPC 2.0 boilerplate](https://www.npmjs.com/package/json-rpc-2.0#example): - -```js -import { JSONRPCServer } from "json-rpc-2.0"; -import express from "express"; -import bodyParser from "body-parser"; - -const app = express(); -app.use(bodyParser.json()); - -const server = new JSONRPCServer(); -app.post("/", (req, res) => { - const jsonRPCRequest = req.body; - server.receive(jsonRPCRequest).then((jsonRPCResponse) => { - if (jsonRPCResponse) { - res.json(jsonRPCResponse); - } else { - res.sendStatus(204); - } - }); -}); - -app.listen(5555); -``` - -Now, we will add our `getSqrt` method, as expected by the `#[oracle(getSqrt)]` decorator in our Noir code. It maps through the params array and returns their square roots: - -```js -server.addMethod("resolve_foreign_call", async (params) => { - if (params[0].function !== "getSqrt") { - throw Error("Unexpected foreign call") - }; - const values = params[0].inputs[0].map((field) => { - return `${Math.sqrt(parseInt(field, 16))}`; - }); - return { values: [values] }; -}); -``` - -If you're using Typescript, the following types may be helpful in understanding the expected return value and making sure they're easy to follow: - -```js -export type ForeignCallSingle = string; - -export type ForeignCallArray = string[]; - -export type ForeignCallResult = { - values: (ForeignCallSingle | ForeignCallArray)[]; -}; -``` - -:::info Multidimensional Arrays - -If the Oracle function is returning an array containing other arrays, such as `[['1','2],['3','4']]`, you need to provide the values in JSON as flattened values. In the previous example, it would be `['1', '2', '3', '4']`. In the Noir program, the Oracle signature can use a nested type, the flattened values will be automatically converted to the nested type. - -::: - -## Step 3 - Usage with Nargo - -Using the [`nargo` CLI tool](../reference/nargo_commands.md), you can use oracles in the `nargo test` and `nargo execute` commands by passing a value to `--oracle-resolver`. For example: - -```bash -nargo test --oracle-resolver http://localhost:5555 -``` - -This tells `nargo` to use your RPC Server URL whenever it finds an oracle decorator. - -## Step 4 - Usage with NoirJS - -In a JS environment, an RPC server is not strictly necessary, as you may want to resolve your oracles without needing any JSON call at all. NoirJS simply expects that you pass a callback function when you generate proofs, and that callback function can be anything. - -For example, if your Noir program expects the host machine to provide CPU pseudo-randomness, you could simply pass it as the `foreignCallHandler`. You don't strictly need to create an RPC server to serve pseudo-randomness, as you may as well get it directly in your app: - -```js -const foreignCallHandler = (name, inputs) => crypto.randomBytes(16) // etc - -await noir.execute(inputs, foreignCallHandler) -``` - -As one can see, in NoirJS, the [`foreignCallHandler`](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) function simply means "a callback function that returns a value of type [`ForeignCallOutput`](../reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md). It doesn't have to be an RPC call like in the case for Nargo. - -:::tip - -Does this mean you don't have to write an RPC server like in [Step #2](#step-2---write-an-rpc-server)? - -You don't technically have to, but then how would you run `nargo test`? To use both `Nargo` and `NoirJS` in your development flow, you will have to write a JSON RPC server. - -::: - -In this case, let's make `foreignCallHandler` call the JSON RPC Server we created in [Step #2](#step-2---write-an-rpc-server), by making it a JSON RPC Client. - -For example, using the same `getSqrt` program in [Step #1](#step-1---modify-your-noir-program) (comments in the code): - -```js -import { JSONRPCClient } from "json-rpc-2.0"; - -// declaring the JSONRPCClient -const client = new JSONRPCClient((jsonRPCRequest) => { -// hitting the same JSON RPC Server we coded above - return fetch("http://localhost:5555", { - method: "POST", - headers: { - "content-type": "application/json", - }, - body: JSON.stringify(jsonRPCRequest), - }).then((response) => { - if (response.status === 200) { - return response - .json() - .then((jsonRPCResponse) => client.receive(jsonRPCResponse)); - } else if (jsonRPCRequest.id !== undefined) { - return Promise.reject(new Error(response.statusText)); - } - }); -}); - -// declaring a function that takes the name of the foreign call (getSqrt) and the inputs -const foreignCallHandler = async (name, input) => { - const inputs = input[0].map((i) => i.toString("hex")) - // notice that the "inputs" parameter contains *all* the inputs - // in this case we to make the RPC request with the first parameter "numbers", which would be input[0] - const oracleReturn = await client.request("resolve_foreign_call", [ - { - function: name, - inputs: [inputs] - }, - ]); - return [oracleReturn.values[0]]; -}; - -// the rest of your NoirJS code -const input = { input: [4, 16] }; -const { witness } = await noir.execute(input, foreignCallHandler); -``` - -:::tip - -If you're in a NoirJS environment running your RPC server together with a frontend app, you'll probably hit a familiar problem in full-stack development: requests being blocked by [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) policy. For development only, you can simply install and use the [`cors` npm package](https://www.npmjs.com/package/cors) to get around the problem: - -```bash -yarn add cors -``` - -and use it as a middleware: - -```js -import cors from "cors"; - -const app = express(); -app.use(cors()) -``` - -::: - -## Conclusion - -Hopefully by the end of this guide, you should be able to: - -- Write your own logic around Oracles and how to write a JSON RPC server to make them work with your Nargo commands. -- Provide custom foreign call handlers for NoirJS. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-recursion.md deleted file mode 100644 index fac79a9cf49..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-recursion.md +++ /dev/null @@ -1,172 +0,0 @@ ---- -title: How to use recursion on NoirJS -description: Learn how to implement recursion with NoirJS, a powerful tool for creating smart contracts on the EVM blockchain. This guide assumes familiarity with NoirJS, solidity verifiers, and the Barretenberg proving backend. Discover how to generate both final and intermediate proofs using `noir_js` and `bb.js`. -keywords: - [ - "NoirJS", - "EVM blockchain", - "smart contracts", - "recursion", - "solidity verifiers", - "Barretenberg backend", - "noir_js", - "intermediate proofs", - "final proofs", - "nargo compile", - "json import", - "recursive circuit", - "recursive app" - ] -sidebar_position: 1 ---- - -This guide shows you how to use recursive proofs in your NoirJS app. For the sake of clarity, it is assumed that: - -- You already have a NoirJS app. If you don't, please visit the [NoirJS tutorial](../tutorials/noirjs_app.md) and the [reference](../reference/NoirJS/noir_js/index.md). -- You are familiar with what are recursive proofs and you have read the [recursion explainer](../explainers/explainer-recursion.md) -- You already built a recursive circuit following [the reference](../noir/standard_library/recursion.mdx), and understand how it works. - -It is also assumed that you're not using `noir_wasm` for compilation, and instead you've used [`nargo compile`](../reference/nargo_commands.md) to generate the `json` you're now importing into your project. However, the guide should work just the same if you're using `noir_wasm`. - -:::info - -As you've read in the [explainer](../explainers/explainer-recursion.md), a recursive proof is an intermediate proof. This means that it doesn't necessarily generate the final step that makes it verifiable in a smart contract. However, it is easy to verify within another circuit. - -::: - -In a standard recursive app, you're also dealing with at least two circuits. For the purpose of this guide, we will assume the following: - -- `main`: a circuit of type `assert(x != y)`, where `main` is marked with a `#[recursive]` attribute. This attribute states that the backend should generate proofs that are friendly for verification within another circuit. -- `recursive`: a circuit that verifies `main` - -For a full example of how recursive proofs work, please refer to the [noir-examples](https://github.com/noir-lang/noir-examples) repository. We will *not* be using it as a reference for this guide. - -## Step 1: Setup - -In a common NoirJS app, you need to instantiate a backend with something like `const backend = new Backend(circuit)`. Then you feed it to the `noir_js` interface. - -For recursion, this doesn't happen, and the only need for `noir_js` is only to `execute` a circuit and get its witness and return value. Everything else is not interfaced, so it needs to happen on the `backend` object. - -It is also recommended that you instantiate the backend with as many threads as possible, to allow for maximum concurrency: - -```js -const backend = new Backend(circuit, { threads: 8 }) -``` - -:::tip -You can use the [`os.cpus()`](https://nodejs.org/api/os.html#oscpus) object in `nodejs` or [`navigator.hardwareConcurrency`](https://developer.mozilla.org/en-US/docs/Web/API/Navigator/hardwareConcurrency) on the browser to make the most out of those glorious cpu cores -::: - -## Step 2: Generating the witness and the proof for `main` - -After instantiating the backend, you should also instantiate `noir_js`. We will use it to execute the circuit and get the witness. - -```js -const noir = new Noir(circuit) -const { witness } = noir.execute(input) -``` - -With this witness, you are now able to generate the intermediate proof for the main circuit: - -```js -const { proof, publicInputs } = await backend.generateProof(witness) -``` - -:::warning - -Always keep in mind what is actually happening on your development process, otherwise you'll quickly become confused about what circuit we are actually running and why! - -In this case, you can imagine that Alice (running the `main` circuit) is proving something to Bob (running the `recursive` circuit), and Bob is verifying her proof within his proof. - -With this in mind, it becomes clear that our intermediate proof is the one *meant to be verified within another circuit*, so it must be Alice's. Actually, the only final proof in this theoretical scenario would be the last one, sent on-chain. - -::: - -## Step 3 - Verification and proof artifacts - -Optionally, you are able to verify the intermediate proof: - -```js -const verified = await backend.verifyProof({ proof, publicInputs }) -``` - -This can be useful to make sure our intermediate proof was correctly generated. But the real goal is to do it within another circuit. For that, we need to generate recursive proof artifacts that will be passed to the circuit that is verifying the proof we just generated. Instead of passing the proof and verification key as a byte array, we pass them as fields which makes it cheaper to verify in a circuit: - -```js -const { proofAsFields, vkAsFields, vkHash } = await backend.generateRecursiveProofArtifacts( { publicInputs, proof }, publicInputsCount) -``` - -This call takes the public inputs and the proof, but also the public inputs count. While this is easily retrievable by simply counting the `publicInputs` length, the backend interface doesn't currently abstract it away. - -:::info - -The `proofAsFields` has a constant size `[Field; 93]` and verification keys in Barretenberg are always `[Field; 114]`. - -::: - -:::warning - -One common mistake is to forget *who* makes this call. - -In a situation where Alice is generating the `main` proof, if she generates the proof artifacts and sends them to Bob, which gladly takes them as true, this would mean Alice could prove anything! - -Instead, Bob needs to make sure *he* extracts the proof artifacts, using his own instance of the `main` circuit backend. This way, Alice has to provide a valid proof for the correct `main` circuit. - -::: - -## Step 4 - Recursive proof generation - -With the artifacts, generating a recursive proof is no different from a normal proof. You simply use the `backend` (with the recursive circuit) to generate it: - -```js -const recursiveInputs = { - verification_key: vkAsFields, // array of length 114 - proof: proofAsFields, // array of length 93 + size of public inputs - publicInputs: [mainInput.y], // using the example above, where `y` is the only public input - key_hash: vkHash, -} - -const { witness, returnValue } = noir.execute(recursiveInputs) // we're executing the recursive circuit now! -const { proof, publicInputs } = backend.generateProof(witness) -const verified = backend.verifyProof({ proof, publicInputs }) -``` - -You can obviously chain this proof into another proof. In fact, if you're using recursive proofs, you're probably interested of using them this way! - -:::tip - -Managing circuits and "who does what" can be confusing. To make sure your naming is consistent, you can keep them in an object. For example: - -```js -const circuits = { - main: mainJSON, - recursive: recursiveJSON -} -const backends = { - main: new BarretenbergBackend(circuits.main), - recursive: new BarretenbergBackend(circuits.recursive) -} -const noir_programs = { - main: new Noir(circuits.main), - recursive: new Noir(circuits.recursive) -} -``` - -This allows you to neatly call exactly the method you want without conflicting names: - -```js -// Alice runs this 👇 -const { witness: mainWitness } = await noir_programs.main.execute(input) -const proof = await backends.main.generateProof(mainWitness) - -// Bob runs this 👇 -const verified = await backends.main.verifyProof(proof) -const { proofAsFields, vkAsFields, vkHash } = await backends.main.generateRecursiveProofArtifacts( - proof, - numPublicInputs, -); -const { witness: recursiveWitness } = await noir_programs.recursive.execute(recursiveInputs) -const recursiveProof = await backends.recursive.generateProof(recursiveWitness); -``` - -::: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/merkle-proof.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/merkle-proof.mdx deleted file mode 100644 index 0a128adb2de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/merkle-proof.mdx +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Prove Merkle Tree Membership -description: - Learn how to use merkle membership proof in Noir to prove that a given leaf is a member of a - merkle tree with a specified root, at a given index. -keywords: - [merkle proof, merkle membership proof, Noir, rust, hash function, Pedersen, sha256, merkle tree] -sidebar_position: 4 ---- - -Let's walk through an example of a merkle membership proof in Noir that proves that a given leaf is -in a merkle tree. - -```rust - -fn main(message : [Field; 62], index : Field, hashpath : [Field; 40], root : Field) { - let leaf = std::hash::hash_to_field(message.as_slice()); - let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); - assert(merkle_root == root); -} - -``` - -The message is hashed using `hash_to_field`. The specific hash function that is being used is chosen -by the backend. The only requirement is that this hash function can heuristically be used as a -random oracle. If only collision resistance is needed, then one can call `std::hash::pedersen_hash` -instead. - -```rust -let leaf = std::hash::hash_to_field(message.as_slice()); -``` - -The leaf is then passed to a compute_merkle_root function with the root, index and hashpath. The returned root can then be asserted to be the same as the provided root. - -```rust -let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); -assert (merkle_root == root); -``` - -> **Note:** It is possible to re-implement the merkle tree implementation without standard library. -> However, for most usecases, it is enough. In general, the standard library will always opt to be -> as conservative as possible, while striking a balance with efficiency. - -An example, the merkle membership proof, only requires a hash function that has collision -resistance, hence a hash function like Pedersen is allowed, which in most cases is more efficient -than the even more conservative sha256. - -[View an example on the starter repo](https://github.com/noir-lang/noir-examples/blob/3ea09545cabfa464124ec2f3ea8e60c608abe6df/stealthdrop/circuits/src/main.nr#L20) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/using-devcontainers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/using-devcontainers.mdx deleted file mode 100644 index 727ec6ca667..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/using-devcontainers.mdx +++ /dev/null @@ -1,110 +0,0 @@ ---- -title: Developer Containers and Codespaces -description: "Learn how to set up a devcontainer in your GitHub repository for a seamless coding experience with Codespaces. Follow our easy 8-step guide to create your own Noir environment without installing Nargo locally." -keywords: ["Devcontainer", "Codespaces", "GitHub", "Noir Environment", "Docker Image", "Development Environment", "Remote Coding", "GitHub Codespaces", "Noir Programming", "Nargo", "VSCode Extensions", "Noirup"] -sidebar_position: 1 ---- - -Adding a developer container configuration file to your Noir project is one of the easiest way to unlock coding in browser. - -## What's a devcontainer after all? - -A [Developer Container](https://containers.dev/) (devcontainer for short) is a Docker image that comes preloaded with tools, extensions, and other tools you need to quickly get started or continue a project, without having to install Nargo locally. Think of it as a development environment in a box. - -There are many advantages to this: - -- It's platform and architecture agnostic -- You don't need to have an IDE installed, or Nargo, or use a terminal at all -- It's safer for using on a public machine or public network - -One of the best ways of using devcontainers is... not using your machine at all, for maximum control, performance, and ease of use. -Enter Codespaces. - -## Codespaces - -If a devcontainer is just a Docker image, then what stops you from provisioning a `p3dn.24xlarge` AWS EC2 instance with 92 vCPUs and 768 GiB RAM and using it to prove your 10-gate SNARK proof? - -Nothing! Except perhaps the 30-40$ per hour it will cost you. - -The problem is that provisioning takes time, and I bet you don't want to see the AWS console every time you want to code something real quick. - -Fortunately, there's an easy and free way to get a decent remote machine ready and loaded in less than 2 minutes: Codespaces. [Codespaces is a Github feature](https://github.com/features/codespaces) that allows you to code in a remote machine by using devcontainers, and it's pretty cool: - -- You can start coding Noir in less than a minute -- It uses the resources of a remote machine, so you can code on your grandma's phone if needed be -- It makes it easy to share work with your frens -- It's fully reusable, you can stop and restart whenever you need to - -:::info - -Don't take out your wallet just yet. Free GitHub accounts get about [15-60 hours of coding](https://github.com/features/codespaces) for free per month, depending on the size of your provisioned machine. - -::: - -## Tell me it's _actually_ easy - -It is! - -Github comes with a default codespace and you can use it to code your own devcontainer. That's exactly what we will be doing in this guide. - - - -8 simple steps: - -#### 1. Create a new repository on GitHub. - -#### 2. Click "Start coding with Codespaces". This will use the default image. - -#### 3. Create a folder called `.devcontainer` in the root of your repository. - -#### 4. Create a Dockerfile in that folder, and paste the following code: - -```docker -FROM --platform=linux/amd64 node:lts-bookworm-slim -SHELL ["/bin/bash", "-c"] -RUN apt update && apt install -y curl bash git tar gzip libc++-dev -RUN curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -ENV PATH="/root/.nargo/bin:$PATH" -RUN noirup -ENTRYPOINT ["nargo"] -``` -#### 5. Create a file called `devcontainer.json` in the same folder, and paste the following code: - -```json -{ - "name": "Noir on Codespaces", - "build": { - "context": ".", - "dockerfile": "Dockerfile" - }, - "customizations": { - "vscode": { - "extensions": ["noir-lang.vscode-noir"] - } - } -} -``` -#### 6. Commit and push your changes - -This will pull the new image and build it, so it could take a minute or so - -#### 8. Done! -Just wait for the build to finish, and there's your easy Noir environment. - - -Refer to [noir-starter](https://github.com/noir-lang/noir-starter/) as an example of how devcontainers can be used together with codespaces. - - - -## How do I use it? - -Using the codespace is obviously much easier than setting it up. -Just navigate to your repository and click "Code" -> "Open with Codespaces". It should take a few seconds to load, and you're ready to go. - -:::info - -If you really like the experience, you can add a badge to your readme, links to existing codespaces, and more. -Check out the [official docs](https://docs.github.com/en/codespaces/setting-up-your-project-for-codespaces/setting-up-your-repository/facilitating-quick-creation-and-resumption-of-codespaces) for more info. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/migration_notes.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/migration_notes.md deleted file mode 100644 index 6bd740024e5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/migration_notes.md +++ /dev/null @@ -1,105 +0,0 @@ ---- -title: Migration notes -description: Read about migration notes from previous versions, which could solve problems while updating -keywords: [Noir, notes, migration, updating, upgrading] ---- - -Noir is in full-speed development. Things break fast, wild, and often. This page attempts to leave some notes on errors you might encounter when upgrading and how to resolve them until proper patches are built. - -### `backend encountered an error: libc++.so.1` - -Depending on your OS, you may encounter the following error when running `nargo prove` for the first time: - -```text -The backend encountered an error: "/home/codespace/.nargo/backends/acvm-backend-barretenberg/backend_binary: error while loading shared libraries: libc++.so.1: cannot open shared object file: No such file or directory\n" -``` - -Install the `libc++-dev` library with: - -```bash -sudo apt install libc++-dev -``` - -## ≥0.19 - -### Enforcing `compiler_version` - -From this version on, the compiler will check for the `compiler_version` field in `Nargo.toml`, and will error if it doesn't match the current Nargo version in use. - -To update, please make sure this field in `Nargo.toml` matches the output of `nargo --version`. - -## ≥0.14 - -The index of the [for loops](noir/concepts/control_flow.md#loops) is now of type `u64` instead of `Field`. An example refactor would be: - -```rust -for i in 0..10 { - let i = i as Field; -} -``` - -## ≥v0.11.0 and Nargo backend - -From this version onwards, Nargo starts managing backends through the `nargo backend` command. Upgrading to the versions per usual steps might lead to: - -### `backend encountered an error` - -This is likely due to the existing locally installed version of proving backend (e.g. barretenberg) is incompatible with the version of Nargo in use. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo prove -``` - -with your Noir program. - -This will trigger the download and installation of the latest version of barretenberg compatible with your Nargo in use. - -### `backend encountered an error: illegal instruction` - -On certain Intel-based systems, an `illegal instruction` error may arise due to incompatibility of barretenberg with certain CPU instructions. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo backend install acvm-backend-barretenberg https://github.com/noir-lang/barretenberg-js-binary/raw/master/run-bb.tar.gz -``` - -This downloads and installs a specific bb.js based version of barretenberg binary from GitHub. - -The gzipped file is running [this bash script](https://github.com/noir-lang/barretenberg-js-binary/blob/master/run-bb-js.sh), where we need to gzip it as the Nargo currently expect the backend to be zipped up. - -Then run: - -``` -DESIRED_BINARY_VERSION=0.8.1 nargo info -``` - -This overrides the bb native binary with a bb.js node application instead, which should be compatible with most if not all hardware. This does come with the drawback of being generally slower than native binary. - -0.8.1 indicates bb.js version 0.8.1, so if you change that it will update to a different version or the default version in the script if none was supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/_category_.json deleted file mode 100644 index 7da08f8a8c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Concepts", - "position": 0, - "collapsible": true, - "collapsed": true -} \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/assert.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/assert.md deleted file mode 100644 index 2132de42072..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/assert.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Assert Function -description: - Learn about the `assert` and `static_assert` functions in Noir, which can be used to explicitly - constrain the predicate or comparison expression that follows to be true, and what happens if - the expression is false at runtime or compile-time, respectively. -keywords: [Noir programming language, assert statement, predicate expression, comparison expression] -sidebar_position: 4 ---- - -Noir includes a special `assert` function which will explicitly constrain the predicate/comparison -expression that follows to be true. If this expression is false at runtime, the program will fail to -be proven. Example: - -```rust -fn main(x : Field, y : Field) { - assert(x == y); -} -``` - -> Assertions only work for predicate operations, such as `==`. If there's any ambiguity on the operation, the program will fail to compile. For example, it is unclear if `assert(x + y)` would check for `x + y == 0` or simply would return `true`. - -You can optionally provide a message to be logged when the assertion fails: - -```rust -assert(x == y, "x and y are not equal"); -``` - -Aside string literals, the optional message can be a format string or any other type supported as input for Noir's [print](../standard_library/logging.md) functions. This feature lets you incorporate runtime variables into your failed assertion logs: - -```rust -assert(x == y, f"Expected x == y, but got {x} == {y}"); -``` - -Using a variable as an assertion message directly: - -```rust -struct myStruct { - myField: Field -} - -let s = myStruct { myField: y }; -assert(s.myField == x, s); -``` - -There is also a special `static_assert` function that behaves like `assert`, -but that runs at compile-time. - -```rust -fn main(xs: [Field; 3]) { - let x = 2 + 2; - let y = 4; - static_assert(x == y, "expected 2 + 2 to equal 4"); - - // This passes since the length of `xs` is known at compile-time - static_assert(xs.len() == 3, "expected the input to have 3 elements"); -} -``` - -This function fails when passed a dynamic (run-time) argument: - -```rust -fn main(x : Field, y : Field) { - // this fails because `x` is not known at compile-time - static_assert(x == 2, "expected x to be known at compile-time and equal to 2"); - - let mut example_slice = &[]; - if y == 4 { - example_slice = example_slice.push_back(0); - } - - // This fails because the length of `example_slice` is not known at - // compile-time - let error_message = "expected an empty slice, known at compile-time"; - static_assert(example_slice.len() == 0, error_message); -} -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/comments.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/comments.md deleted file mode 100644 index b51a85f5c94..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/comments.md +++ /dev/null @@ -1,33 +0,0 @@ ---- -title: Comments -description: - Learn how to write comments in Noir programming language. A comment is a line of code that is - ignored by the compiler, but it can be read by programmers. Single-line and multi-line comments - are supported in Noir. -keywords: [Noir programming language, comments, single-line comments, multi-line comments] -sidebar_position: 10 ---- - -A comment is a line in your codebase which the compiler ignores, however it can be read by -programmers. - -Here is a single line comment: - -```rust -// This is a comment and is ignored -``` - -`//` is used to tell the compiler to ignore the rest of the line. - -Noir also supports multi-line block comments. Start a block comment with `/*` and end the block with `*/`. - -Noir does not natively support doc comments. You may be able to use [Rust doc comments](https://doc.rust-lang.org/reference/comments.html) in your code to leverage some Rust documentation build tools with Noir code. - -```rust -/* - This is a block comment describing a complex function. -*/ -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_bus.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_bus.mdx deleted file mode 100644 index e55e58622ce..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_bus.mdx +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: Data Bus -sidebar_position: 13 ---- -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -The data bus is an optimization that the backend can use to make recursion more efficient. -In order to use it, you must define some inputs of the program entry points (usually the `main()` -function) with the `call_data` modifier, and the return values with the `return_data` modifier. -These modifiers are incompatible with `pub` and `mut` modifiers. - -## Example - -```rust -fn main(mut x: u32, y: call_data u32, z: call_data [u32;4] ) -> return_data u32 { - let a = z[x]; - a+y -} -``` - -As a result, both call_data and return_data will be treated as private inputs and encapsulated into a read-only array each, for the backend to process. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/booleans.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/booleans.md deleted file mode 100644 index 2507af710e7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/booleans.md +++ /dev/null @@ -1,28 +0,0 @@ ---- -title: Booleans -description: - Delve into the Boolean data type in Noir. Understand its methods, practical examples, and best practices for using Booleans in your Noir programs. -keywords: - [ - noir, - boolean type, - methods, - examples, - logical operations, - ] -sidebar_position: 2 ---- - - -The `bool` type in Noir has two possible values: `true` and `false`: - -```rust -fn main() { - let t = true; - let f: bool = false; -} -``` - -The boolean type is most commonly used in conditionals like `if` expressions and `assert` -statements. More about conditionals is covered in the [Control Flow](../control_flow.md) and -[Assert Function](../assert.md) sections. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/fields.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/fields.md deleted file mode 100644 index b9b56f7ecc3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/fields.md +++ /dev/null @@ -1,246 +0,0 @@ ---- -title: Fields -description: - Dive deep into the Field data type in Noir. Understand its methods, practical examples, and best practices to effectively use Fields in your Noir programs. -keywords: - [ - noir, - field type, - methods, - examples, - best practices, - ] -sidebar_position: 0 ---- - -The field type corresponds to the native field type of the proving backend. - -The size of a Noir field depends on the elliptic curve's finite field for the proving backend -adopted. For example, a field would be a 254-bit integer when paired with the default backend that -spans the Grumpkin curve. - -Fields support integer arithmetic and are often used as the default numeric type in Noir: - -```rust -fn main(x : Field, y : Field) { - let z = x + y; -} -``` - -`x`, `y` and `z` are all private fields in this example. Using the `let` keyword we defined a new -private value `z` constrained to be equal to `x + y`. - -If proving efficiency is of priority, fields should be used as a default for solving problems. -Smaller integer types (e.g. `u64`) incur extra range constraints. - -## Methods - -After declaring a Field, you can use these common methods on it: - -### to_le_bits - -Transforms the field into an array of bits, Little Endian. - -```rust title="to_le_bits" showLineNumbers -pub fn to_le_bits(self: Self) -> [u1; N] {} -``` -> Source code: noir_stdlib/src/field/mod.nr#L32-L34 - - -example: - -```rust title="to_le_bits_example" showLineNumbers -fn test_to_le_bits() { - let field = 2; - let bits: [u1; 8] = field.to_le_bits(); - assert_eq(bits, [0, 1, 0, 0, 0, 0, 0, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L276-L282 - - - -### to_be_bits - -Transforms the field into an array of bits, Big Endian. - -```rust title="to_be_bits" showLineNumbers -pub fn to_be_bits(self: Self) -> [u1; N] {} -``` -> Source code: noir_stdlib/src/field/mod.nr#L48-L50 - - -example: - -```rust title="to_be_bits_example" showLineNumbers -fn test_to_be_bits() { - let field = 2; - let bits: [u1; 8] = field.to_be_bits(); - assert_eq(bits, [0, 0, 0, 0, 0, 0, 1, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L267-L273 - - - -### to_le_bytes - -Transforms into an array of bytes, Little Endian - -```rust title="to_le_bytes" showLineNumbers -pub fn to_le_bytes(self: Self) -> [u8; N] { -``` -> Source code: noir_stdlib/src/field/mod.nr#L61-L63 - - -example: - -```rust title="to_le_bytes_example" showLineNumbers -fn test_to_le_bytes() { - let field = 2; - let bytes: [u8; 8] = field.to_le_bytes(); - assert_eq(bytes, [2, 0, 0, 0, 0, 0, 0, 0]); - assert_eq(Field::from_le_bytes::<8>(bytes), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L295-L302 - - -### to_be_bytes - -Transforms into an array of bytes, Big Endian - -```rust title="to_be_bytes" showLineNumbers -pub fn to_be_bytes(self: Self) -> [u8; N] { -``` -> Source code: noir_stdlib/src/field/mod.nr#L94-L96 - - -example: - -```rust title="to_be_bytes_example" showLineNumbers -fn test_to_be_bytes() { - let field = 2; - let bytes: [u8; 8] = field.to_be_bytes(); - assert_eq(bytes, [0, 0, 0, 0, 0, 0, 0, 2]); - assert_eq(Field::from_be_bytes::<8>(bytes), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L285-L292 - - - -### to_le_radix - -Decomposes into an array over the specified base, Little Endian - -```rust title="to_le_radix" showLineNumbers -pub fn to_le_radix(self: Self, radix: u32) -> [u8; N] { - // Brillig does not need an immediate radix - if !crate::runtime::is_unconstrained() { - crate::assert_constant(radix); - } - self.__to_le_radix(radix) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L118-L126 - - - -example: - -```rust title="to_le_radix_example" showLineNumbers -fn test_to_le_radix() { - let field = 2; - let bytes: [u8; 8] = field.to_le_radix(256); - assert_eq(bytes, [2, 0, 0, 0, 0, 0, 0, 0]); - assert_eq(Field::from_le_bytes::<8>(bytes), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L315-L322 - - - -### to_be_radix - -Decomposes into an array over the specified base, Big Endian - -```rust title="to_be_radix" showLineNumbers -pub fn to_be_radix(self: Self, radix: u32) -> [u8; N] { - // Brillig does not need an immediate radix - if !crate::runtime::is_unconstrained() { - crate::assert_constant(radix); - } - self.__to_be_radix(radix) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L128-L136 - - -example: - -```rust title="to_be_radix_example" showLineNumbers -fn test_to_be_radix() { - let field = 2; - let bytes: [u8; 8] = field.to_be_radix(256); - assert_eq(bytes, [0, 0, 0, 0, 0, 0, 0, 2]); - assert_eq(Field::from_be_bytes::<8>(bytes), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L305-L312 - - - -### pow_32 - -Returns the value to the power of the specified exponent - -```rust -fn pow_32(self, exponent: Field) -> Field -``` - -example: - -```rust -fn main() { - let field = 2 - let pow = field.pow_32(4); - assert(pow == 16); -} -``` - -### assert_max_bit_size - -Adds a constraint to specify that the field can be represented with `bit_size` number of bits - -```rust title="assert_max_bit_size" showLineNumbers -pub fn assert_max_bit_size(self) { -``` -> Source code: noir_stdlib/src/field/mod.nr#L10-L12 - - -example: - -```rust -fn main() { - let field = 2 - field.assert_max_bit_size(32); -} -``` - -### sgn0 - -Parity of (prime) Field element, i.e. sgn0(x mod p) = 0 if x ∈ \{0, ..., p-1\} is even, otherwise sgn0(x mod p) = 1. - -```rust -fn sgn0(self) -> u1 -``` - - -### lt - -Returns true if the field is less than the other field - -```rust -pub fn lt(self, another: Field) -> bool -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/function_types.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/function_types.md deleted file mode 100644 index f6121af17e2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/function_types.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Function types -sidebar_position: 10 ---- - -Noir supports higher-order functions. The syntax for a function type is as follows: - -```rust -fn(arg1_type, arg2_type, ...) -> return_type -``` - -Example: - -```rust -fn assert_returns_100(f: fn() -> Field) { // f takes no args and returns a Field - assert(f() == 100); -} - -fn main() { - assert_returns_100(|| 100); // ok - assert_returns_100(|| 150); // fails -} -``` - -A function type also has an optional capture environment - this is necessary to support closures. -See [Lambdas](../lambdas.md) for more details. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/integers.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/integers.md deleted file mode 100644 index a1d59bf3166..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/integers.md +++ /dev/null @@ -1,156 +0,0 @@ ---- -title: Integers -description: Explore the Integer data type in Noir. Learn about its methods, see real-world examples, and grasp how to efficiently use Integers in your Noir code. -keywords: [noir, integer types, methods, examples, arithmetic] -sidebar_position: 1 ---- - -An integer type is a range constrained field type. -The Noir frontend supports both unsigned and signed integer types. -The allowed sizes are 1, 8, 16, 32 and 64 bits. - -:::info - -When an integer is defined in Noir without a specific type, it will default to `Field`. - -The one exception is for loop indices which default to `u64` since comparisons on `Field`s are not possible. - -::: - -## Unsigned Integers - -An unsigned integer type is specified first with the letter `u` (indicating its unsigned nature) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: u8 = 1; - let y: u8 = 1; - let z = x + y; - assert (z == 2); -} -``` - -The bit size determines the maximum value the integer type can store. For example, a `u8` variable can store a value in the range of 0 to 255 (i.e. $\\2^{8}-1\\$). - -## Signed Integers - -A signed integer type is specified first with the letter `i` (which stands for integer) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: i8 = -1; - let y: i8 = -1; - let z = x + y; - assert (z == -2); -} -``` - -The bit size determines the maximum and minimum range of value the integer type can store. For example, an `i8` variable can store a value in the range of -128 to 127 (i.e. $\\-2^{7}\\$ to $\\2^{7}-1\\$). - -## 128 bits Unsigned Integers - -The built-in structure `U128` allows you to use 128-bit unsigned integers almost like a native integer type. However, there are some differences to keep in mind: -- You cannot cast between a native integer and `U128` -- There is a higher performance cost when using `U128`, compared to a native type. - -Conversion between unsigned integer types and U128 are done through the use of `from_integer` and `to_integer` functions. `from_integer` also accepts the `Field` type as input. - -```rust -fn main() { - let x = U128::from_integer(23); - let y = U128::from_hex("0x7"); - let z = x + y; - assert(z.to_integer() == 30); -} -``` - -`U128` is implemented with two 64 bits limbs, representing the low and high bits, which explains the performance cost. You should expect `U128` to be twice more costly for addition and four times more costly for multiplication. -You can construct a U128 from its limbs: -```rust -fn main(x: u64, y: u64) { - let x = U128::from_u64s_be(x,y); - assert(z.hi == x as Field); - assert(z.lo == y as Field); -} -``` - -Note that the limbs are stored as Field elements in order to avoid unnecessary conversions. -Apart from this, most operations will work as usual: - -```rust -fn main(x: U128, y: U128) { - // multiplication - let c = x * y; - // addition and subtraction - let c = c - x + y; - // division - let c = x / y; - // bit operation; - let c = x & y | y; - // bit shift - let c = x << y; - // comparisons; - let c = x < y; - let c = x == y; -} -``` - -## Overflows - -Computations that exceed the type boundaries will result in overflow errors. This happens with both signed and unsigned integers. For example, attempting to prove: - -```rust -fn main(x: u8, y: u8) { - let z = x + y; -} -``` - -With: - -```toml -x = "255" -y = "1" -``` - -Would result in: - -``` -$ nargo execute -error: Assertion failed: 'attempt to add with overflow' -┌─ ~/src/main.nr:9:13 -│ -│ let z = x + y; -│ ----- -│ -= Call stack: - ... -``` - -A similar error would happen with signed integers: - -```rust -fn main() { - let x: i8 = -118; - let y: i8 = -11; - let z = x + y; -} -``` - -### Wrapping methods - -Although integer overflow is expected to error, some use-cases rely on wrapping. For these use-cases, the standard library provides `wrapping` variants of certain common operations: - -```rust -fn wrapping_add(x: T, y: T) -> T; -fn wrapping_sub(x: T, y: T) -> T; -fn wrapping_mul(x: T, y: T) -> T; -``` - -Example of how it is used: - -```rust - -fn main(x: u8, y: u8) -> pub u8 { - std::wrapping_add(x, y) -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/references.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/references.md deleted file mode 100644 index a5293d11cfb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/references.md +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: References -sidebar_position: 9 ---- - -Noir supports first-class references. References are a bit like pointers: they point to a specific address that can be followed to access the data stored at that address. You can use Rust-like syntax to use pointers in Noir: the `&` operator references the variable, the `*` operator dereferences it. - -Example: - -```rust -fn main() { - let mut x = 2; - - // you can reference x as &mut and pass it to multiplyBy2 - multiplyBy2(&mut x); -} - -// you can access &mut here -fn multiplyBy2(x: &mut Field) { - // and dereference it with * - *x = *x * 2; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/slices.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/slices.mdx deleted file mode 100644 index cfee564a302..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/slices.mdx +++ /dev/null @@ -1,358 +0,0 @@ ---- -title: Slices -description: Explore the Slice data type in Noir. Understand its methods, see real-world examples, and learn how to effectively use Slices in your Noir programs. -keywords: [noir, slice type, methods, examples, subarrays] -sidebar_position: 5 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A slice is a dynamically-sized view into a sequence of elements. They can be resized at runtime, but because they don't own the data, they cannot be returned from a circuit. You can treat slices as arrays without a constrained size. - -```rust -fn main() -> pub u32 { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -To write a slice literal, use a preceding ampersand as in: `&[0; 2]` or -`&[1, 2, 3]`. - -It is important to note that slices are not references to arrays. In Noir, -`&[..]` is more similar to an immutable, growable vector. - -View the corresponding test file [here][test-file]. - -[test-file]: https://github.com/noir-lang/noir/blob/f387ec1475129732f72ba294877efdf6857135ac/crates/nargo_cli/tests/test_data_ssa_refactor/slices/src/main.nr - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for slices: - -### push_back - -Pushes a new element to the end of the slice, returning a new slice with a length one greater than the original unmodified slice. - -```rust -fn push_back(_self: [T], _elem: T) -> [T] -``` - -example: - -```rust -fn main() -> pub Field { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -View the corresponding test file [here][test-file]. - -### push_front - -Returns a new array with the specified element inserted at index 0. The existing elements indexes are incremented by 1. - -```rust -fn push_front(_self: Self, _elem: T) -> Self -``` - -Example: - -```rust -let mut new_slice: [Field] = &[]; -new_slice = new_slice.push_front(20); -assert(new_slice[0] == 20); // returns true -``` - -View the corresponding test file [here][test-file]. - -### pop_front - -Returns a tuple of two items, the first element of the array and the rest of the array. - -```rust -fn pop_front(_self: Self) -> (T, Self) -``` - -Example: - -```rust -let (first_elem, rest_of_slice) = slice.pop_front(); -``` - -View the corresponding test file [here][test-file]. - -### pop_back - -Returns a tuple of two items, the beginning of the array with the last element omitted and the last element. - -```rust -fn pop_back(_self: Self) -> (Self, T) -``` - -Example: - -```rust -let (popped_slice, last_elem) = slice.pop_back(); -``` - -View the corresponding test file [here][test-file]. - -### append - -Loops over a slice and adds it to the end of another. - -```rust -fn append(mut self, other: Self) -> Self -``` - -Example: - -```rust -let append = &[1, 2].append(&[3, 4, 5]); -``` - -### insert - -Inserts an element at a specified index and shifts all following elements by 1. - -```rust -fn insert(_self: Self, _index: Field, _elem: T) -> Self -``` - -Example: - -```rust -new_slice = rest_of_slice.insert(2, 100); -assert(new_slice[2] == 100); -``` - -View the corresponding test file [here][test-file]. - -### remove - -Remove an element at a specified index, shifting all elements after it to the left, returning the altered slice and the removed element. - -```rust -fn remove(_self: Self, _index: Field) -> (Self, T) -``` - -Example: - -```rust -let (remove_slice, removed_elem) = slice.remove(3); -``` - -### len - -Returns the length of a slice - -```rust -fn len(self) -> Field -``` - -Example: - -```rust -fn main() { - let slice = &[42, 42]; - assert(slice.len() == 2); -} -``` - -### as_array - -Converts this slice into an array. - -Make sure to specify the size of the resulting array. -Panics if the resulting array length is different than the slice's length. - -```rust -fn as_array(self) -> [T; N] -``` - -Example: - -```rust -fn main() { - let slice = &[5, 6]; - - // Always specify the length of the resulting array! - let array: [Field; 2] = slice.as_array(); - - assert(array[0] == slice[0]); - assert(array[1] == slice[1]); -} -``` - -### map - -Applies a function to each element of the slice, returning a new slice containing the mapped elements. - -```rust -fn map(self, f: fn[Env](T) -> U) -> [U] -``` - -example - -```rust -let a = &[1, 2, 3]; -let b = a.map(|a| a * 2); // b is now &[2, 4, 6] -``` - -### fold - -Applies a function to each element of the slice, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn[Env](U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the slice, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = &[1]; -let a2 = &[1, 2]; -let a3 = &[1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let folded = slice.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn[Env](T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let reduced = slice.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### filter - -Returns a new slice containing only elements for which the given predicate returns true. - -```rust -fn filter(self, f: fn[Env](T) -> bool) -> Self -``` - -example: - -```rust -fn main() { - let slice = &[1, 2, 3, 4, 5]; - let odds = slice.filter(|x| x % 2 == 1); - assert_eq(odds, &[1, 3, 5]); -} -``` - -### join - -Flatten each element in the slice into one value, separated by `separator`. - -Note that although slices implement `Append`, `join` cannot be used on slice -elements since nested slices are prohibited. - -```rust -fn join(self, separator: T) -> T where T: Append -``` - -example: - -```rust -struct Accumulator { - total: Field, -} - -// "Append" two accumulators by adding them -impl Append for Accumulator { - fn empty() -> Self { - Self { total: 0 } - } - - fn append(self, other: Self) -> Self { - Self { total: self.total + other.total } - } -} - -fn main() { - let slice = &[1, 2, 3, 4, 5].map(|total| Accumulator { total }); - - let result = slice.join(Accumulator::empty()); - assert_eq(result, Accumulator { total: 15 }); - - // We can use a non-empty separator to insert additional elements to sum: - let separator = Accumulator { total: 10 }; - let result = slice.join(separator); - assert_eq(result, Accumulator { total: 55 }); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let all = slice.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 5]; - let any = slice.any(|a| a == 5); - assert(any); -} - -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/strings.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/strings.md deleted file mode 100644 index 1fdee42425e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/strings.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Strings -description: - Discover the String data type in Noir. Learn about its methods, see real-world examples, and understand how to effectively manipulate and use Strings in Noir. -keywords: - [ - noir, - string type, - methods, - examples, - concatenation, - ] -sidebar_position: 3 ---- - - -The string type is a fixed length value defined with `str`. - -You can use strings in `assert()` functions or print them with -`println()`. See more about [Logging](../../standard_library/logging.md). - -```rust - -fn main(message : pub str<11>, hex_as_string : str<4>) { - println(message); - assert(message == "hello world"); - assert(hex_as_string == "0x41"); -} -``` - -You can convert a `str` to a byte array by calling `as_bytes()` -or a vector by calling `as_bytes_vec()`. - -```rust -fn main() { - let message = "hello world"; - let message_bytes = message.as_bytes(); - let mut message_vec = message.as_bytes_vec(); - assert(message_bytes.len() == 11); - assert(message_bytes[0] == 104); - assert(message_bytes[0] == message_vec.get(0)); -} -``` - -## Escape characters - -You can use escape characters for your strings: - -| Escape Sequence | Description | -|-----------------|-----------------| -| `\r` | Carriage Return | -| `\n` | Newline | -| `\t` | Tab | -| `\0` | Null Character | -| `\"` | Double Quote | -| `\\` | Backslash | - -Example: - -```rust -let s = "Hello \"world" // prints "Hello "world" -let s = "hey \tyou"; // prints "hey you" -``` - -## Raw strings - -A raw string begins with the letter `r` and is optionally delimited by a number of hashes `#`. - -Escape characters are *not* processed within raw strings. All contents are interpreted literally. - -Example: - -```rust -let s = r"Hello world"; -let s = r#"Simon says "hello world""#; - -// Any number of hashes may be used (>= 1) as long as the string also terminates with the same number of hashes -let s = r#####"One "#, Two "##, Three "###, Four "####, Five will end the string."#####; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/tuples.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/tuples.md deleted file mode 100644 index 2ec5c9c4113..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/tuples.md +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Tuples -description: - Dive into the Tuple data type in Noir. Understand its methods, practical examples, and best practices for efficiently using Tuples in your Noir code. -keywords: - [ - noir, - tuple type, - methods, - examples, - multi-value containers, - ] -sidebar_position: 7 ---- - -A tuple collects multiple values like an array, but with the added ability to collect values of -different types: - -```rust -fn main() { - let tup: (u8, u64, Field) = (255, 500, 1000); -} -``` - -One way to access tuple elements is via destructuring using pattern matching: - -```rust -fn main() { - let tup = (1, 2); - - let (one, two) = tup; - - let three = one + two; -} -``` - -Another way to access tuple elements is via direct member access, using a period (`.`) followed by -the index of the element we want to access. Index `0` corresponds to the first tuple element, `1` to -the second and so on: - -```rust -fn main() { - let tup = (5, 6, 7, 8); - - let five = tup.0; - let eight = tup.3; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/functions.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/functions.md deleted file mode 100644 index f656cdfd97a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/functions.md +++ /dev/null @@ -1,226 +0,0 @@ ---- -title: Functions -description: - Learn how to declare functions and methods in Noir, a programming language with Rust semantics. - This guide covers parameter declaration, return types, call expressions, and more. -keywords: [Noir, Rust, functions, methods, parameter declaration, return types, call expressions] -sidebar_position: 1 ---- - -Functions in Noir follow the same semantics of Rust, though Noir does not support early returns. - -To declare a function the `fn` keyword is used. - -```rust -fn foo() {} -``` - -By default, functions are visible only within the package they are defined. To make them visible outside of that package (for example, as part of a [library](../modules_packages_crates/crates_and_packages.md#libraries)), you should mark them as `pub`: - -```rust -pub fn foo() {} -``` - -You can also restrict the visibility of the function to only the crate it was defined in, by specifying `pub(crate)`: - -```rust -pub(crate) fn foo() {} //foo can only be called within its crate -``` - -All parameters in a function must have a type and all types are known at compile time. The parameter -is pre-pended with a colon and the parameter type. Multiple parameters are separated using a comma. - -```rust -fn foo(x : Field, y : Field){} -``` - -The return type of a function can be stated by using the `->` arrow notation. The function below -states that the foo function must return a `Field`. If the function returns no value, then the arrow -is omitted. - -```rust -fn foo(x : Field, y : Field) -> Field { - x + y -} -``` - -Note that a `return` keyword is unneeded in this case - the last expression in a function's body is -returned. - -## Main function - -If you're writing a binary, the `main` function is the starting point of your program. You can pass all types of expressions to it, as long as they have a fixed size at compile time: - -```rust -fn main(x : Field) // this is fine: passing a Field -fn main(x : [Field; 2]) // this is also fine: passing a Field with known size at compile-time -fn main(x : (Field, bool)) // 👌: passing a (Field, bool) tuple means size 2 -fn main(x : str<5>) // this is fine, as long as you pass a string of size 5 - -fn main(x : Vec) // can't compile, has variable size -fn main(x : [Field]) // can't compile, has variable size -fn main(....// i think you got it by now -``` - -Keep in mind [tests](../../tooling/testing.md) don't differentiate between `main` and any other function. The following snippet passes tests, but won't compile or prove: - -```rust -fn main(x : [Field]) { - assert(x[0] == 1); -} - -#[test] -fn test_one() { - main(&[1, 2]); -} -``` - -```bash -$ nargo test -[testing] Running 1 test functions -[testing] Testing test_one... ok -[testing] All tests passed - -$ nargo check -The application panicked (crashed). -Message: Cannot have variable sized arrays as a parameter to main -``` - -## Call Expressions - -Calling a function in Noir is executed by using the function name and passing in the necessary -arguments. - -Below we show how to call the `foo` function from the `main` function using a call expression: - -```rust -fn main(x : Field, y : Field) { - let z = foo(x); -} - -fn foo(x : Field) -> Field { - x + x -} -``` - -## Methods - -You can define methods in Noir on any struct type in scope. - -```rust -struct MyStruct { - foo: Field, - bar: Field, -} - -impl MyStruct { - fn new(foo: Field) -> MyStruct { - MyStruct { - foo, - bar: 2, - } - } - - fn sum(self) -> Field { - self.foo + self.bar - } -} - -fn main() { - let s = MyStruct::new(40); - assert(s.sum() == 42); -} -``` - -Methods are just syntactic sugar for functions, so if we wanted to we could also call `sum` as -follows: - -```rust -assert(MyStruct::sum(s) == 42); -``` - -It is also possible to specialize which method is chosen depending on the [generic](./generics.md) type that is used. In this example, the `foo` function returns different values depending on its type: - -```rust -struct Foo {} - -impl Foo { - fn foo(self) -> Field { 1 } -} - -impl Foo { - fn foo(self) -> Field { 2 } -} - -fn main() { - let f1: Foo = Foo{}; - let f2: Foo = Foo{}; - assert(f1.foo() + f2.foo() == 3); -} -``` - -Also note that impls with the same method name defined in them cannot overlap. For example, if we already have `foo` defined for `Foo` and `Foo` like we do above, we cannot also define `foo` in an `impl Foo` since it would be ambiguous which version of `foo` to choose. - -```rust -// Including this impl in the same project as the above snippet would -// cause an overlapping impls error -impl Foo { - fn foo(self) -> Field { 3 } -} -``` - -## Lambdas - -Lambdas are anonymous functions. They follow the syntax of Rust - `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -See [Lambdas](./lambdas.md) for more details. - -## Attributes - -Attributes are metadata that can be applied to a function, using the following syntax: `#[attribute(value)]`. - -Supported attributes include: - -- **builtin**: the function is implemented by the compiler, for efficiency purposes. -- **deprecated**: mark the function as _deprecated_. Calling the function will generate a warning: `warning: use of deprecated function` -- **field**: Used to enable conditional compilation of code depending on the field size. See below for more details -- **oracle**: mark the function as _oracle_; meaning it is an external unconstrained function, implemented in noir_js. See [Unconstrained](./unconstrained.md) and [NoirJS](../../reference/NoirJS/noir_js/index.md) for more details. -- **test**: mark the function as unit tests. See [Tests](../../tooling/testing.md) for more details - -### Field Attribute - -The field attribute defines which field the function is compatible for. The function is conditionally compiled, under the condition that the field attribute matches the Noir native field. -The field can be defined implicitly, by using the name of the elliptic curve usually associated to it - for instance bn254, bls12_381 - or explicitly by using the field (prime) order, in decimal or hexadecimal form. -As a result, it is possible to define multiple versions of a function with each version specialized for a different field attribute. This can be useful when a function requires different parameters depending on the underlying elliptic curve. - -Example: we define the function `foo()` three times below. Once for the default Noir bn254 curve, once for the field $\mathbb F_{23}$, which will normally never be used by Noir, and once again for the bls12_381 curve. - -```rust -#[field(bn254)] -fn foo() -> u32 { - 1 -} - -#[field(23)] -fn foo() -> u32 { - 2 -} - -// This commented code would not compile as foo would be defined twice because it is the same field as bn254 -// #[field(21888242871839275222246405745257275088548364400416034343698204186575808495617)] -// fn foo() -> u32 { -// 2 -// } - -#[field(bls12_381)] -fn foo() -> u32 { - 3 -} -``` - -If the field name is not known to Noir, it will discard the function. Field names are case insensitive. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/generics.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/generics.md deleted file mode 100644 index c180a0ce7e6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/generics.md +++ /dev/null @@ -1,251 +0,0 @@ ---- -title: Generics -description: Learn how to use Generics in Noir -keywords: [Noir, Rust, generics, functions, structs] -sidebar_position: 7 ---- - -Generics allow you to use the same functions with multiple different concrete data types. You can -read more about the concept of generics in the Rust documentation -[here](https://doc.rust-lang.org/book/ch10-01-syntax.html). - -Here is a trivial example showing the identity function that supports any type. In Rust, it is -common to refer to the most general type as `T`. We follow the same convention in Noir. - -```rust -fn id(x: T) -> T { - x -} -``` - -## Numeric Generics - -If we want to be generic over array lengths (which are type-level integers), we can use numeric -generics. Using these looks similar to using regular generics, but introducing them into scope -requires declaring them with `let MyGenericName: IntegerType`. This can be done anywhere a normal -generic is declared. Instead of types, these generics resolve to integers at compile-time. -Here's an example of a struct that is generic over the size of the array it contains internally: - -```rust -struct BigInt { - limbs: [u32; N], -} - -impl BigInt { - // `N` is in scope of all methods in the impl - fn first(first: BigInt, second: BigInt) -> Self { - assert(first.limbs != second.limbs); - first - - fn second(first: BigInt, second: Self) -> Self { - assert(first.limbs != second.limbs); - second - } -} -``` - -## In Structs - -Generics are useful for specifying types in structs. For example, we can specify that a field in a -struct will be of a certain generic type. In this case `value` is of type `T`. - -```rust -struct RepeatedValue { - value: T, - count: Field, -} - -impl RepeatedValue { - fn print(self) { - for _i in 0 .. self.count { - println(self.value); - } - } -} - -fn main() { - let repeated = RepeatedValue { value: "Hello!", count: 2 }; - repeated.print(); -} -``` - -The `print` function will print `Hello!` an arbitrary number of times, twice in this case. - -## Calling functions on generic parameters - -Since a generic type `T` can represent any type, how can we call functions on the underlying type? -In other words, how can we go from "any type `T`" to "any type `T` that has certain methods available?" - -This is what [traits](../concepts/traits.md) are for in Noir. Here's an example of a function generic over -any type `T` that implements the `Eq` trait for equality: - -```rust -fn first_element_is_equal(array1: [T; N], array2: [T; N]) -> bool - where T: Eq -{ - if (array1.len() == 0) | (array2.len() == 0) { - true - } else { - array1[0] == array2[0] - } -} - -fn main() { - assert(first_element_is_equal([1, 2, 3], [1, 5, 6])); - - // We can use first_element_is_equal for arrays of any type - // as long as we have an Eq impl for the types we pass in - let array = [MyStruct::new(), MyStruct::new()]; - assert(array_eq(array, array, MyStruct::eq)); -} - -impl Eq for MyStruct { - fn eq(self, other: MyStruct) -> bool { - self.foo == other.foo - } -} -``` - -You can find more details on traits and trait implementations on the [traits page](../concepts/traits.md). - -## Manually Specifying Generics with the Turbofish Operator - -There are times when the compiler cannot reasonably infer what type should be used for a generic, or when the developer themselves may want to manually distinguish generic type parameters. This is where the `::<>` turbofish operator comes into play. - -The `::<>` operator can follow a variable or path and can be used to manually specify generic arguments within the angle brackets. -The name "turbofish" comes from that `::<>` looks like a little fish. - -Examples: -```rust -fn main() { - let mut slice = []; - slice = slice.push_back(1); - slice = slice.push_back(2); - // Without turbofish a type annotation would be needed on the left hand side - let array = slice.as_array::<2>(); -} -``` - - -```rust -trait MyTrait { - fn ten() -> Self; -} - -impl MyTrait for Field { - fn ten() -> Self { 10 } -} - -struct Foo { - inner: T -} - -impl Foo { - fn generic_method(_self: Self) -> U where U: MyTrait { - U::ten() - } -} - -fn example() { - let foo: Foo = Foo { inner: 1 }; - // Using a type other than `Field` here (e.g. u32) would fail as - // there is no matching impl for `u32: MyTrait`. - // - // Substituting the `10` on the left hand side of this assert - // with `10 as u32` would also fail with a type mismatch as we - // are expecting a `Field` from the right hand side. - assert(10 as u32 == foo.generic_method::()); -} -``` - -## Arithmetic Generics - -In addition to numeric generics, Noir also allows a limited form of arithmetic on generics. -When you have a numeric generic such as `N`, you can use the following operators on it in a -type position: `+`, `-`, `*`, `/`, and `%`. - -Note that type checking arithmetic generics is a best effort guess from the compiler and there -are many cases of types that are equal that the compiler may not see as such. For example, -we know that `T * (N + M)` should be equal to `T*N + T*M` but the compiler does not currently -apply the distributive law and thus sees these as different types. - -Even with this limitation though, the compiler can handle common cases decently well: - -```rust -trait Serialize { - fn serialize(self) -> [Field; N]; -} - -impl Serialize<1> for Field { - fn serialize(self) -> [Field; 1] { - [self] - } -} - -impl Serialize for [T; N] - where T: Serialize { .. } - -impl Serialize for (T, U) - where T: Serialize, U: Serialize { .. } - -fn main() { - let data = (1, [2, 3, 4]); - assert_eq(data.serialize().len(), 4); -} -``` - -Note that if there is any over or underflow the types will fail to unify: - -```rust title="underflow-example" showLineNumbers -fn pop(array: [Field; N]) -> [Field; N - 1] { - let mut result: [Field; N - 1] = std::mem::zeroed(); - for i in 0..N - 1 { - result[i] = array[i]; - } - result -} - -fn main() { - // error: Could not determine array length `(0 - 1)` - pop([]); -} -``` -> Source code: test_programs/compile_failure/arithmetic_generics_underflow/src/main.nr#L1-L14 - - -This also applies if there is underflow in an intermediate calculation: - -```rust title="intermediate-underflow-example" showLineNumbers -fn main() { - // From main it looks like there's nothing sketchy going on - seems_fine([]); -} - -// Since `seems_fine` says it can receive and return any length N -fn seems_fine(array: [Field; N]) -> [Field; N] { - // But inside `seems_fine` we pop from the array which - // requires the length to be greater than zero. - - // error: Could not determine array length `(0 - 1)` - push_zero(pop(array)) -} - -fn pop(array: [Field; N]) -> [Field; N - 1] { - let mut result: [Field; N - 1] = std::mem::zeroed(); - for i in 0..N - 1 { - result[i] = array[i]; - } - result -} - -fn push_zero(array: [Field; N]) -> [Field; N + 1] { - let mut result: [Field; N + 1] = std::mem::zeroed(); - for i in 0..N { - result[i] = array[i]; - } - // index N is already zeroed - result -} -``` -> Source code: test_programs/compile_failure/arithmetic_generics_intermediate_underflow/src/main.nr#L1-L32 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/lambdas.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/lambdas.md deleted file mode 100644 index be3c7e0b5ca..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/lambdas.md +++ /dev/null @@ -1,81 +0,0 @@ ---- -title: Lambdas -description: Learn how to use anonymous functions in Noir programming language. -keywords: [Noir programming language, lambda, closure, function, anonymous function] -sidebar_position: 9 ---- - -## Introduction - -Lambdas are anonymous functions. The syntax is `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -A block can be used as the body of a lambda, allowing you to declare local variables inside it: - -```rust -let cool = || { - let x = 100; - let y = 100; - x + y -} - -assert(cool() == 200); -``` - -## Closures - -Inside the body of a lambda, you can use variables defined in the enclosing function. Such lambdas are called **closures**. In this example `x` is defined inside `main` and is accessed from within the lambda: - -```rust -fn main() { - let x = 100; - let closure = || x + 150; - assert(closure() == 250); -} -``` - -## Passing closures to higher-order functions - -It may catch you by surprise that the following code fails to compile: - -```rust -fn foo(f: fn () -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // error :( -} -``` - -The reason is that the closure's capture environment affects its type - we have a closure that captures two Fields and `foo` -expects a regular function as an argument - those are incompatible. -:::note - -Variables contained within the `||` are the closure's parameters, and the expression that follows it is the closure's body. The capture environment is comprised of any variables used in the closure's body that are not parameters. - -E.g. in |x| x + y, y would be a captured variable, but x would not be, since it is a parameter of the closure. - -::: -The syntax for the type of a closure is `fn[env](args) -> ret_type`, where `env` is the capture environment of the closure - -in this example that's `(Field, Field)`. - -The best solution in our case is to make `foo` generic over the environment type of its parameter, so that it can be called -with closures with any environment, as well as with regular functions: - -```rust -fn foo(f: fn[Env]() -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // compiles fine - assert(foo(|| 60) == 60); // compiles fine -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/mutability.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/mutability.md deleted file mode 100644 index fdeef6a87c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/mutability.md +++ /dev/null @@ -1,121 +0,0 @@ ---- -title: Mutability -description: - Learn about mutable variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, mutability in noir, mutable variables] -sidebar_position: 8 ---- - -Variables in noir can be declared mutable via the `mut` keyword. Mutable variables can be reassigned -to via an assignment expression. - -```rust -let x = 2; -x = 3; // error: x must be mutable to be assigned to - -let mut y = 3; -let y = 4; // OK -``` - -The `mut` modifier can also apply to patterns: - -```rust -let (a, mut b) = (1, 2); -a = 11; // error: a must be mutable to be assigned to -b = 12; // OK - -let mut (c, d) = (3, 4); -c = 13; // OK -d = 14; // OK - -// etc. -let MyStruct { x: mut y } = MyStruct { x: a }; -// y is now in scope -``` - -Note that mutability in noir is local and everything is passed by value, so if a called function -mutates its parameters then the parent function will keep the old value of the parameters. - -```rust -fn main() -> pub Field { - let x = 3; - helper(x); - x // x is still 3 -} - -fn helper(mut x: i32) { - x = 4; -} -``` - -## Non-local mutability - -Non-local mutability can be achieved through the mutable reference type `&mut T`: - -```rust -fn set_to_zero(x: &mut Field) { - *x = 0; -} - -fn main() { - let mut y = 42; - set_to_zero(&mut y); - assert(*y == 0); -} -``` - -When creating a mutable reference, the original variable being referred to (`y` in this -example) must also be mutable. Since mutable references are a reference type, they must -be explicitly dereferenced via `*` to retrieve the underlying value. Note that this yields -a copy of the value, so mutating this copy will not change the original value behind the -reference: - -```rust -fn main() { - let mut x = 1; - let x_ref = &mut x; - - let mut y = *x_ref; - let y_ref = &mut y; - - x = 2; - *x_ref = 3; - - y = 4; - *y_ref = 5; - - assert(x == 3); - assert(*x_ref == 3); - assert(y == 5); - assert(*y_ref == 5); -} -``` - -Note that types in Noir are actually deeply immutable so the copy that occurs when -dereferencing is only a conceptual copy - no additional constraints will occur. - -Mutable references can also be stored within structs. Note that there is also -no lifetime parameter on these unlike rust. This is because the allocated memory -always lasts the entire program - as if it were an array of one element. - -```rust -struct Foo { - x: &mut Field -} - -impl Foo { - fn incr(mut self) { - *self.x += 1; - } -} - -fn main() { - let foo = Foo { x: &mut 0 }; - foo.incr(); - assert(*foo.x == 1); -} -``` - -In general, you should avoid non-local & shared mutability unless it is needed. Sticking -to only local mutability will improve readability and potentially improve compiler optimizations as well. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/ops.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/ops.md deleted file mode 100644 index c35c36c38a9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/ops.md +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: Logical Operations -description: - Learn about the supported arithmetic and logical operations in the Noir programming language. - Discover how to perform operations on private input types, integers, and booleans. -keywords: - [ - Noir programming language, - supported operations, - arithmetic operations, - logical operations, - predicate operators, - bitwise operations, - short-circuiting, - backend, - ] -sidebar_position: 3 ---- - -# Operations - -## Table of Supported Operations - -| Operation | Description | Requirements | -| :-------- | :------------------------------------------------------------: | -------------------------------------: | -| + | Adds two private input types together | Types must be private input | -| - | Subtracts two private input types together | Types must be private input | -| \* | Multiplies two private input types together | Types must be private input | -| / | Divides two private input types together | Types must be private input | -| ^ | XOR two private input types together | Types must be integer | -| & | AND two private input types together | Types must be integer | -| \| | OR two private input types together | Types must be integer | -| \<\< | Left shift an integer by another integer amount | Types must be integer, shift must be u8 | -| >> | Right shift an integer by another integer amount | Types must be integer, shift must be u8 | -| ! | Bitwise not of a value | Type must be integer or boolean | -| \< | returns a bool if one value is less than the other | Upper bound must have a known bit size | -| \<= | returns a bool if one value is less than or equal to the other | Upper bound must have a known bit size | -| > | returns a bool if one value is more than the other | Upper bound must have a known bit size | -| >= | returns a bool if one value is more than or equal to the other | Upper bound must have a known bit size | -| == | returns a bool if one value is equal to the other | Both types must not be constants | -| != | returns a bool if one value is not equal to the other | Both types must not be constants | - -### Predicate Operators - -`<,<=, !=, == , >, >=` are known as predicate/comparison operations because they compare two values. -This differs from the operations such as `+` where the operands are used in _computation_. - -### Bitwise Operations Example - -```rust -fn main(x : Field) { - let y = x as u32; - let z = y & y; -} -``` - -`z` is implicitly constrained to be the result of `y & y`. The `&` operand is used to denote bitwise -`&`. - -> `x & x` would not compile as `x` is a `Field` and not an integer type. - -### Logical Operators - -Noir has no support for the logical operators `||` and `&&`. This is because encoding the -short-circuiting that these operators require can be inefficient for Noir's backend. Instead you can -use the bitwise operators `|` and `&` which operate identically for booleans, just without the -short-circuiting. - -```rust -let my_val = 5; - -let mut flag = 1; -if (my_val > 6) | (my_val == 0) { - flag = 0; -} -assert(flag == 1); - -if (my_val != 10) & (my_val < 50) { - flag = 0; -} -assert(flag == 0); -``` - -### Shorthand operators - -Noir shorthand operators for most of the above operators, namely `+=, -=, *=, /=, %=, &=, |=, ^=, <<=`, and `>>=`. These allow for more concise syntax. For example: - -```rust -let mut i = 0; -i = i + 1; -``` - -could be written as: - -```rust -let mut i = 0; -i += 1; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/oracles.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/oracles.mdx deleted file mode 100644 index 77a2ac1550a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/oracles.mdx +++ /dev/null @@ -1,29 +0,0 @@ ---- -title: Oracles -description: Dive into how Noir supports Oracles via RPC calls, and learn how to declare an Oracle in Noir with our comprehensive guide. -keywords: - - Noir - - Oracles - - RPC Calls - - Unconstrained Functions - - Programming - - Blockchain -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -Noir has support for Oracles via RPC calls. This means Noir will make an RPC call and use the return value for proof generation. - -Since Oracles are not resolved by Noir, they are [`unconstrained` functions](./unconstrained.md) - -You can declare an Oracle through the `#[oracle()]` flag. Example: - -```rust -#[oracle(get_number_sequence)] -unconstrained fn get_number_sequence(_size: Field) -> [Field] {} -``` - -The timeout for when using an external RPC oracle resolver can be set with the `NARGO_FOREIGN_CALL_TIMEOUT` environment variable. This timeout is in units of milliseconds. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/shadowing.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/shadowing.md deleted file mode 100644 index 5ce6130d201..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/shadowing.md +++ /dev/null @@ -1,44 +0,0 @@ ---- -title: Shadowing -sidebar_position: 12 ---- - -Noir allows for inheriting variables' values and re-declaring them with the same name similar to Rust, known as shadowing. - -For example, the following function is valid in Noir: - -```rust -fn main() { - let x = 5; - - { - let x = x * 2; - assert (x == 10); - } - - assert (x == 5); -} -``` - -In this example, a variable x is first defined with the value 5. - -The local scope that follows shadows the original x, i.e. creates a local mutable x based on the value of the original x. It is given a value of 2 times the original x. - -When we return to the main scope, x once again refers to just the original x, which stays at the value of 5. - -## Temporal mutability - -One way that shadowing is useful, in addition to ergonomics across scopes, is for temporarily mutating variables. - -```rust -fn main() { - let age = 30; - // age = age + 5; // Would error as `age` is immutable by default. - - let mut age = age + 5; // Temporarily mutates `age` with a new value. - - let age = age; // Locks `age`'s mutability again. - - assert (age == 35); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/traits.md deleted file mode 100644 index 9da00a77587..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/traits.md +++ /dev/null @@ -1,501 +0,0 @@ ---- -title: Traits -description: - Traits in Noir can be used to abstract out a common interface for functions across - several data types. -keywords: [noir programming language, traits, interfaces, generic, protocol] -sidebar_position: 14 ---- - -## Overview - -Traits in Noir are a useful abstraction similar to interfaces or protocols in other languages. Each trait defines -the interface of several methods contained within the trait. Types can then implement this trait by providing -implementations for these methods. For example in the program: - -```rust -struct Rectangle { - width: Field, - height: Field, -} - -impl Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -fn log_area(r: Rectangle) { - println(r.area()); -} -``` - -We have a function `log_area` to log the area of a `Rectangle`. Now how should we change the program if we want this -function to work on `Triangle`s as well?: - -```rust -struct Triangle { - width: Field, - height: Field, -} - -impl Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Making `log_area` generic over all types `T` would be invalid since not all types have an `area` method. Instead, we can -introduce a new `Area` trait and make `log_area` generic over all types `T` that implement `Area`: - -```rust -trait Area { - fn area(self) -> Field; -} - -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -We also need to explicitly implement `Area` for `Rectangle` and `Triangle`. We can do that by changing their existing -impls slightly. Note that the parameter types and return type of each of our `area` methods must match those defined -by the `Area` trait. - -```rust -impl Area for Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -impl Area for Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Now we have a working program that is generic over any type of Shape that is used! Others can even use this program -as a library with their own types - such as `Circle` - as long as they also implement `Area` for these types. - -## Where Clauses - -As seen in `log_area` above, when we want to create a function or method that is generic over any type that implements -a trait, we can add a where clause to the generic function. - -```rust -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -It is also possible to apply multiple trait constraints on the same variable at once by combining traits with the `+` -operator. Similarly, we can have multiple trait constraints by separating each with a comma: - -```rust -fn foo(elements: [T], thing: U) where - T: Default + Add + Eq, - U: Bar, -{ - let mut sum = T::default(); - - for element in elements { - sum += element; - } - - if sum == T::default() { - thing.bar(); - } -} -``` - -## Generic Implementations - -You can add generics to a trait implementation by adding the generic list after the `impl` keyword: - -```rust -trait Second { - fn second(self) -> Field; -} - -impl Second for (T, Field) { - fn second(self) -> Field { - self.1 - } -} -``` - -You can also implement a trait for every type this way: - -```rust -trait Debug { - fn debug(self); -} - -impl Debug for T { - fn debug(self) { - println(self); - } -} - -fn main() { - 1.debug(); -} -``` - -### Generic Trait Implementations With Where Clauses - -Where clauses can be placed on trait implementations themselves to restrict generics in a similar way. -For example, while `impl Foo for T` implements the trait `Foo` for every type, `impl Foo for T where T: Bar` -will implement `Foo` only for types that also implement `Bar`. This is often used for implementing generic types. -For example, here is the implementation for array equality: - -```rust -impl Eq for [T; let N: u32] where T: Eq { - // Test if two arrays have the same elements. - // Because both arrays must have length N, we know their lengths already match. - fn eq(self, other: Self) -> bool { - let mut result = true; - - for i in 0 .. self.len() { - // The T: Eq constraint is needed to call == on the array elements here - result &= self[i] == other[i]; - } - - result - } -} -``` - -Where clauses can also be placed on struct implementations. -For example, here is a method utilizing a generic type that implements the equality trait. - -```rust -struct Foo { - a: u32, - b: T, -} - -impl Foo where T: Eq { - fn eq(self, other: Self) -> bool { - (self.a == other.a) & self.b.eq(other.b) - } -} -``` - -## Generic Traits - -Traits themselves can also be generic by placing the generic arguments after the trait name. These generics are in -scope of every item within the trait. - -```rust -trait Into { - // Convert `self` to type `T` - fn into(self) -> T; -} -``` - -When implementing generic traits the generic arguments of the trait must be specified. This is also true anytime -when referencing a generic trait (e.g. in a `where` clause). - -```rust -struct MyStruct { - array: [Field; 2], -} - -impl Into<[Field; 2]> for MyStruct { - fn into(self) -> [Field; 2] { - self.array - } -} - -fn as_array(x: T) -> [Field; 2] - where T: Into<[Field; 2]> -{ - x.into() -} - -fn main() { - let array = [1, 2]; - let my_struct = MyStruct { array }; - - assert_eq(as_array(my_struct), array); -} -``` - -### Associated Types and Constants - -Traits also support associated types and constraints which can be thought of as additional generics that are referred to by name. - -Here's an example of a trait with an associated type `Foo` and a constant `Bar`: - -```rust -trait MyTrait { - type Foo; - - let Bar: u32; -} -``` - -Now when we're implementing `MyTrait` we also have to provide values for `Foo` and `Bar`: - -```rust -impl MyTrait for Field { - type Foo = i32; - - let Bar: u32 = 11; -} -``` - -Since associated constants can also be used in a type position, its values are limited to only other -expression kinds allowed in numeric generics. - -Note that currently all associated types and constants must be explicitly specified in a trait constraint. -If we leave out any, we'll get an error that we're missing one: - -```rust -// Error! Constraint is missing associated constant for `Bar` -fn foo(x: T) where T: MyTrait { - ... -} -``` - -Because all associated types and constants must be explicitly specified, they are essentially named generics, -although this is set to change in the future. Future versions of Noir will allow users to elide associated types -in trait constraints similar to Rust. When this is done, you may still refer to their value with the `::AssociatedType` -syntax: - -```rust -// Only valid in future versions of Noir: -fn foo(x: T) where T: MyTrait { - let _: ::Foo = ...; -} -``` - -The type as trait syntax is possible in Noir today but is less useful when each type must be explicitly specified anyway: - -```rust -fn foo(x: T) where T: MyTrait { - // Works, but could just use F directly - let _: >::Foo = ...; - - let _: F = ...; -} -``` - -## Trait Methods With No `self` - -A trait can contain any number of methods, each of which have access to the `Self` type which represents each type -that eventually implements the trait. Similarly, the `self` variable is available as well but is not required to be used. -For example, we can define a trait to create a default value for a type. This trait will need to return the `Self` type -but doesn't need to take any parameters: - -```rust -trait Default { - fn default() -> Self; -} -``` - -Implementing this trait can be done similarly to any other trait: - -```rust -impl Default for Field { - fn default() -> Field { - 0 - } -} - -struct MyType {} - -impl Default for MyType { - fn default() -> Field { - MyType {} - } -} -``` - -However, since there is no `self` parameter, we cannot call it via the method call syntax `object.method()`. -Instead, we'll need to refer to the function directly. This can be done either by referring to the -specific impl `MyType::default()` or referring to the trait itself `Default::default()`. In the later -case, type inference determines the impl that is selected. - -```rust -let my_struct = MyStruct::default(); - -let x: Field = Default::default(); -let result = x + Default::default(); -``` - -:::warning - -```rust -let _ = Default::default(); -``` - -If type inference cannot select which impl to use because of an ambiguous `Self` type, an impl will be -arbitrarily selected. This occurs most often when the result of a trait function call with no parameters -is unused. To avoid this, when calling a trait function with no `self` or `Self` parameters or return type, -always refer to it via the implementation type's namespace - e.g. `MyType::default()`. -This is set to change to an error in future Noir versions. - -::: - -## Default Method Implementations - -A trait can also have default implementations of its methods by giving a body to the desired functions. -Note that this body must be valid for all types that may implement the trait. As a result, the only -valid operations on `self` will be operations valid for any type or other operations on the trait itself. - -```rust -trait Numeric { - fn add(self, other: Self) -> Self; - - // Default implementation of double is (self + self) - fn double(self) -> Self { - self.add(self) - } -} -``` - -When implementing a trait with default functions, a type may choose to implement only the required functions: - -```rust -impl Numeric for Field { - fn add(self, other: Field) -> Field { - self + other - } -} -``` - -Or it may implement the optional methods as well: - -```rust -impl Numeric for u32 { - fn add(self, other: u32) -> u32 { - self + other - } - - fn double(self) -> u32 { - self * 2 - } -} -``` - -## Impl Specialization - -When implementing traits for a generic type it is possible to implement the trait for only a certain combination -of generics. This can be either as an optimization or because those specific generics are required to implement the trait. - -```rust -trait Sub { - fn sub(self, other: Self) -> Self; -} - -struct NonZero { - value: T, -} - -impl Sub for NonZero { - fn sub(self, other: Self) -> Self { - let value = self.value - other.value; - assert(value != 0); - NonZero { value } - } -} -``` - -## Overlapping Implementations - -Overlapping implementations are disallowed by Noir to ensure Noir's decision on which impl to select is never ambiguous. -This means if a trait `Foo` is already implemented -by a type `Bar` for all `T`, then we cannot also have a separate impl for `Bar` (or any other -type argument). Similarly, if there is an impl for all `T` such as `impl Debug for T`, we cannot create -any more impls to `Debug` for other types since it would be ambiguous which impl to choose for any given -method call. - -```rust -trait Trait {} - -// Previous impl defined here -impl Trait for (A, B) {} - -// error: Impl for type `(Field, Field)` overlaps with existing impl -impl Trait for (Field, Field) {} -``` - -## Trait Coherence - -Another restriction on trait implementations is coherence. This restriction ensures other crates cannot create -impls that may overlap with other impls, even if several unrelated crates are used as dependencies in the same -program. - -The coherence restriction is: to implement a trait, either the trait itself or the object type must be declared -in the crate the impl is in. - -In practice this often comes up when using types provided by libraries. If a library provides a type `Foo` that does -not implement a trait in the standard library such as `Default`, you may not `impl Default for Foo` in your own crate. -While restrictive, this prevents later issues or silent changes in the program if the `Foo` library later added its -own impl for `Default`. If you are a user of the `Foo` library in this scenario and need a trait not implemented by the -library your choices are to either submit a patch to the library or use the newtype pattern. - -### The Newtype Pattern - -The newtype pattern gets around the coherence restriction by creating a new wrapper type around the library type -that we cannot create `impl`s for. Since the new wrapper type is defined in our current crate, we can create -impls for any trait we need on it. - -```rust -struct Wrapper { - foo: some_library::Foo, -} - -impl Default for Wrapper { - fn default() -> Wrapper { - Wrapper { - foo: some_library::Foo::new(), - } - } -} -``` - -Since we have an impl for our own type, the behavior of this code will not change even if `some_library` is updated -to provide its own `impl Default for Foo`. The downside of this pattern is that it requires extra wrapping and -unwrapping of values when converting to and from the `Wrapper` and `Foo` types. - -### Trait Inheritance - -Sometimes, you might need one trait to use another trait’s functionality (like "inheritance" in some other languages). In this case, you can specify this relationship by listing any child traits after the parent trait's name and a colon. Now, whenever the parent trait is implemented it will require the child traits to be implemented as well. A parent trait is also called a "super trait." - -```rust -trait Person { - fn name(self) -> String; -} - -// Person is a supertrait of Student. -// Implementing Student requires you to also impl Person. -trait Student: Person { - fn university(self) -> String; -} - -trait Programmer { - fn fav_language(self) -> String; -} - -// CompSciStudent (computer science student) is a subtrait of both Programmer -// and Student. Implementing CompSciStudent requires you to impl both supertraits. -trait CompSciStudent: Programmer + Student { - fn git_username(self) -> String; -} -``` - -### Visibility - -By default, like functions, traits are private to the module they exist in. You can use `pub` -to make the trait public or `pub(crate)` to make it public to just its crate: - -```rust -// This trait is now public -pub trait Trait {} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/unconstrained.md deleted file mode 100644 index b5221b8d2dd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/unconstrained.md +++ /dev/null @@ -1,104 +0,0 @@ ---- -title: Unconstrained Functions -description: "Learn about what unconstrained functions in Noir are, how to use them and when you'd want to." - -keywords: [Noir programming language, unconstrained, open] -sidebar_position: 5 ---- - -Unconstrained functions are functions which do not constrain any of the included computation and allow for non-deterministic computation. - -## Why? - -Zero-knowledge (ZK) domain-specific languages (DSL) enable developers to generate ZK proofs from their programs by compiling code down to the constraints of an NP complete language (such as R1CS or PLONKish languages). However, the hard bounds of a constraint system can be very limiting to the functionality of a ZK DSL. - -Enabling a circuit language to perform unconstrained execution is a powerful tool. Said another way, unconstrained execution lets developers generate witnesses from code that does not generate any constraints. Being able to execute logic outside of a circuit is critical for both circuit performance and constructing proofs on information that is external to a circuit. - -Fetching information from somewhere external to a circuit can also be used to enable developers to improve circuit efficiency. - -A ZK DSL does not just prove computation, but proves that some computation was handled correctly. Thus, it is necessary that when we switch from performing some operation directly inside of a circuit to inside of an unconstrained environment that the appropriate constraints are still laid down elsewhere in the circuit. - -## Example - -An in depth example might help drive the point home. This example comes from the excellent [post](https://discord.com/channels/1113924620781883405/1124022445054111926/1128747641853972590) by Tom in the Noir Discord. - -Let's look at how we can optimize a function to turn a `u72` into an array of `u8`s. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u72 & 0xff) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 91 -Backend circuit size: 3619 -``` - -A lot of the operations in this function are optimized away by the compiler (all the bit-shifts turn into divisions by constants). However we can save a bunch of gates by casting to u8 a bit earlier. This automatically truncates the bit-shifted value to fit in a u8 which allows us to remove the AND against 0xff. This saves us ~480 gates in total. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 75 -Backend circuit size: 3143 -``` - -Those are some nice savings already but we can do better. This code is all constrained so we're proving every step of calculating out using num, but we don't actually care about how we calculate this, just that it's correct. This is where brillig comes in. - -It turns out that truncating a u72 into a u8 is hard to do inside a snark, each time we do as u8 we lay down 4 ACIR opcodes which get converted into multiple gates. It's actually much easier to calculate num from out than the other way around. All we need to do is multiply each element of out by a constant and add them all together, both relatively easy operations inside a snark. - -We can then run `u72_to_u8` as unconstrained brillig code in order to calculate out, then use that result in our constrained function and assert that if we were to do the reverse calculation we'd get back num. This looks a little like the below: - -```rust -fn main(num: u72) -> pub [u8; 8] { - let out = unsafe { - u72_to_u8(num) - }; - - let mut reconstructed_num: u72 = 0; - for i in 0..8 { - reconstructed_num += (out[i] as u72 << (56 - (8 * i))); - } - assert(num == reconstructed_num); - out -} - -unconstrained fn u72_to_u8(num: u72) -> [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8))) as u8; - } - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 78 -Backend circuit size: 2902 -``` - -This ends up taking off another ~250 gates from our circuit! We've ended up with more ACIR opcodes than before but they're easier for the backend to prove (resulting in fewer gates). - -Note that in order to invoke unconstrained functions we need to wrap them in an `unsafe` block, -to make it clear that the call is unconstrained. - -Generally we want to use brillig whenever there's something that's easy to verify but hard to compute within the circuit. For example, if you wanted to calculate a square root of a number it'll be a much better idea to calculate this in brillig and then assert that if you square the result you get back your number. - -## Break and Continue - -In addition to loops over runtime bounds, `break` and `continue` are also available in unconstrained code. See [break and continue](../concepts/control_flow.md#break-and-continue) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/_category_.json deleted file mode 100644 index 1debcfe7675..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Modules, Packages and Crates", - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/crates_and_packages.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/crates_and_packages.md deleted file mode 100644 index 95ee9f52ab2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/crates_and_packages.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Crates and Packages -description: Learn how to use Crates and Packages in your Noir project -keywords: [Nargo, dependencies, package management, crates, package] -sidebar_position: 0 ---- - -## Crates - -A crate is the smallest amount of code that the Noir compiler considers at a time. -Crates can contain modules, and the modules may be defined in other files that get compiled with the crate, as we’ll see in the coming sections. - -### Crate Types - -A Noir crate can come in several forms: binaries, libraries or contracts. - -#### Binaries - -_Binary crates_ are programs which you can compile to an ACIR circuit which you can then create proofs against. Each must have a function called `main` that defines the ACIR circuit which is to be proved. - -#### Libraries - -_Library crates_ don't have a `main` function and they don't compile down to ACIR. Instead they define functionality intended to be shared with multiple projects, and eventually included in a binary crate. - -#### Contracts - -Contract crates are similar to binary crates in that they compile to ACIR which you can create proofs against. They are different in that they do not have a single `main` function, but are a collection of functions to be deployed to the [Aztec network](https://aztec.network). You can learn more about the technical details of Aztec in the [monorepo](https://github.com/AztecProtocol/aztec-packages) or contract [examples](https://github.com/AztecProtocol/aztec-packages/tree/master/noir-projects/noir-contracts/contracts). - -### Crate Root - -Every crate has a root, which is the source file that the compiler starts, this is also known as the root module. The Noir compiler does not enforce any conditions on the name of the file which is the crate root, however if you are compiling via Nargo the crate root must be called `lib.nr` or `main.nr` for library or binary crates respectively. - -## Packages - -A Nargo _package_ is a collection of one of more crates that provides a set of functionality. A package must include a Nargo.toml file. - -A package _must_ contain either a library or a binary crate, but not both. - -### Differences from Cargo Packages - -One notable difference between Rust's Cargo and Noir's Nargo is that while Cargo allows a package to contain an unlimited number of binary crates and a single library crate, Nargo currently only allows a package to contain a single crate. - -In future this restriction may be lifted to allow a Nargo package to contain both a binary and library crate or multiple binary crates. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/dependencies.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/dependencies.md deleted file mode 100644 index 24e02de08fe..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/dependencies.md +++ /dev/null @@ -1,124 +0,0 @@ ---- -title: Dependencies -description: - Learn how to specify and manage dependencies in Nargo, allowing you to upload packages to GitHub - and use them easily in your project. -keywords: [Nargo, dependencies, GitHub, package management, versioning] -sidebar_position: 1 ---- - -Nargo allows you to upload packages to GitHub and use them as dependencies. - -## Specifying a dependency - -Specifying a dependency requires a tag to a specific commit and the git url to the url containing -the package. - -Currently, there are no requirements on the tag contents. If requirements are added, it would follow -semver 2.0 guidelines. - -> Note: Without a `tag` , there would be no versioning and dependencies would change each time you -> compile your project. - -For example, to add the [ecrecover-noir library](https://github.com/colinnielsen/ecrecover-noir) to your project, add it to `Nargo.toml`: - -```toml -# Nargo.toml - -[dependencies] -ecrecover = {tag = "v0.8.0", git = "https://github.com/colinnielsen/ecrecover-noir"} -``` - -If the module is in a subdirectory, you can define a subdirectory in your git repository, for example: - -```toml -# Nargo.toml - -[dependencies] -easy_private_token_contract = {tag ="v0.1.0-alpha62", git = "https://github.com/AztecProtocol/aztec-packages", directory = "noir-contracts/contracts/easy_private_token_contract"} -``` - -## Specifying a local dependency - -You can also specify dependencies that are local to your machine. - -For example, this file structure has a library and binary crate - -```tree -├── binary_crate -│   ├── Nargo.toml -│   └── src -│   └── main.nr -└── lib_a - ├── Nargo.toml - └── src - └── lib.nr -``` - -Inside of the binary crate, you can specify: - -```toml -# Nargo.toml - -[dependencies] -lib_a = { path = "../lib_a" } -``` - -## Importing dependencies - -You can import a dependency to a Noir file using the following syntax. For example, to import the -ecrecover-noir library and local lib_a referenced above: - -```rust -use ecrecover; -use lib_a; -``` - -You can also import only the specific parts of dependency that you want to use, like so: - -```rust -use std::hash::sha256; -use std::scalar_mul::fixed_base_embedded_curve; -``` - -Lastly, as demonstrated in the -[elliptic curve example](../standard_library/cryptographic_primitives/ec_primitives.md#examples), you -can import multiple items in the same line by enclosing them in curly braces: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; -``` - -We don't have a way to consume libraries from inside a [workspace](./workspaces.md) as external dependencies right now. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -## Dependencies of Dependencies - -Note that when you import a dependency, you also get access to all of the dependencies of that package. - -For example, the [phy_vector](https://github.com/resurgencelabs/phy_vector) library imports an [fraction](https://github.com/resurgencelabs/fraction) library. If you're importing the phy_vector library, then you can access the functions in fractions library like so: - -```rust -use phy_vector; - -fn main(x : Field, y : pub Field) { - //... - let f = phy_vector::fraction::toFraction(true, 2, 1); - //... -} -``` - -## Available Libraries - -Noir does not currently have an official package manager. You can find a list of available Noir libraries in the [awesome-noir repo here](https://github.com/noir-lang/awesome-noir#libraries). - -Some libraries that are available today include: - -- [Standard Library](https://github.com/noir-lang/noir/tree/master/noir_stdlib) - the Noir Standard Library -- [Ethereum Storage Proof Verification](https://github.com/aragonzkresearch/noir-trie-proofs) - a library that contains the primitives necessary for RLP decoding (in the form of look-up table construction) and Ethereum state and storage proof verification (or verification of any trie proof involving 32-byte long keys) -- [BigInt](https://github.com/shuklaayush/noir-bigint) - a library that provides a custom BigUint56 data type, allowing for computations on large unsigned integers -- [ECrecover](https://github.com/colinnielsen/ecrecover-noir/tree/main) - a library to verify an ECDSA signature and return the source Ethereum address -- [Sparse Merkle Tree Verifier](https://github.com/vocdoni/smtverifier-noir/tree/main) - a library for verification of sparse Merkle trees -- [Signed Int](https://github.com/resurgencelabs/signed_int) - a library for accessing a custom Signed Integer data type, allowing access to negative numbers on Noir -- [Fraction](https://github.com/resurgencelabs/fraction) - a library for accessing fractional number data type in Noir, allowing results that aren't whole numbers diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/workspaces.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/workspaces.md deleted file mode 100644 index 513497f12bf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/workspaces.md +++ /dev/null @@ -1,42 +0,0 @@ ---- -title: Workspaces -sidebar_position: 3 ---- - -Workspaces are a feature of nargo that allow you to manage multiple related Noir packages in a single repository. A workspace is essentially a group of related projects that share common build output directories and configurations. - -Each Noir project (with it's own Nargo.toml file) can be thought of as a package. Each package is expected to contain exactly one "named circuit", being the "name" defined in Nargo.toml with the program logic defined in `./src/main.nr`. - -For a project with the following structure: - -```tree -├── crates -│ ├── a -│ │ ├── Nargo.toml -│ │ └── Prover.toml -│ │ └── src -│ │ └── main.nr -│ └── b -│ ├── Nargo.toml -│ └── Prover.toml -│ └── src -│ └── main.nr -│ -└── Nargo.toml -``` - -You can define a workspace in Nargo.toml like so: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -`members` indicates which packages are included in the workspace. As such, all member packages of a workspace will be processed when the `--workspace` flag is used with various commands or if a `default-member` is not specified. - -`default-member` indicates which package various commands process by default. - -Libraries can be defined in a workspace. Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/_category_.json deleted file mode 100644 index af04c0933fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Standard Library", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/bigint.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/bigint.md deleted file mode 100644 index 05c3011634f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/bigint.md +++ /dev/null @@ -1,127 +0,0 @@ ---- -title: Big Integers -description: How to use big integers from Noir standard library -keywords: - [ - Big Integer, - Noir programming language, - Noir libraries, - ] ---- - -The BigInt module in the standard library exposes some class of integers which do not fit (well) into a Noir native field. It implements modulo arithmetic, modulo a 'big' prime number. - -:::note - -The module can currently be considered as `Field`s with fixed modulo sizes used by a set of elliptic curves, in addition to just the native curve. [More work](https://github.com/noir-lang/noir/issues/510) is needed to achieve arbitrarily sized big integers. - -:::note - -`nargo` can be built with `--profile release-pedantic` to enable extra overflow checks which may affect `BigInt` results in some cases. -Consider the [`noir-bignum`](https://github.com/noir-lang/noir-bignum) library for an optimized alternative approach. - -::: - -Currently 6 classes of integers (i.e 'big' prime numbers) are available in the module, namely: - -- BN254 Fq: Bn254Fq -- BN254 Fr: Bn254Fr -- Secp256k1 Fq: Secpk1Fq -- Secp256k1 Fr: Secpk1Fr -- Secp256r1 Fr: Secpr1Fr -- Secp256r1 Fq: Secpr1Fq - -Where XXX Fq and XXX Fr denote respectively the order of the base and scalar field of the (usual) elliptic curve XXX. -For instance the big integer 'Secpk1Fq' in the standard library refers to integers modulo $2^{256}-2^{32}-977$. - -Feel free to explore the source code for the other primes: - -```rust title="big_int_definition" showLineNumbers -pub struct BigInt { - pointer: u32, - modulus: u32, -} -``` -> Source code: noir_stdlib/src/bigint.nr#L28-L33 - - -## Example usage - -A common use-case is when constructing a big integer from its bytes representation, and performing arithmetic operations on it: - -```rust title="big_int_example" showLineNumbers -fn big_int_example(x: u8, y: u8) { - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - let b = Secpk1Fq::from_le_bytes(&[y, x, 9]); - let c = (a + b) * b / a; - let d = c.to_le_bytes(); - println(d[0]); -} -``` -> Source code: test_programs/execution_success/bigint/src/main.nr#L74-L82 - - -## Methods - -The available operations for each big integer are: - -### from_le_bytes - -Construct a big integer from its little-endian bytes representation. Example: - -```rust - // Construct a big integer from a slice of bytes - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - // Construct a big integer from an array of 32 bytes - let a = Secpk1Fq::from_le_bytes_32([1;32]); - ``` - -Sure, here's the formatted version of the remaining methods: - -### to_le_bytes - -Return the little-endian bytes representation of a big integer. Example: - -```rust -let bytes = a.to_le_bytes(); -``` - -### add - -Add two big integers. Example: - -```rust -let sum = a + b; -``` - -### sub - -Subtract two big integers. Example: - -```rust -let difference = a - b; -``` - -### mul - -Multiply two big integers. Example: - -```rust -let product = a * b; -``` - -### div - -Divide two big integers. Note that division is field division and not euclidean division. Example: - -```rust -let quotient = a / b; -``` - -### eq - -Compare two big integers. Example: - -```rust -let are_equal = a == b; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/black_box_fns.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/black_box_fns.md deleted file mode 100644 index d6079ab182c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/black_box_fns.md +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Black Box Functions -description: Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. -keywords: [noir, black box functions] ---- - -Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. This makes certain zk-snark unfriendly computations cheaper than if they were implemented in Noir. - -The ACVM spec defines a set of blackbox functions which backends will be expected to implement. This allows backends to use optimized implementations of these constraints if they have them, however they may also fallback to less efficient naive implementations if not. - -## Function list - -Here is a list of the current black box functions: - -- [AES128](./cryptographic_primitives/ciphers.mdx#aes128) -- [SHA256](./cryptographic_primitives/hashes.mdx#sha256) -- [Schnorr signature verification](./cryptographic_primitives/schnorr.mdx) -- [Blake2s](./cryptographic_primitives/hashes.mdx#blake2s) -- [Blake3](./cryptographic_primitives/hashes.mdx#blake3) -- [Pedersen Hash](./cryptographic_primitives/hashes.mdx#pedersen_hash) -- [Pedersen Commitment](./cryptographic_primitives/hashes.mdx#pedersen_commitment) -- [ECDSA signature verification](./cryptographic_primitives/ecdsa_sig_verification.mdx) -- [Embedded curve operations (MSM, addition, ...)](./cryptographic_primitives/embedded_curve_ops.mdx) -- AND -- XOR -- RANGE -- [Keccak256](./cryptographic_primitives/hashes.mdx#keccak256) -- [Recursive proof verification](./recursion.mdx) - -Most black box functions are included as part of the Noir standard library, however `AND`, `XOR` and `RANGE` are used as part of the Noir language syntax. For instance, using the bitwise operator `&` will invoke the `AND` black box function. - -You can view the black box functions defined in the ACVM code [here](https://github.com/noir-lang/noir/blob/master/acvm-repo/acir/src/circuit/black_box_functions.rs). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/bn254.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/bn254.md deleted file mode 100644 index 3294f005dbb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/bn254.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Bn254 Field Library ---- - -Noir provides a module in standard library with some optimized functions for bn254 Fr in `std::field::bn254`. - -## decompose - -```rust -fn decompose(x: Field) -> (Field, Field) {} -``` - -Decomposes a single field into two fields, low and high. The low field contains the lower 16 bytes of the input field and the high field contains the upper 16 bytes of the input field. Both field results are range checked to 128 bits. - - -## assert_gt - -```rust -fn assert_gt(a: Field, b: Field) {} -``` - -Asserts that a > b. This will generate less constraints than using `assert(gt(a, b))`. - -## assert_lt - -```rust -fn assert_lt(a: Field, b: Field) {} -``` - -Asserts that a < b. This will generate less constraints than using `assert(lt(a, b))`. - -## gt - -```rust -fn gt(a: Field, b: Field) -> bool {} -``` - -Returns true if a > b. - -## lt - -```rust -fn lt(a: Field, b: Field) -> bool {} -``` - -Returns true if a < b. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/hashmap.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/hashmap.md deleted file mode 100644 index 395cc312705..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/hashmap.md +++ /dev/null @@ -1,587 +0,0 @@ ---- -title: HashMap -keywords: [noir, map, hash, hashmap] -sidebar_position: 1 ---- - -`HashMap` is used to efficiently store and look up key-value pairs. - -`HashMap` is a bounded type which can store anywhere from zero to `MaxLen` total elements. -Note that due to hash collisions, the actual maximum number of elements stored by any particular -hashmap is likely lower than `MaxLen`. This is true even with cryptographic hash functions since -every hash value will be performed modulo `MaxLen`. - -Example: - -```rust -// Create a mapping from Fields to u32s with a maximum length of 12 -// using a poseidon2 hasher -use std::hash::poseidon2::Poseidon2Hasher; -let mut map: HashMap> = HashMap::default(); - -map.insert(1, 2); -map.insert(3, 4); - -let two = map.get(1).unwrap(); -``` - -## Methods - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default, -{ - /// Constructs an empty HashMap. - /// - /// Example: - /// - /// ```noir - /// let hashmap: HashMap> = HashMap::default(); - /// assert(hashmap.is_empty()); - /// ``` - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L681-L696 - - -Creates a fresh, empty HashMap. - -When using this function, always make sure to specify the maximum size of the hash map. - -This is the same `default` from the `Default` implementation given further below. It is -repeated here for convenience since it is the recommended way to create a hashmap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L207-L210 - - -Because `HashMap` has so many generic arguments that are likely to be the same throughout -your program, it may be helpful to create a type alias: - -```rust title="type_alias" showLineNumbers -type MyMap = HashMap>; -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L203 - - -### with_hasher - -```rust title="with_hasher" showLineNumbers -pub fn with_hasher(_build_hasher: B) -> Self - where - B: BuildHasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L103-L108 - - -Creates a hashmap with an existing `BuildHasher`. This can be used to ensure multiple -hashmaps are created with the same hasher instance. - -Example: - -```rust title="with_hasher_example" showLineNumbers -let my_hasher: BuildHasherDefault = Default::default(); - let hashmap: HashMap> = - HashMap::with_hasher(my_hasher); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L211-L216 - - -### get - -```rust title="get" showLineNumbers -pub fn get(self, key: K) -> Option - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L465-L472 - - -Retrieves a value from the hashmap, returning `Option::none()` if it was not found. - -Example: - -```rust title="get_example" showLineNumbers -fn get_example(map: HashMap>) { - let x = map.get(12); - - if x.is_some() { - assert(x.unwrap() == 42); - } -} -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L296-L304 - - -### insert - -```rust title="insert" showLineNumbers -pub fn insert(&mut self, key: K, value: V) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L507-L514 - - -Inserts a new key-value pair into the map. If the key was already in the map, its -previous value will be overridden with the newly provided one. - -Example: - -```rust title="insert_example" showLineNumbers -let mut map: HashMap> = HashMap::default(); - map.insert(12, 42); - assert(map.len() == 1); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L217-L221 - - -### remove - -```rust title="remove" showLineNumbers -pub fn remove(&mut self, key: K) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L563-L570 - - -Removes the given key-value pair from the map. If the key was not already present -in the map, this does nothing. - -Example: - -```rust title="remove_example" showLineNumbers -map.remove(12); - assert(map.is_empty()); - - // If a key was not present in the map, remove does nothing - map.remove(12); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L224-L231 - - -### is_empty - -```rust title="is_empty" showLineNumbers -pub fn is_empty(self) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L167-L169 - - -True if the length of the hash map is empty. - -Example: - -```rust title="is_empty_example" showLineNumbers -assert(map.is_empty()); - - map.insert(1, 2); - assert(!map.is_empty()); - - map.remove(1); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L232-L240 - - -### len - -```rust title="len" showLineNumbers -pub fn len(self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L424-L426 - - -Returns the current length of this hash map. - -Example: - -```rust title="len_example" showLineNumbers -// This is equivalent to checking map.is_empty() - assert(map.len() == 0); - - map.insert(1, 2); - map.insert(3, 4); - map.insert(5, 6); - assert(map.len() == 3); - - // 3 was already present as a key in the hash map, so the length is unchanged - map.insert(3, 7); - assert(map.len() == 3); - - map.remove(1); - assert(map.len() == 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L241-L256 - - -### capacity - -```rust title="capacity" showLineNumbers -pub fn capacity(_self: Self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L446-L448 - - -Returns the maximum capacity of this hashmap. This is always equal to the capacity -specified in the hashmap's type. - -Unlike hashmaps in general purpose programming languages, hashmaps in Noir have a -static capacity that does not increase as the map grows larger. Thus, this capacity -is also the maximum possible element count that can be inserted into the hashmap. -Due to hash collisions (modulo the hashmap length), it is likely the actual maximum -element count will be lower than the full capacity. - -Example: - -```rust title="capacity_example" showLineNumbers -let empty_map: HashMap> = - HashMap::default(); - assert(empty_map.len() == 0); - assert(empty_map.capacity() == 42); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L257-L262 - - -### clear - -```rust title="clear" showLineNumbers -pub fn clear(&mut self) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L123-L125 - - -Clears the hashmap, removing all key-value pairs from it. - -Example: - -```rust title="clear_example" showLineNumbers -assert(!map.is_empty()); - map.clear(); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L263-L267 - - -### contains_key - -```rust title="contains_key" showLineNumbers -pub fn contains_key(self, key: K) -> bool - where - K: Hash + Eq, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L143-L150 - - -True if the hashmap contains the given key. Unlike `get`, this will not also return -the value associated with the key. - -Example: - -```rust title="contains_key_example" showLineNumbers -if map.contains_key(7) { - let value = map.get(7); - assert(value.is_some()); - } else { - println("No value for key 7!"); - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L268-L275 - - -### entries - -```rust title="entries" showLineNumbers -pub fn entries(self) -> BoundedVec<(K, V), N> { -``` -> Source code: noir_stdlib/src/collections/map.nr#L191-L193 - - -Returns a vector of each key-value pair present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="entries_example" showLineNumbers -let entries = map.entries(); - - // The length of a hashmap may not be compile-time known, so we - // need to loop over its capacity instead - for i in 0..map.capacity() { - if i < entries.len() { - let (key, value) = entries.get(i); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L307-L318 - - -### keys - -```rust title="keys" showLineNumbers -pub fn keys(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L227-L229 - - -Returns a vector of each key present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="keys_example" showLineNumbers -let keys = map.keys(); - - for i in 0..keys.max_len() { - if i < keys.len() { - let key = keys.get_unchecked(i); - let value = map.get(key).unwrap_unchecked(); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L319-L329 - - -### values - -```rust title="values" showLineNumbers -pub fn values(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L262-L264 - - -Returns a vector of each value present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="values_example" showLineNumbers -let values = map.values(); - - for i in 0..values.max_len() { - if i < values.len() { - let value = values.get_unchecked(i); - println(f"Found value {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L330-L339 - - -### iter_mut - -```rust title="iter_mut" showLineNumbers -pub fn iter_mut(&mut self, f: fn(K, V) -> (K, V)) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L297-L304 - - -Iterates through each key-value pair of the HashMap, setting each key-value pair to the -result returned from the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If this is not desired, use `iter_values_mut` if only values need to be mutated, -or `entries` if neither keys nor values need to be mutated. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_mut_example" showLineNumbers -// Add 1 to each key in the map, and double the value associated with that key. - map.iter_mut(|k, v| (k + 1, v * 2)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L343-L346 - - -### iter_keys_mut - -```rust title="iter_keys_mut" showLineNumbers -pub fn iter_keys_mut(&mut self, f: fn(K) -> K) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L335-L342 - - -Iterates through the HashMap, mutating each key to the result returned from -the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If only iteration is desired and the keys are not intended to be mutated, -prefer using `entries` instead. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_keys_mut_example" showLineNumbers -// Double each key, leaving the value associated with that key untouched - map.iter_keys_mut(|k| k * 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L347-L350 - - -### iter_values_mut - -```rust title="iter_values_mut" showLineNumbers -pub fn iter_values_mut(&mut self, f: fn(V) -> V) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L367-L369 - - -Iterates through the HashMap, applying the given function to each value and mutating the -value to equal the result. This function is more efficient than `iter_mut` and `iter_keys_mut` -because the keys are untouched and the underlying hashmap thus does not need to be reordered. - -Example: - -```rust title="iter_values_mut_example" showLineNumbers -// Halve each value - map.iter_values_mut(|v| v / 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L351-L354 - - -### retain - -```rust title="retain" showLineNumbers -pub fn retain(&mut self, f: fn(K, V) -> bool) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L388-L390 - - -Retains only the key-value pairs for which the given function returns true. -Any key-value pairs for which the function returns false will be removed from the map. - -Example: - -```rust title="retain_example" showLineNumbers -map.retain(|k, v| (k != 0) & (v != 0)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L279-L281 - - -## Trait Implementations - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default, -{ - /// Constructs an empty HashMap. - /// - /// Example: - /// - /// ```noir - /// let hashmap: HashMap> = HashMap::default(); - /// assert(hashmap.is_empty()); - /// ``` - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L681-L696 - - -Constructs an empty HashMap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L207-L210 - - -### eq - -```rust title="eq" showLineNumbers -impl Eq for HashMap -where - K: Eq + Hash, - V: Eq, - B: BuildHasher, - H: Hasher, -{ - /// Checks if two HashMaps are equal. - /// - /// Example: - /// - /// ```noir - /// let mut map1: HashMap> = HashMap::default(); - /// let mut map2: HashMap> = HashMap::default(); - /// - /// map1.insert(1, 2); - /// map1.insert(3, 4); - /// - /// map2.insert(3, 4); - /// map2.insert(1, 2); - /// - /// assert(map1 == map2); - /// ``` - fn eq(self, other: HashMap) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L629-L654 - - -Checks if two HashMaps are equal. - -Example: - -```rust title="eq_example" showLineNumbers -let mut map1: HashMap> = HashMap::default(); - let mut map2: HashMap> = HashMap::default(); - - map1.insert(1, 2); - map1.insert(3, 4); - - map2.insert(3, 4); - map2.insert(1, 2); - - assert(map1 == map2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L282-L293 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/index.md deleted file mode 100644 index ea84c6d5c21..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/index.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -title: Containers -description: Container types provided by Noir's standard library for storing and retrieving data -keywords: [containers, data types, vec, hashmap] ---- diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/vec.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/vec.mdx deleted file mode 100644 index 475011922f8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/vec.mdx +++ /dev/null @@ -1,170 +0,0 @@ ---- -title: Vectors -description: Delve into the Vec data type in Noir. Learn about its methods, practical examples, and best practices for using Vectors in your Noir code. -keywords: [noir, vector type, methods, examples, dynamic arrays] -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A vector is a collection type similar to Rust's `Vec` type. In Noir, it is a convenient way to use slices as mutable arrays. - -Example: - -```rust -let mut vector: Vec = Vec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -``` - -## Methods - -### new - -Creates a new, empty vector. - -```rust -pub fn new() -> Self -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### from_slice - -Creates a vector containing each element from a given slice. Mutations to the resulting vector will not affect the original slice. - -```rust -pub fn from_slice(slice: [T]) -> Self -``` - -Example: - -```rust -let slice: [Field] = &[1, 2, 3]; -let vector_from_slice = Vec::from_slice(slice); -assert(vector_from_slice.len() == 3); -``` - -### len - -Returns the number of elements in the vector. - -```rust -pub fn len(self) -> Field -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### get - -Retrieves an element from the vector at a given index. Panics if the index points beyond the vector's end. - -```rust -pub fn get(self, index: Field) -> T -``` - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -``` - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -Panics if the index points beyond the vector's end. - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -vector.set(1, 42); -assert(vector.get(1) == 42); -``` - -### push - -Adds a new element to the vector's end, returning a new vector with a length one greater than the original unmodified vector. - -```rust -pub fn push(&mut self, elem: T) -``` - -Example: - -```rust -let mut vector: Vec = Vec::new(); -vector.push(10); -assert(vector.len() == 1); -``` - -### pop - -Removes an element from the vector's end, returning a new vector with a length one less than the original vector, along with the removed element. Panics if the vector's length is zero. - -```rust -pub fn pop(&mut self) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20]); -let popped_elem = vector.pop(); -assert(popped_elem == 20); -assert(vector.len() == 1); -``` - -### insert - -Inserts an element at a specified index, shifting subsequent elements to the right. - -```rust -pub fn insert(&mut self, index: Field, elem: T) -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 30]); -vector.insert(1, 20); -assert(vector.get(1) == 20); -``` - -### remove - -Removes an element at a specified index, shifting subsequent elements to the left, and returns the removed element. - -```rust -pub fn remove(&mut self, index: Field) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20, 30]); -let removed_elem = vector.remove(1); -assert(removed_elem == 20); -assert(vector.len() == 2); -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/ciphers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/ciphers.mdx deleted file mode 100644 index d6a5e1a79eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/ciphers.mdx +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Ciphers -description: - Learn about the implemented ciphers ready to use for any Noir project -keywords: - [ciphers, Noir project, aes128, encrypt] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## aes128 - -Given a plaintext as an array of bytes, returns the corresponding aes128 ciphertext (CBC mode). Input padding is automatically performed using PKCS#7, so that the output length is `input.len() + (16 - input.len() % 16)`. - -```rust title="aes128" showLineNumbers -pub fn aes128_encrypt(input: [u8; N], iv: [u8; 16], key: [u8; 16]) -> [u8] {} -``` -> Source code: noir_stdlib/src/aes128.nr#L2-L4 - - -```rust -fn main() { - let input: [u8; 4] = [0, 12, 3, 15] // Random bytes, will be padded to 16 bytes. - let iv: [u8; 16] = [0; 16]; // Initialisation vector - let key: [u8; 16] = [0; 16] // AES key - let ciphertext = std::aes128::aes128_encrypt(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()); // In this case, the output length will be 16 bytes. -} -``` - - - \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/ec_primitives.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/ec_primitives.md deleted file mode 100644 index f262d8160d6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/ec_primitives.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Elliptic Curve Primitives -keywords: [cryptographic primitives, Noir project] -sidebar_position: 4 ---- - -Data structures and methods on them that allow you to carry out computations involving elliptic -curves over the (mathematical) field corresponding to `Field`. For the field currently at our -disposal, applications would involve a curve embedded in BN254, e.g. the -[Baby Jubjub curve](https://eips.ethereum.org/EIPS/eip-2494). - -## Data structures - -### Elliptic curve configurations - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Curve`), i.e. the specific elliptic -curve you want to use, which would be specified using any one of the methods -`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::new` which take the coefficients in the -defining equation together with a generator point as parameters. You can find more detail in the -comments in -[`noir_stdlib/src/ec/mod.nr`](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr), but -the gist of it is that the elliptic curves of interest are usually expressed in one of the standard -forms implemented here (Twisted Edwards, Montgomery and Short Weierstraß), and in addition to that, -you could choose to use `affine` coordinates (Cartesian coordinates - the usual (x,y) - possibly -together with a point at infinity) or `curvegroup` coordinates (some form of projective coordinates -requiring more coordinates but allowing for more efficient implementations of elliptic curve -operations). Conversions between all of these forms are provided, and under the hood these -conversions are done whenever an operation is more efficient in a different representation (or a -mixed coordinate representation is employed). - -### Points - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Point`), i.e. points lying on the -elliptic curve. For a curve configuration `c` and a point `p`, it may be checked that `p` -does indeed lie on `c` by calling `c.contains(p1)`. - -## Methods - -(given a choice of curve representation, e.g. use `std::ec::tecurve::affine::Curve` and use -`std::ec::tecurve::affine::Point`) - -- The **zero element** is given by `Point::zero()`, and we can verify whether a point `p: Point` is - zero by calling `p.is_zero()`. -- **Equality**: Points `p1: Point` and `p2: Point` may be checked for equality by calling - `p1.eq(p2)`. -- **Addition**: For `c: Curve` and points `p1: Point` and `p2: Point` on the curve, adding these two - points is accomplished by calling `c.add(p1,p2)`. -- **Negation**: For a point `p: Point`, `p.negate()` is its negation. -- **Subtraction**: For `c` and `p1`, `p2` as above, subtracting `p2` from `p1` is accomplished by - calling `c.subtract(p1,p2)`. -- **Scalar multiplication**: For `c` as above, `p: Point` a point on the curve and `n: Field`, - scalar multiplication is given by `c.mul(n,p)`. If instead `n :: [u1; N]`, i.e. `n` is a bit - array, the `bit_mul` method may be used instead: `c.bit_mul(n,p)` -- **Multi-scalar multiplication**: For `c` as above and arrays `n: [Field; N]` and `p: [Point; N]`, - multi-scalar multiplication is given by `c.msm(n,p)`. -- **Coordinate representation conversions**: The `into_group` method converts a point or curve - configuration in the affine representation to one in the CurveGroup representation, and - `into_affine` goes in the other direction. -- **Curve representation conversions**: `tecurve` and `montcurve` curves and points are equivalent - and may be converted between one another by calling `into_montcurve` or `into_tecurve` on their - configurations or points. `swcurve` is more general and a curve c of one of the other two types - may be converted to this representation by calling `c.into_swcurve()`, whereas a point `p` lying - on the curve given by `c` may be mapped to its corresponding `swcurve` point by calling - `c.map_into_swcurve(p)`. -- **Map-to-curve methods**: The Elligator 2 method of mapping a field element `n: Field` into a - `tecurve` or `montcurve` with configuration `c` may be called as `c.elligator2_map(n)`. For all of - the curve configurations, the SWU map-to-curve method may be called as `c.swu_map(z,n)`, where - `z: Field` depends on `Field` and `c` and must be chosen by the user (the conditions it needs to - satisfy are specified in the comments - [here](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr)). - -## Examples - -The -[ec_baby_jubjub test](https://github.com/noir-lang/noir/blob/master/test_programs/compile_success_empty/ec_baby_jubjub/src/main.nr) -illustrates all of the above primitives on various forms of the Baby Jubjub curve. A couple of more -interesting examples in Noir would be: - -Public-key cryptography: Given an elliptic curve and a 'base point' on it, determine the public key -from the private key. This is a matter of using scalar multiplication. In the case of Baby Jubjub, -for example, this code would do: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; - -fn bjj_pub_key(priv_key: Field) -> Point -{ - - let bjj = Curve::new(168700, 168696, G::new(995203441582195749578291179787384436505546430278305826713579947235728471134,5472060717959818805561601436314318772137091100104008585924551046643952123905)); - - let base_pt = Point::new(5299619240641551281634865583518297030282874472190772894086521144482721001553, 16950150798460657717958625567821834550301663161624707787222815936182638968203); - - bjj.mul(priv_key,base_pt) -} -``` - -This would come in handy in a Merkle proof. - -- EdDSA signature verification: This is a matter of combining these primitives with a suitable hash - function. See - [feat(stdlib): EdDSA sig verification noir#1136](https://github.com/noir-lang/noir/pull/1136) for - the case of Baby Jubjub and the Poseidon hash function. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/eddsa.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/eddsa.mdx deleted file mode 100644 index b283de693c8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/eddsa.mdx +++ /dev/null @@ -1,37 +0,0 @@ ---- -title: EdDSA Verification -description: Learn about the cryptographic primitives regarding EdDSA -keywords: [cryptographic primitives, Noir project, eddsa, signatures] -sidebar_position: 5 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## eddsa::eddsa_poseidon_verify - -Verifier for EdDSA signatures - -```rust -fn eddsa_poseidon_verify(public_key_x : Field, public_key_y : Field, signature_s: Field, signature_r8_x: Field, signature_r8_y: Field, message: Field) -> bool -``` - -It is also possible to specify the hash algorithm used for the signature by using the `eddsa_verify` function by passing a type implementing the Hasher trait with the turbofish operator. -For instance, if you want to use Poseidon2 instead, you can do the following: -```rust -use std::hash::poseidon2::Poseidon2Hasher; - -eddsa_verify::(pub_key_a.x, pub_key_a.y, s_a, r8_a.x, r8_a.y, msg); -``` - - - -## eddsa::eddsa_to_pub - -Private to public key conversion. - -Returns `(pub_key_x, pub_key_y)` - -```rust -fn eddsa_to_pub(secret : Field) -> (Field, Field) -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/index.md deleted file mode 100644 index 650f30165d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/index.md +++ /dev/null @@ -1,14 +0,0 @@ ---- -title: Cryptographic Primitives -description: - Learn about the cryptographic primitives ready to use for any Noir project -keywords: - [ - cryptographic primitives, - Noir project, - ] ---- - -The Noir team is progressively adding new cryptographic primitives to the standard library. Reach out for news or if you would be interested in adding more of these calculations in Noir. - -Some methods are available thanks to the Aztec backend, not being performed using Noir. When using other backends, these methods may or may not be supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/schnorr.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/schnorr.mdx deleted file mode 100644 index 030452645c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/schnorr.mdx +++ /dev/null @@ -1,64 +0,0 @@ ---- -title: Schnorr Signatures -description: Learn how you can verify Schnorr signatures using Noir -keywords: [cryptographic primitives, Noir project, schnorr, signatures] -sidebar_position: 2 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## schnorr::verify_signature - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin). -See schnorr::verify_signature_slice for a version that works directly on slices. - -```rust title="schnorr_verify" showLineNumbers -pub fn verify_signature( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8; N], -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L4-L11 - - -where `_signature` can be generated like so using the npm package -[@noir-lang/barretenberg](https://www.npmjs.com/package/@noir-lang/barretenberg) - -```js -const { BarretenbergWasm } = require('@noir-lang/barretenberg/dest/wasm'); -const { Schnorr } = require('@noir-lang/barretenberg/dest/crypto/schnorr'); - -... - -const barretenberg = await BarretenbergWasm.new(); -const schnorr = new Schnorr(barretenberg); -const pubKey = schnorr.computePublicKey(privateKey); -const message = ... -const signature = Array.from( - schnorr.constructSignature(hash, privateKey).toBuffer() -); - -... -``` - - - -## schnorr::verify_signature_slice - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin) -where the message is a slice. - -```rust title="schnorr_verify_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8], -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L15-L22 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/is_unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/is_unconstrained.md deleted file mode 100644 index 51bb1bda8f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/is_unconstrained.md +++ /dev/null @@ -1,69 +0,0 @@ ---- -title: Is Unconstrained Function -description: - The is_unconstrained function returns wether the context at that point of the program is unconstrained or not. -keywords: - [ - unconstrained - ] ---- - -It's very common for functions in circuits to take unconstrained hints of an expensive computation and then verify it. This is done by running the hint in an unconstrained context and then verifying the result in a constrained context. - -When a function is marked as unconstrained, any subsequent functions that it calls will also be run in an unconstrained context. However, if we are implementing a library function, other users might call it within an unconstrained context or a constrained one. Generally, in an unconstrained context we prefer just computing the result instead of taking a hint of it and verifying it, since that'd mean doing the same computation twice: - -```rust - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - my_expensive_computation_hint(); - // verify my_expensive_computation: If external_interface is called from unconstrained, this is redundant - ... -} - -``` - -In order to improve the performance in an unconstrained context you can use the function at `std::runtime::is_unconstrained() -> bool`: - - -```rust -use dep::std::runtime::is_unconstrained; - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - if is_unconstrained() { - my_expensive_computation(); - } else { - my_expensive_computation_hint(); - // verify my_expensive_computation - ... - } -} - -``` - -The is_unconstrained result is resolved at compile time, so in unconstrained contexts the compiler removes the else branch, and in constrained contexts the compiler removes the if branch, reducing the amount of compute necessary to run external_interface. - -Note that using `is_unconstrained` in a `comptime` context will also return `true`: - -``` -fn main() { - comptime { - assert(is_unconstrained()); - } -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/logging.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/logging.md deleted file mode 100644 index db75ef9f86f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/logging.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Logging -description: - Learn how to use the println statement for debugging in Noir with this tutorial. Understand the - basics of logging in Noir and how to implement it in your code. -keywords: - [ - noir logging, - println statement, - print statement, - debugging in noir, - noir std library, - logging tutorial, - basic logging in noir, - noir logging implementation, - noir debugging techniques, - rust, - ] ---- - -The standard library provides two familiar statements you can use: `println` and `print`. Despite being a limited implementation of rust's `println!` and `print!` macros, these constructs can be useful for debugging. - -You can print the output of both statements in your Noir code by using the `nargo execute` command or the `--show-output` flag when using `nargo test` (provided there are print statements in your tests). - -It is recommended to use `nargo execute` if you want to debug failing constraints with `println` or `print` statements. This is due to every input in a test being a constant rather than a witness, so we issue an error during compilation while we only print during execution (which comes after compilation). Neither `println`, nor `print` are callable for failed constraints caught at compile time. - -Both `print` and `println` are generic functions which can work on integers, fields, strings, and even structs or expressions. Note however, that slices are currently unsupported. For example: - -```rust -struct Person { - age: Field, - height: Field, -} - -fn main(age: Field, height: Field) { - let person = Person { - age: age, - height: height, - }; - println(person); - println(age + height); - println("Hello world!"); -} -``` - -You can print different types in the same statement (including strings) with a type called `fmtstr`. It can be specified in the same way as a normal string, just prepended with an "f" character: - -```rust - let fmt_str = f"i: {i}, j: {j}"; - println(fmt_str); - - let s = myStruct { y: x, x: y }; - println(s); - - println(f"i: {i}, s: {s}"); - - println(x); - println([x, y]); - - let foo = fooStruct { my_struct: s, foo: 15 }; - println(f"s: {s}, foo: {foo}"); - - println(15); // prints 0x0f, implicit Field - println(-1 as u8); // prints 255 - println(-1 as i8); // prints -1 -``` - -Examples shown above are interchangeable between the two `print` statements: - -```rust -let person = Person { age : age, height : height }; - -println(person); -print(person); - -println("Hello world!"); // Prints with a newline at the end of the input -print("Hello world!"); // Prints the input and keeps cursor on the same line -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/merkle_trees.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/merkle_trees.md deleted file mode 100644 index 6a9ebf72ada..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/merkle_trees.md +++ /dev/null @@ -1,58 +0,0 @@ ---- -title: Merkle Trees -description: Learn about Merkle Trees in Noir with this tutorial. Explore the basics of computing a merkle root using a proof, with examples. -keywords: - [ - Merkle trees in Noir, - Noir programming language, - check membership, - computing root from leaf, - Noir Merkle tree implementation, - Merkle tree tutorial, - Merkle tree code examples, - Noir libraries, - pedersen hash., - ] ---- - -## compute_merkle_root - -Returns the root of the tree from the provided leaf and its hash path, using a [Pedersen hash](./cryptographic_primitives/hashes.mdx#pedersen_hash). - -```rust -fn compute_merkle_root(leaf : Field, index : Field, hash_path: [Field]) -> Field -``` - -example: - -```rust -/** - // these values are for this example only - index = "0" - priv_key = "0x000000000000000000000000000000000000000000000000000000616c696365" - secret = "0x1929ea3ab8d9106a899386883d9428f8256cfedb3c4f6b66bf4aa4d28a79988f" - note_hash_path = [ - "0x1e61bdae0f027b1b2159e1f9d3f8d00fa668a952dddd822fda80dc745d6f65cc", - "0x0e4223f3925f98934393c74975142bd73079ab0621f4ee133cee050a3c194f1a", - "0x2fd7bb412155bf8693a3bd2a3e7581a679c95c68a052f835dddca85fa1569a40" - ] - */ -fn main(index: Field, priv_key: Field, secret: Field, note_hash_path: [Field; 3]) { - - let pubkey = std::scalar_mul::fixed_base_embedded_curve(priv_key); - let pubkey_x = pubkey[0]; - let pubkey_y = pubkey[1]; - let note_commitment = std::hash::pedersen(&[pubkey_x, pubkey_y, secret]); - - let root = std::merkle::compute_merkle_root(note_commitment[0], index, note_hash_path.as_slice()); - println(root); -} -``` - -To check merkle tree membership: - -1. Include a merkle root as a program input. -2. Compute the merkle root of a given leaf, index and hash path. -3. Assert the merkle roots are equal. - -For more info about merkle trees, see the Wikipedia [page](https://en.wikipedia.org/wiki/Merkle_tree). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/trait_constraint.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/trait_constraint.md deleted file mode 100644 index 3106f732b5a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/trait_constraint.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: TraitConstraint ---- - -`std::meta::trait_constraint` contains methods on the built-in `TraitConstraint` type which represents -a trait constraint that can be used to search for a trait implementation. This is similar -syntactically to just the trait itself, but can also contain generic arguments. E.g. `Eq`, `Default`, -`BuildHasher`. - -This type currently has no public methods but it can be used alongside `Type` in `implements` or `get_trait_impl`. - -## Trait Implementations - -```rust -impl Eq for TraitConstraint -impl Hash for TraitConstraint -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/typ.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/typ.md deleted file mode 100644 index 90222c222f5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/typ.md +++ /dev/null @@ -1,264 +0,0 @@ ---- -title: Type ---- - -`std::meta::typ` contains methods on the built-in `Type` type used for representing -a type in the source program. - -## Functions - -```rust title="fresh_type_variable" showLineNumbers -pub comptime fn fresh_type_variable() -> Type {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L57-L59 - - -Creates and returns an unbound type variable. This is a special kind of type internal -to type checking which will type check with any other type. When it is type checked -against another type it will also be set to that type. For example, if `a` is a type -variable and we have the type equality `(a, i32) = (u8, i32)`, the compiler will set -`a` equal to `u8`. - -Unbound type variables will often be rendered as `_` while printing them. Bound type -variables will appear as the type they are bound to. - -This can be used in conjunction with functions which internally perform type checks -such as `Type::implements` or `Type::get_trait_impl` to potentially grab some of the types used. - -Note that calling `Type::implements` or `Type::get_trait_impl` on a type variable will always -fail. - -Example: - -```rust title="serialize-setup" showLineNumbers -trait Serialize {} - -impl Serialize<1> for Field {} - -impl Serialize for [T; N] -where - T: Serialize, -{} - -impl Serialize for (T, U) -where - T: Serialize, - U: Serialize, -{} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L14-L29 - -```rust title="fresh-type-variable-example" showLineNumbers -let typevar1 = std::meta::typ::fresh_type_variable(); - let constraint = quote { Serialize<$typevar1> }.as_trait_constraint(); - let field_type = quote { Field }.as_type(); - - // Search for a trait impl (binding typevar1 to 1 when the impl is found): - assert(field_type.implements(constraint)); - - // typevar1 should be bound to the "1" generic now: - assert_eq(typevar1.as_constant().unwrap(), 1); - - // If we want to do the same with a different type, we need to - // create a new type variable now that `typevar1` is bound - let typevar2 = std::meta::typ::fresh_type_variable(); - let constraint = quote { Serialize<$typevar2> }.as_trait_constraint(); - let array_type = quote { [(Field, Field); 5] }.as_type(); - assert(array_type.implements(constraint)); - - // Now typevar2 should be bound to the serialized pair size 2 times the array length 5 - assert_eq(typevar2.as_constant().unwrap(), 10); -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L129-L149 - - -## Methods - -### as_array - -```rust title="as_array" showLineNumbers -pub comptime fn as_array(self) -> Option<(Type, Type)> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L76-L78 - - -If this type is an array, return a pair of (element type, size type). - -Example: - -```rust -comptime { - let array_type = quote { [Field; 3] }.as_type(); - let (field_type, three_type) = array_type.as_array().unwrap(); - - assert(field_type.is_field()); - assert_eq(three_type.as_constant().unwrap(), 3); -} -``` - -### as_constant - -```rust title="as_constant" showLineNumbers -pub comptime fn as_constant(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L83-L85 - - -If this type is a constant integer (such as the `3` in the array type `[Field; 3]`), -return the numeric constant. - -### as_integer - -```rust title="as_integer" showLineNumbers -pub comptime fn as_integer(self) -> Option<(bool, u8)> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L90-L92 - - -If this is an integer type, return a boolean which is `true` -if the type is signed, as well as the number of bits of this integer type. - -### as_mutable_reference - -```rust title="as_mutable_reference" showLineNumbers -comptime fn as_mutable_reference(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L96-L98 - - -If this is a mutable reference type `&mut T`, returns the mutable type `T`. - -### as_slice - -```rust title="as_slice" showLineNumbers -pub comptime fn as_slice(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L102-L104 - - -If this is a slice type, return the element type of the slice. - -### as_str - -```rust title="as_str" showLineNumbers -pub comptime fn as_str(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L108-L110 - - -If this is a `str` type, returns the length `N` as a type. - -### as_struct - -```rust title="as_struct" showLineNumbers -pub comptime fn as_struct(self) -> Option<(StructDefinition, [Type])> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L114-L116 - - -If this is a struct type, returns the struct in addition to -any generic arguments on this type. - -### as_tuple - -```rust title="as_tuple" showLineNumbers -pub comptime fn as_tuple(self) -> Option<[Type]> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L120-L122 - - -If this is a tuple type, returns each element type of the tuple. - -### get_trait_impl - -```rust title="get_trait_impl" showLineNumbers -pub comptime fn get_trait_impl(self, constraint: TraitConstraint) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L143-L145 - - -Retrieves the trait implementation that implements the given -trait constraint for this type. If the trait constraint is not -found, `None` is returned. Note that since the concrete trait implementation -for a trait constraint specified from a `where` clause is unknown, -this function will return `None` in these cases. If you only want to know -whether a type implements a trait, use `implements` instead. - -Example: - -```rust -comptime { - let field_type = quote { Field }.as_type(); - let default = quote { Default }.as_trait_constraint(); - - let the_impl: TraitImpl = field_type.get_trait_impl(default).unwrap(); - assert(the_impl.methods().len(), 1); -} -``` - -### implements - -```rust title="implements" showLineNumbers -pub comptime fn implements(self, constraint: TraitConstraint) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L166-L168 - - -`true` if this type implements the given trait. Note that unlike -`get_trait_impl` this will also return true for any `where` constraints -in scope. - -Example: - -```rust -fn foo() where T: Default { - comptime { - let field_type = quote { Field }.as_type(); - let default = quote { Default }.as_trait_constraint(); - assert(field_type.implements(default)); - - let t = quote { T }.as_type(); - assert(t.implements(default)); - } -} -``` - -### is_bool - -```rust title="is_bool" showLineNumbers -pub comptime fn is_bool(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L172-L174 - - -`true` if this type is `bool`. - -### is_field - -```rust title="is_field" showLineNumbers -pub comptime fn is_field(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L178-L180 - - -`true` if this type is `Field`. - -### is_unit - -```rust title="is_unit" showLineNumbers -comptime fn is_unit(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L184-L186 - - -`true` if this type is the unit `()` type. - -## Trait Implementations - -```rust -impl Eq for Type -impl Hash for Type -``` -Note that this is syntactic equality, this is not the same as whether two types will type check -to be the same type. Unless type inference or generics are being used however, users should not -typically have to worry about this distinction unless `std::meta::typ::fresh_type_variable` is used. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/options.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/options.md deleted file mode 100644 index a1bd4e1de5f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/options.md +++ /dev/null @@ -1,101 +0,0 @@ ---- -title: Option Type ---- - -The `Option` type is a way to express that a value might be present (`Some(T))` or absent (`None`). It's a safer way to handle potential absence of values, compared to using nulls in many other languages. - -```rust -struct Option { - None, - Some(T), -} -``` - -The `Option` type, already imported into your Noir program, can be used directly: - -```rust -fn main() { - let none = Option::none(); - let some = Option::some(3); -} -``` - -See [this test](https://github.com/noir-lang/noir/blob/5cbfb9c4a06c8865c98ff2b594464b037d821a5c/crates/nargo_cli/tests/test_data/option/src/main.nr) for a more comprehensive set of examples of each of the methods described below. - -## Methods - -### none - -Constructs a none value. - -### some - -Constructs a some wrapper around a given value. - -### is_none - -Returns true if the Option is None. - -### is_some - -Returns true of the Option is Some. - -### unwrap - -Asserts `self.is_some()` and returns the wrapped value. - -### unwrap_unchecked - -Returns the inner value without asserting `self.is_some()`. This method can be useful within an if condition when we already know that `option.is_some()`. If the option is None, there is no guarantee what value will be returned, only that it will be of type T for an `Option`. - -### unwrap_or - -Returns the wrapped value if `self.is_some()`. Otherwise, returns the given default value. - -### unwrap_or_else - -Returns the wrapped value if `self.is_some()`. Otherwise, calls the given function to return a default value. - -### expect - -Asserts `self.is_some()` with a provided custom message and returns the contained `Some` value. The custom message is expected to be a format string. - -### map - -If self is `Some(x)`, this returns `Some(f(x))`. Otherwise, this returns `None`. - -### map_or - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns the given default value. - -### map_or_else - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns `default()`. - -### and - -Returns None if self is None. Otherwise, this returns `other`. - -### and_then - -If self is None, this returns None. Otherwise, this calls the given function with the Some value contained within self, and returns the result of that call. In some languages this function is called `flat_map` or `bind`. - -### or - -If self is Some, return self. Otherwise, return `other`. - -### or_else - -If self is Some, return self. Otherwise, return `default()`. - -### xor - -If only one of the two Options is Some, return that option. Otherwise, if both options are Some or both are None, None is returned. - -### filter - -Returns `Some(x)` if self is `Some(x)` and `predicate(x)` is true. Otherwise, this returns `None`. - -### flatten - -Flattens an `Option>` into a `Option`. This returns `None` if the outer Option is None. Otherwise, this returns the inner Option. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/recursion.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/recursion.mdx deleted file mode 100644 index 60414a2fa51..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/recursion.mdx +++ /dev/null @@ -1,85 +0,0 @@ ---- -title: Recursive Proofs -description: Learn about how to write recursive proofs in Noir. -keywords: [recursion, recursive proofs, verification_key, verify_proof] ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports recursively verifying proofs, meaning you verify the proof of a Noir program in another Noir program. This enables creating proofs of arbitrary size by doing step-wise verification of smaller components of a large proof. - -Read [the explainer on recursion](../../explainers/explainer-recursion.md) to know more about this function and the [guide on how to use it.](../../how_to/how-to-recursion.md) - -## The `#[recursive]` Attribute - -In Noir, the `#[recursive]` attribute is used to indicate that a circuit is designed for recursive proof generation. When applied, it informs the compiler and the tooling that the circuit should be compiled in a way that makes its proofs suitable for recursive verification. This attribute eliminates the need for manual flagging of recursion at the tooling level, streamlining the proof generation process for recursive circuits. - -### Example usage with `#[recursive]` - -```rust -#[recursive] -fn main(x: Field, y: pub Field) { - assert(x == y, "x and y are not equal"); -} - -// This marks the circuit as recursion-friendly and indicates that proofs generated from this circuit -// are intended for recursive verification. -``` - -By incorporating this attribute directly in the circuit's definition, tooling like Nargo and NoirJS can automatically execute recursive-specific duties for Noir programs (e.g. recursive-friendly proof artifact generation) without additional flags or configurations. - -## Verifying Recursive Proofs - -```rust -#[foreign(recursive_aggregation)] -pub fn verify_proof(verification_key: [Field], proof: [Field], public_inputs: [Field], key_hash: Field) {} -``` - - - -## Example usage - -```rust - -fn main( - verification_key : [Field; 114], - proof : [Field; 93], - public_inputs : [Field; 1], - key_hash : Field, - proof_b : [Field; 93], -) { - std::verify_proof( - verification_key, - proof, - public_inputs, - key_hash - ); - - std::verify_proof( - verification_key, - proof_b, - public_inputs, - key_hash - ); -} -``` - -You can see a full example of recursive proofs in [this example recursion demo repo](https://github.com/noir-lang/noir-examples/tree/master/recursion). - -## Parameters - -### `verification_key` - -The verification key for the zk program that is being verified. - -### `proof` - -The proof for the zk program that is being verified. - -### `public_inputs` - -These represent the public inputs of the proof we are verifying. - -### `key_hash` - -A key hash is used to check the validity of the verification key. The circuit implementing this opcode can use this hash to ensure that the key provided to the circuit matches the key produced by the circuit creator. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/classes/Noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/classes/Noir.md deleted file mode 100644 index ead255bc504..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/classes/Noir.md +++ /dev/null @@ -1,52 +0,0 @@ -# Noir - -## Constructors - -### new Noir(circuit) - -```ts -new Noir(circuit): Noir -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `circuit` | `CompiledCircuit` | - -#### Returns - -[`Noir`](Noir.md) - -## Methods - -### execute() - -```ts -execute(inputs, foreignCallHandler?): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `inputs` | `InputMap` | -| `foreignCallHandler`? | [`ForeignCallHandler`](../type-aliases/ForeignCallHandler.md) | - -#### Returns - -`Promise`\<`object`\> - -#### Description - -Allows to execute a circuit to get its witness and return value. - -#### Example - -```typescript -async execute(inputs) -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/and.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/and.md deleted file mode 100644 index c783283e396..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/and.md +++ /dev/null @@ -1,22 +0,0 @@ -# and() - -```ts -and(lhs, rhs): string -``` - -Performs a bitwise AND operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/blake2s256.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/blake2s256.md deleted file mode 100644 index 7882d0da8d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/blake2s256.md +++ /dev/null @@ -1,21 +0,0 @@ -# blake2s256() - -```ts -blake2s256(inputs): Uint8Array -``` - -Calculates the Blake2s256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md deleted file mode 100644 index 5e3cd53e9d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256k1\_verify() - -```ts -ecdsa_secp256k1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256k1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md deleted file mode 100644 index 0b20ff68957..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256r1\_verify() - -```ts -ecdsa_secp256r1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256r1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/xor.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/xor.md deleted file mode 100644 index 8d762b895d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/functions/xor.md +++ /dev/null @@ -1,22 +0,0 @@ -# xor() - -```ts -xor(lhs, rhs): string -``` - -Performs a bitwise XOR operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md deleted file mode 100644 index e8c2f4aef3d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md +++ /dev/null @@ -1,15 +0,0 @@ -# ErrorWithPayload - -```ts -type ErrorWithPayload: ExecutionError & object; -``` - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `decodedAssertionPayload` | `any` | - | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md deleted file mode 100644 index 812b8b16481..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md +++ /dev/null @@ -1,24 +0,0 @@ -# ForeignCallHandler - -```ts -type ForeignCallHandler: (name, inputs) => Promise; -``` - -A callback which performs an foreign call and returns the response. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | The identifier for the type of foreign call being performed. | -| `inputs` | [`ForeignCallInput`](ForeignCallInput.md)[] | An array of hex encoded inputs to the foreign call. | - -## Returns - -`Promise`\<[`ForeignCallOutput`](ForeignCallOutput.md)[]\> - -outputs - An array of hex encoded outputs containing the results of the foreign call. - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md deleted file mode 100644 index dd95809186a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallInput - -```ts -type ForeignCallInput: string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md deleted file mode 100644 index b71fb78a946..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallOutput - -```ts -type ForeignCallOutput: string | string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md deleted file mode 100644 index 258c46f9d0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md +++ /dev/null @@ -1,9 +0,0 @@ -# WitnessMap - -```ts -type WitnessMap: Map; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/compile.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/compile.md deleted file mode 100644 index 6faf763b37f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/compile.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile() - -```ts -compile( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ProgramCompilationArtifacts`](../index.md#programcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_program(fm); -``` - -```typescript -// Browser - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_program(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/compile_contract.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/compile_contract.md deleted file mode 100644 index 7d0b39a43ef..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/compile_contract.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile\_contract() - -```ts -compile_contract( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ContractCompilationArtifacts`](../index.md#contractcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_contract(fm); -``` - -```typescript -// Browser - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_contract(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/createFileManager.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/createFileManager.md deleted file mode 100644 index 7e65c1d69c7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/createFileManager.md +++ /dev/null @@ -1,21 +0,0 @@ -# createFileManager() - -```ts -createFileManager(dataDir): FileManager -``` - -Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `dataDir` | `string` | root of the file system | - -## Returns - -`FileManager` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md deleted file mode 100644 index fcea9275341..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md +++ /dev/null @@ -1,21 +0,0 @@ -# inflateDebugSymbols() - -```ts -inflateDebugSymbols(debugSymbols): any -``` - -Decompresses and decodes the debug symbols - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `debugSymbols` | `string` | The base64 encoded debug symbols | - -## Returns - -`any` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/index.md deleted file mode 100644 index b6e0f9d1bc0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_wasm - -## Exports - -### Functions - -| Function | Description | -| :------ | :------ | -| [compile](functions/compile.md) | Compiles a Noir project | -| [compile\_contract](functions/compile_contract.md) | Compiles a Noir project | -| [createFileManager](functions/createFileManager.md) | Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) | -| [inflateDebugSymbols](functions/inflateDebugSymbols.md) | Decompresses and decodes the debug symbols | - -## References - -### compile\_program - -Renames and re-exports [compile](functions/compile.md) - -## Interfaces - -### ContractCompilationArtifacts - -The compilation artifacts of a given contract. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `contract` | `ContractArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -### ProgramCompilationArtifacts - -The compilation artifacts of a given program. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | not part of the compilation output, injected later | -| `program` | `ProgramArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs deleted file mode 100644 index e0870710349..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"doc","id":"reference/NoirJS/noir_wasm/index","label":"API"},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile","label":"compile"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile_contract","label":"compile_contract"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/createFileManager","label":"createFileManager"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/inflateDebugSymbols","label":"inflateDebugSymbols"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/_category_.json deleted file mode 100644 index 5b6a20a609a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 4, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/_category_.json deleted file mode 100644 index 27869205ad3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugger", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_known_limitations.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_known_limitations.md deleted file mode 100644 index 936d416ac4b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_known_limitations.md +++ /dev/null @@ -1,59 +0,0 @@ ---- -title: Known limitations -description: - An overview of known limitations of the current version of the Noir debugger -keywords: - [ - Nargo, - Noir Debugger, - VS Code, - ] -sidebar_position: 2 ---- - -# Debugger Known Limitations - -There are currently some limits to what the debugger can observe. - -## Mutable references - -The debugger is currently blind to any state mutated via a mutable reference. For example, in: - -``` -let mut x = 1; -let y = &mut x; -*y = 2; -``` - -The update on `x` will not be observed by the debugger. That means, when running `vars` from the debugger REPL, or inspecting the _local variables_ pane in the VS Code debugger, `x` will appear with value 1 despite having executed `*y = 2;`. - -## Variables of type function or mutable references are opaque - -When inspecting variables, any variable of type `Function` or `MutableReference` will render its value as `<>` or `<>`. - -## Debugger instrumentation affects resulting ACIR - -In order to make the state of local variables observable, the debugger compiles Noir circuits interleaving foreign calls that track any mutations to them. While this works (except in the cases described above) and doesn't introduce any behavior changes, it does as a side effect produce bigger bytecode. In particular, when running the command `opcodes` on the REPL debugger, you will notice Unconstrained VM blocks that look like this: - -``` -... -5 BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [], q_c: 2 }), Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(2))], q_c: 0 })] - | outputs=[] - 5.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 5.1 | Mov { destination: RegisterIndex(3), source: RegisterIndex(1) } - 5.2 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 5.3 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 5.4 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 5.5 | Mov { destination: RegisterIndex(3), source: RegisterIndex(3) } - 5.6 | Call { location: 8 } - 5.7 | Stop - 5.8 | ForeignCall { function: "__debug_var_assign", destinations: [], inputs: [RegisterIndex(RegisterIndex(2)), RegisterIndex(RegisterIndex(3))] } -... -``` - -If you are interested in debugging/inspecting compiled ACIR without these synthetic changes, you can invoke the REPL debugger with the `--skip-instrumentation` flag or launch the VS Code debugger with the `skipConfiguration` property set to true in its launch configuration. You can find more details about those in the [Debugger REPL reference](debugger_repl.md) and the [VS Code Debugger reference](debugger_vscode.md). - -:::note -Skipping debugger instrumentation means you won't be able to inspect values of local variables. -::: - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_repl.md deleted file mode 100644 index 46e2011304e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_repl.md +++ /dev/null @@ -1,360 +0,0 @@ ---- -title: REPL Debugger -description: - Noir Debugger REPL options and commands. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -## Running the REPL debugger - -`nargo debug [OPTIONS] [WITNESS_NAME]` - -Runs the Noir REPL debugger. If a `WITNESS_NAME` is provided the debugger writes the resulting execution witness to a `WITNESS_NAME` file. - -### Options - -| Option | Description | -| --------------------- | ------------------------------------------------------------ | -| `-p, --prover-name ` | The name of the toml file which contains the inputs for the prover [default: Prover]| -| `--package ` | The name of the package to debug | -| `--print-acir` | Display the ACIR for compiled circuit | -| `--deny-warnings` | Treat all warnings as errors | -| `--silence-warnings` | Suppress warnings | -| `-h, --help` | Print help | - -None of these options are required. - -:::note -Since the debugger starts by compiling the target package, all Noir compiler options are also available. Check out the [compiler reference](../nargo_commands.md#nargo-compile) to learn more about the compiler options. -::: - -## REPL commands - -Once the debugger is running, it accepts the following commands. - -#### `help` (h) - -Displays the menu of available commands. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) value - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -### Stepping through programs - -#### `next` (n) - -Step until the next Noir source code location. While other commands, such as [`into`](#into-i) and [`step`](#step-s), allow for finer grained control of the program's execution at the opcode level, `next` is source code centric. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `next` here would cause the debugger to jump to the definition of `deep_entry_point` (if available). - -If you want to step over `deep_entry_point` and go straight to line 8, use [the `over` command](#over) instead. - -#### `over` - -Step until the next source code location, without diving into function calls. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `over` here would cause the debugger to execute until line 8 (`multiple_values_entry_point(x);`). - -If you want to step into `deep_entry_point` instead, use [the `next` command](#next-n). - -#### `out` - -Step until the end of the current function call. For example: - -``` - 3 ... - 4 fn main(x: u32) { - 5 assert(entry_point(x) == 2); - 6 swap_entry_point(x, x + 1); - 7 -> assert(deep_entry_point(x) == 4); - 8 multiple_values_entry_point(x); - 9 } - 10 - 11 unconstrained fn returns_multiple_values(x: u32) -> (u32, u32, u32, u32) { - 12 ... - ... - 55 - 56 unconstrained fn deep_entry_point(x: u32) -> u32 { - 57 -> level_1(x + 1) - 58 } - -``` - -Running `out` here will resume execution until line 8. - -#### `step` (s) - -Skips to the next ACIR code. A compiled Noir program is a sequence of ACIR opcodes. However, an unconstrained VM opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `step` command at this point would result in the debugger stopping at ACIR opcode 2, `EXPR`, skipping unconstrained computation steps. - -Use [the `into` command](#into-i) instead if you want to follow unconstrained computation step by step. - -#### `into` (i) - -Steps into the next opcode. A compiled Noir program is a sequence of ACIR opcodes. However, a BRILLIG opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `into` command at this point would result in the debugger stopping at opcode 1.0, `Mov ...`, allowing the debugger user to follow unconstrained computation step by step. - -Use [the `step` command](#step-s) instead if you want to skip to the next ACIR code directly. - -#### `continue` (c) - -Continues execution until the next breakpoint, or the end of the program. - -#### `restart` (res) - -Interrupts execution, and restarts a new debugging session from scratch. - -#### `opcodes` (o) - -Display the program's ACIR opcode sequence. For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -### Breakpoints - -#### `break [Opcode]` (or shorthand `b [Opcode]`) - -Sets a breakpoint on the specified opcode index. To get a list of the program opcode numbers, see [the `opcode` command](#opcodes-o). For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -In this example, issuing a `break 1.2` command adds break on opcode 1.2, as denoted by the `*` character: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | * Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -Running [the `continue` command](#continue-c) at this point would cause the debugger to execute the program until opcode 1.2. - -#### `delete [Opcode]` (or shorthand `d [Opcode]`) - -Deletes a breakpoint at an opcode location. Usage is analogous to [the `break` command](#). - -### Variable inspection - -#### vars - -Show variable values available at this point in execution. - -:::note -The ability to inspect variable values from the debugger depends on compilation to be run in a special debug instrumentation mode. This instrumentation weaves variable tracing code with the original source code. - -So variable value inspection comes at the expense of making the resulting ACIR bytecode bigger and harder to understand and optimize. - -If you find this compromise unacceptable, you can run the debugger with the flag `--skip-debug-instrumentation`. This will compile your circuit without any additional debug information, so the resulting ACIR bytecode will be identical to the one produced by standard Noir compilation. However, if you opt for this, the `vars` command will not be available while debugging. -::: - - -### Stacktrace - -#### `stacktrace` - -Displays the current stack trace. - - -### Witness map - -#### `witness` (w) - -Show witness map. For example: - -``` -_0 = 0 -_1 = 2 -_2 = 1 -``` - -#### `witness [Witness Index]` - -Display a single witness from the witness map. For example: - -``` -> witness 1 -_1 = 2 -``` - -#### `witness [Witness Index] [New value]` - -Overwrite the given index with a new value. For example: - -``` -> witness 1 3 -_1 = 3 -``` - - -### Unconstrained VM memory - -#### `memory` - -Show unconstrained VM memory state. For example: - -``` -> memory -At opcode 1.13: Store { destination_pointer: RegisterIndex(0), source: RegisterIndex(3) } -... -> registers -0 = 0 -1 = 10 -2 = 0 -3 = 1 -4 = 1 -5 = 2³² -6 = 1 -> into -At opcode 1.14: Const { destination: RegisterIndex(5), value: Value { inner: 1 } } -... -> memory -0 = 1 -> -``` - -In the example above: we start with clean memory, then step through a `Store` opcode which stores the value of register 3 (1) into the memory address stored in register 0 (0). Thus now `memory` shows memory address 0 contains value 1. - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: - -#### `memset [Memory address] [New value]` - -Update a memory cell with the given value. For example: - -``` -> memory -0 = 1 -> memset 0 2 -> memory -0 = 2 -> memset 1 4 -> memory -0 = 2 -1 = 4 -> -``` - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_vscode.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_vscode.md deleted file mode 100644 index c027332b3b0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/debugger/debugger_vscode.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: VS Code Debugger -description: - VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -# VS Code Noir Debugger Reference - -The Noir debugger enabled by the vscode-noir extension ships with default settings such that the most common scenario should run without any additional configuration steps. - -These defaults can nevertheless be overridden by defining a launch configuration file. This page provides a reference for the properties you can override via a launch configuration file, as well as documenting the Nargo `dap` command, which is a dependency of the VS Code Noir debugger. - - -## Creating and editing launch configuration files - -To create a launch configuration file from VS Code, open the _debug pane_, and click on _create a launch.json file_. - -![Creating a launch configuration file](@site/static/img/debugger/ref1-create-launch.png) - -A `launch.json` file will be created, populated with basic defaults. - -### Noir Debugger launch.json properties - -#### projectFolder - -_String, optional._ - -Absolute path to the Nargo project to debug. By default, it is dynamically determined by looking for the nearest `Nargo.toml` file to the active file at the moment of launching the debugger. - -#### proverName - -_String, optional._ - -Name of the prover input to use. Defaults to `Prover`, which looks for a file named `Prover.toml` at the `projectFolder`. - -#### generateAcir - -_Boolean, optional._ - -If true, generate ACIR opcodes instead of unconstrained opcodes which will be closer to release binaries but less convenient for debugging. Defaults to `false`. - -#### skipInstrumentation - -_Boolean, optional._ - -Skips variables debugging instrumentation of code, making debugging less convenient but the resulting binary smaller and closer to production. Defaults to `false`. - -:::note -Skipping instrumentation causes the debugger to be unable to inspect local variables. -::: - -## `nargo dap [OPTIONS]` - -When run without any option flags, it starts the Nargo Debug Adapter Protocol server, which acts as the debugging backend for the VS Code Noir Debugger. - -All option flags are related to preflight checks. The Debug Adapter Protocol specifies how errors are to be informed from a running DAP server, but it doesn't specify mechanisms to communicate server initialization errors between the DAP server and its client IDE. - -Thus `nargo dap` ships with a _preflight check_ mode. If flag `--preflight-check` and the rest of the `--preflight-*` flags are provided, Nargo will run the same initialization routine except it will not start the DAP server. - -`vscode-noir` will then run `nargo dap` in preflight check mode first before a debugging session starts. If the preflight check ends in error, vscode-noir will present stderr and stdout output from this process through its own Output pane in VS Code. This makes it possible for users to diagnose what pieces of configuration might be wrong or missing in case of initialization errors. - -If the preflight check succeeds, `vscode-noir` proceeds to start the DAP server normally but running `nargo dap` without any additional flags. - -### Options - -| Option | Description | -| --------------------------------------------------------- | --------------------------------------------------------------------------------------------------------- | -| `--preflight-check` | If present, dap runs in preflight check mode. | -| `--preflight-project-folder ` | Absolute path to the project to debug for preflight check. | -| `--preflight-prover-name ` | Name of prover file to use for preflight check | -| `--preflight-generate-acir` | Optional. If present, compile in ACIR mode while running preflight check. | -| `--preflight-skip-instrumentation` | Optional. If present, compile without introducing debug instrumentation while running preflight check. | -| `-h, --help` | Print help. | diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/nargo_commands.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/nargo_commands.md deleted file mode 100644 index 44fe8a2e62c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/nargo_commands.md +++ /dev/null @@ -1,301 +0,0 @@ ---- -title: Nargo -description: - Noir CLI Commands for Noir Prover and Verifier to create, execute, prove and verify programs, - generate Solidity verifier smart contract and compile into JSON file containing ACIR - representation and ABI of circuit. -keywords: - [ - Nargo, - Noir CLI, - Noir Prover, - Noir Verifier, - generate Solidity verifier, - compile JSON file, - ACIR representation, - ABI of circuit, - TypeScript, - ] -sidebar_position: 0 ---- - -# Command-Line Help for `nargo` - -This document contains the help content for the `nargo` command-line program. - -**Command Overview:** - -* [`nargo`↴](#nargo) -* [`nargo check`↴](#nargo-check) -* [`nargo fmt`↴](#nargo-fmt) -* [`nargo compile`↴](#nargo-compile) -* [`nargo new`↴](#nargo-new) -* [`nargo init`↴](#nargo-init) -* [`nargo execute`↴](#nargo-execute) -* [`nargo debug`↴](#nargo-debug) -* [`nargo test`↴](#nargo-test) -* [`nargo info`↴](#nargo-info) -* [`nargo lsp`↴](#nargo-lsp) - -## `nargo` - -Noir's package manager - -**Usage:** `nargo ` - -###### **Subcommands:** - -* `check` — Checks the constraint system for errors -* `fmt` — Format the Noir files in a workspace -* `compile` — Compile the program and its secret execution trace into ACIR format -* `new` — Create a Noir project in a new directory -* `init` — Create a Noir project in the current directory -* `execute` — Executes a circuit to calculate its return value -* `debug` — Executes a circuit in debug mode -* `test` — Run the tests for this program -* `info` — Provides detailed information on each of a program's function (represented by a single circuit) -* `lsp` — Starts the Noir LSP server - -###### **Options:** - - - - -## `nargo check` - -Checks the constraint system for errors - -**Usage:** `nargo check [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to check -* `--workspace` — Check all packages in the workspace -* `--overwrite` — Force overwrite of existing files -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo fmt` - -Format the Noir files in a workspace - -**Usage:** `nargo fmt [OPTIONS]` - -###### **Options:** - -* `--check` — Run noirfmt in check mode - - - -## `nargo compile` - -Compile the program and its secret execution trace into ACIR format - -**Usage:** `nargo compile [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to compile -* `--workspace` — Compile all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo new` - -Create a Noir project in a new directory - -**Usage:** `nargo new [OPTIONS] ` - -###### **Arguments:** - -* `` — The path to save the new project - -###### **Options:** - -* `--name ` — Name of the package [default: package directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo init` - -Create a Noir project in the current directory - -**Usage:** `nargo init [OPTIONS]` - -###### **Options:** - -* `--name ` — Name of the package [default: current directory name] -* `--lib` — Use a library template -* `--bin` — Use a binary template [default] -* `--contract` — Use a contract template - - - -## `nargo execute` - -Executes a circuit to calculate its return value - -**Usage:** `nargo execute [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -Defaults to the name of the package being executed. - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--workspace` — Execute all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo debug` - -Executes a circuit in debug mode - -**Usage:** `nargo debug [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--acir-mode` — Force ACIR output (disabling instrumentation) -* `--skip-instrumentation ` — Disable vars debug instrumentation (enabled by default) - - Possible values: `true`, `false` - - - - -## `nargo test` - -Run the tests for this program - -**Usage:** `nargo test [OPTIONS] [TEST_NAME]` - -###### **Arguments:** - -* `` — If given, only tests with names containing this string will be run - -###### **Options:** - -* `--show-output` — Display output of `println` statements -* `--exact` — Only run tests that match exactly -* `--package ` — The name of the package to test -* `--workspace` — Test all packages in the workspace -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo info` - -Provides detailed information on each of a program's function (represented by a single circuit) - -Current information provided per circuit: 1. The number of ACIR opcodes 2. Counts the final number gates in the circuit used by a backend - -**Usage:** `nargo info [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to detail -* `--workspace` — Detail all packages in the workspace -* `--profile-execution` -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` -* `--force` — Force a full recompilation -* `--print-acir` — Display the ACIR for compiled circuit -* `--deny-warnings` — Treat all warnings as errors -* `--silence-warnings` — Suppress warnings -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - - -## `nargo lsp` - -Starts the Noir LSP server - -Starts an LSP server which allows IDEs such as VS Code to display diagnostics in Noir source. - -VS Code Noir Language Support: https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir - -**Usage:** `nargo lsp` - - - -
- - - This document was generated automatically by - clap-markdown. - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tooling/language_server.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tooling/language_server.md deleted file mode 100644 index 81e0356ef8a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tooling/language_server.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Language Server -description: Learn about the Noir Language Server, how to install the components, and configuration that may be required. -keywords: [Nargo, Language Server, LSP, VSCode, Visual Studio Code] -sidebar_position: 0 ---- - -This section helps you install and configure the Noir Language Server. - -The Language Server Protocol (LSP) has two components, the [Server](#language-server) and the [Client](#language-client). Below we describe each in the context of Noir. - -## Language Server - -The Server component is provided by the Nargo command line tool that you installed at the beginning of this guide. -As long as Nargo is installed and you've used it to run other commands in this guide, it should be good to go! - -If you'd like to verify that the `nargo lsp` command is available, you can run `nargo --help` and look for `lsp` in the list of commands. If you see it, you're using a version of Noir with LSP support. - -## Language Client - -The Client component is usually an editor plugin that launches the Server. It communicates LSP messages between the editor and the Server. For example, when you save a file, the Client will alert the Server, so it can try to compile the project and report any errors. - -Currently, Noir provides a Language Client for Visual Studio Code via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). - -> **Note:** Noir's Language Server Protocol support currently assumes users' VSCode workspace root to be the same as users' Noir project root (i.e. where Nargo.toml lies). -> -> If LSP features seem to be missing / malfunctioning, make sure you are opening your Noir project directly (instead of as a sub-folder) in your VSCode instance. - -When your language server is running correctly and the VSCode plugin is installed, you should see handy codelens buttons for compilation, measuring circuit size, execution, and tests: - -![Compile and Execute](@site/static/img/codelens_compile_execute.png) -![Run test](@site/static/img/codelens_run_test.png) - -You should also see your tests in the `testing` panel: - -![Testing panel](@site/static/img/codelens_testing_panel.png) - -### Configuration - -- **Noir: Enable LSP** - If checked, the extension will launch the Language Server via `nargo lsp` and communicate with it. -- **Noir: Nargo Flags** - Additional flags may be specified if you require them to be added when the extension calls `nargo lsp`. -- **Noir: Nargo Path** - An absolute path to a Nargo binary with the `lsp` command. This may be useful if Nargo is not within the `PATH` of your editor. -- **Noir > Trace: Server** - Setting this to `"messages"` or `"verbose"` will log LSP messages between the Client and Server. Useful for debugging. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tooling/testing.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tooling/testing.md deleted file mode 100644 index 866677da567..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tooling/testing.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Testing in Noir -description: Learn how to use Nargo to test your Noir program in a quick and easy way -keywords: [Nargo, testing, Noir, compile, test] -sidebar_position: 1 ---- - -You can test your Noir programs using Noir circuits. - -Nargo will automatically compile and run any functions which have the decorator `#[test]` on them if -you run `nargo test`. - -For example if you have a program like: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test] -fn test_add() { - assert(add(2,2) == 4); - assert(add(0,1) == 1); - assert(add(1,0) == 1); -} -``` - -Running `nargo test` will test that the `test_add` function can be executed while satisfying all -the constraints which allows you to test that add returns the expected values. Test functions can't -have any arguments currently. - -### Test fail - -You can write tests that are expected to fail by using the decorator `#[test(should_fail)]`. For example: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test(should_fail)] -fn test_add() { - assert(add(2,2) == 5); -} -``` - -You can be more specific and make it fail with a specific reason by using `should_fail_with = ""`: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "What is the airspeed velocity of an unladen swallow")] -fn test_bridgekeeper() { - main(32); -} -``` - -The string given to `should_fail_with` doesn't need to exactly match the failure reason, it just needs to be a substring of it: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "airspeed velocity")] -fn test_bridgekeeper() { - main(32); -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tutorials/noirjs_app.md deleted file mode 100644 index 6e69ea0bbed..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tutorials/noirjs_app.md +++ /dev/null @@ -1,366 +0,0 @@ ---- -title: Building a web app with NoirJS -description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. -keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] -sidebar_position: 0 -pagination_next: noir/concepts/data_types/index ---- - -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! - -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/quick_start.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! - -### Nargo - -Run: - -```bash -nargo new circuit -``` - -And... That's about it. Your program is ready to be compiled and run. - -To compile, let's `cd` into the `circuit` folder to enter our project, and call: - -```bash -nargo compile -``` - -This compiles our circuit into `json` format and add it to a new `target` folder. - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` - -::: - -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/quick_start.md) guide. However, we want our app to run on the browser, so we need Vite. - -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. - -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". - -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: - -```bash -cd vite-project -``` - -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; - -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); -``` - -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: - -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev -``` - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... -``` - -::: - -#### Some cleanup - -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. - -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) - -## HTML - -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: - -```html - - - - - - -

Noir app

-
- - -
-
-

Logs

-

Proof

-
- - -``` - -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). - -Start by pasting in this boilerplate code: - -```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} - -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } -}); -``` - -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html -``` - -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - -::: - -## Some NoirJS - -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: - -```ts -import circuit from '../circuit/target/circuit.json'; -``` - -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: - -```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -import { Noir } from '@noir-lang/noir_js'; -``` - -And instantiate them inside our try-catch block: - -```ts -// try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); -// } -``` - -:::note - -For the remainder of the tutorial, everything will be happening inside the `try` block - -::: - -## Our app - -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: - -```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; -``` - -Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: - -```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); -const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); -``` - -You're probably eager to see stuff happening, so go and run your app now! - -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. - -![Getting Started 0](@site/static/img/noir_getting_started_1.png) - -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! - -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. - -## Verifying - -Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: - -```js -display('logs', 'Verifying proof... ⌛'); -const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); -``` - -You have successfully generated a client-side Noir web app! - -![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) - -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. - -## UltraHonk Backend - -Barretenberg has recently exposed a new UltraHonk backend. We can use UltraHonk in NoirJS after version 0.33.0. Everything will be the same as the tutorial above, except that the class we need to import will change: - -```js -import { UltraHonkBackend, UltraHonkVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -``` - -The backend will then be instantiated as such: - -```js -const backend = new UltraHonkBackend(circuit); -``` - -Then all the commands to prove and verify your circuit will be same. - -The only feature currently unsupported with UltraHonk are [recursive proofs](../explainers/explainer-recursion.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/cspell.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/cspell.json deleted file mode 100644 index c60b0a597b1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/cspell.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "words": [ - "Cryptdoku" - ] -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-oracle.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-oracle.md deleted file mode 100644 index 821e1f95c04..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-oracle.md +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: Oracles -description: This guide provides an in-depth understanding of how Oracles work in Noir programming. Learn how to use outside calculations in your programs, constrain oracles, and understand their uses and limitations. -keywords: - - Noir Programming - - Oracles - - JSON-RPC - - Foreign Call Handlers - - Constrained Functions - - Blockchain Programming -sidebar_position: 1 ---- - -If you've seen "The Matrix" you may recall "The Oracle" as Gloria Foster smoking cigarettes and baking cookies. While she appears to "know things", she is actually providing a calculation of a pre-determined future. Noir Oracles are similar, in a way. They don't calculate the future (yet), but they allow you to use outside calculations in your programs. - -![matrix oracle prediction](@site/static/img/memes/matrix_oracle.jpeg) - -A Noir program is usually self-contained. You can pass certain inputs to it, and it will generate a deterministic output for those inputs. But what if you wanted to defer some calculation to an outside process or source? - -Oracles are functions that provide this feature. - -## Use cases - -An example usage for Oracles is proving something on-chain. For example, proving that the ETH-USDC quote was below a certain target at a certain block time. Or even making more complex proofs like proving the ownership of an NFT as an anonymous login method. - -Another interesting use case is to defer expensive calculations to be made outside of the Noir program, and then constraining the result; similar to the use of [unconstrained functions](../noir/concepts//unconstrained.md). - -In short, anything that can be constrained in a Noir program but needs to be fetched from an external source is a great candidate to be used in oracles. - -## Constraining oracles - -Just like in The Matrix, Oracles are powerful. But with great power, comes great responsibility. Just because you're using them in a Noir program doesn't mean they're true. Noir has no superpowers. If you want to prove that Portugal won the Euro Cup 2016, you're still relying on potentially untrusted information. - -To give a concrete example, Alice wants to login to the [NounsDAO](https://nouns.wtf/) forum with her username "noir_nouner" by proving she owns a noun without revealing her ethereum address. Her Noir program could have an oracle call like this: - -```rust -#[oracle(getNoun)] -unconstrained fn get_noun(address: Field) -> Field -``` - -This oracle could naively resolve with the number of Nouns she possesses. However, it is useless as a trusted source, as the oracle could resolve to anything Alice wants. In order to make this oracle call actually useful, Alice would need to constrain the response from the oracle, by proving her address and the noun count belongs to the state tree of the contract. - -In short, **Oracles don't prove anything. Your Noir program does.** - -:::danger - -If you don't constrain the return of your oracle, you could be clearly opening an attack vector on your Noir program. Make double-triple sure that the return of an oracle call is constrained! - -::: - -## How to use Oracles - -On CLI, Nargo resolves oracles by making JSON RPC calls, which means it would require an RPC node to be running. - -In JavaScript, NoirJS accepts and resolves arbitrary call handlers (that is, not limited to JSON) as long as they match the expected types the developer defines. Refer to [Foreign Call Handler](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) to learn more about NoirJS's call handling. - -If you want to build using oracles, follow through to the [oracle guide](../how_to/how-to-oracles.md) for a simple example on how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-recursion.md deleted file mode 100644 index df8529ef4e0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-recursion.md +++ /dev/null @@ -1,176 +0,0 @@ ---- -title: Recursive proofs -description: Explore the concept of recursive proofs in Zero-Knowledge programming. Understand how recursion works in Noir, a language for writing smart contracts on the EVM blockchain. Learn through practical examples like Alice and Bob's guessing game, Charlie's recursive merkle tree, and Daniel's reusable components. Discover how to use recursive proofs to optimize computational resources and improve efficiency. - -keywords: - [ - "Recursive Proofs", - "Zero-Knowledge Programming", - "Noir", - "EVM Blockchain", - "Smart Contracts", - "Recursion in Noir", - "Alice and Bob Guessing Game", - "Recursive Merkle Tree", - "Reusable Components", - "Optimizing Computational Resources", - "Improving Efficiency", - "Verification Key", - "Aggregation", - "Recursive zkSNARK schemes", - "PLONK", - "Proving and Verification Keys" - ] -sidebar_position: 1 -pagination_next: how_to/how-to-recursion ---- - -In programming, we tend to think of recursion as something calling itself. A classic example would be the calculation of the factorial of a number: - -```js -function factorial(n) { - if (n === 0 || n === 1) { - return 1; - } else { - return n * factorial(n - 1); - } -} -``` - -In this case, while `n` is not `1`, this function will keep calling itself until it hits the base case, bubbling up the result on the call stack: - -```md - Is `n` 1? <--------- - /\ / - / \ n = n -1 - / \ / - Yes No -------- -``` - -In Zero-Knowledge, recursion has some similarities. - -It is not a Noir function calling itself, but a proof being used as an input to another circuit. In short, you verify one proof *inside* another proof, returning the proof that both proofs are valid. - -This means that, given enough computational resources, you can prove the correctness of any arbitrary number of proofs in a single proof. This could be useful to design state channels (for which a common example would be [Bitcoin's Lightning Network](https://en.wikipedia.org/wiki/Lightning_Network)), to save on gas costs by settling one proof on-chain, or simply to make business logic less dependent on a consensus mechanism. - -## Examples - -Let us look at some of these examples - -### Alice and Bob - Guessing game - -Alice and Bob are friends, and they like guessing games. They want to play a guessing game online, but for that, they need a trusted third-party that knows both of their secrets and finishes the game once someone wins. - -So, they use zero-knowledge proofs. Alice tries to guess Bob's number, and Bob will generate a ZK proof stating whether she succeeded or failed. - -This ZK proof can go on a smart contract, revealing the winner and even giving prizes. However, this means every turn needs to be verified on-chain. This incurs some cost and waiting time that may simply make the game too expensive or time-consuming to be worth it. - -As a solution, Alice proposes the following: "what if Bob generates his proof, and instead of sending it on-chain, I verify it *within* my own proof before playing my own turn?". - -She can then generate a proof that she verified his proof, and so on. - -```md - Did you fail? <-------------------------- - / \ / - / \ n = n -1 - / \ / - Yes No / - | | / - | | / - | You win / - | / - | / -Generate proof of that / - + / - my own guess ---------------- -``` - -### Charlie - Recursive merkle tree - -Charlie is a concerned citizen, and wants to be sure his vote in an election is accounted for. He votes with a ZK proof, but he has no way of knowing that his ZK proof was included in the total vote count! - -If the vote collector puts all of the votes into a [Merkle tree](https://en.wikipedia.org/wiki/Merkle_tree), everyone can prove the verification of two proofs within one proof, as such: - -```md - abcd - __________|______________ - | | - ab cd - _____|_____ ______|______ - | | | | - alice bob charlie daniel -``` - -Doing this recursively allows us to arrive on a final proof `abcd` which if true, verifies the correctness of all the votes. - -### Daniel - Reusable components - -Daniel has a big circuit and a big headache. A part of his circuit is a setup phase that finishes with some assertions that need to be made. But that section alone takes most of the proving time, and is largely independent of the rest of the circuit. - -He might find it more efficient to generate a proof for that setup phase separately, and verify that proof recursively in the actual business logic section of his circuit. This will allow for parallelization of both proofs, which results in a considerable speedup. - -## What params do I need - -As you can see in the [recursion reference](noir/standard_library/recursion.mdx), a simple recursive proof requires: - -- The proof to verify -- The Verification Key of the circuit that generated the proof -- A hash of this verification key, as it's needed for some backends -- The public inputs for the proof - -:::info - -Recursive zkSNARK schemes do not necessarily "verify a proof" in the sense that you expect a true or false to be spit out by the verifier. Rather an aggregation object is built over the public inputs. - -So, taking the example of Alice and Bob and their guessing game: - -- Alice makes her guess. Her proof is *not* recursive: it doesn't verify any proof within it! It's just a standard `assert(x != y)` circuit -- Bob verifies Alice's proof and makes his own guess. In this circuit, he doesn't exactly *prove* the verification of Alice's proof. Instead, he *aggregates* his proof to Alice's proof. The actual verification is done when the full proof is verified, for example when using `nargo verify` or through the verifier smart contract. - -We can imagine recursive proofs a [relay race](https://en.wikipedia.org/wiki/Relay_race). The first runner doesn't have to receive the baton from anyone else, as he/she already starts with it. But when his/her turn is over, the next runner needs to receive it, run a bit more, and pass it along. Even though every runner could theoretically verify the baton mid-run (why not? 🏃🔍), only at the end of the race does the referee verify that the whole race is valid. - -::: - -## Some architecture - -As with everything in computer science, there's no one-size-fits all. But there are some patterns that could help understanding and implementing them. To give three examples: - -### Adding some logic to a proof verification - -This would be an approach for something like our guessing game, where proofs are sent back and forth and are verified by each opponent. This circuit would be divided in two sections: - -- A `recursive verification` section, which would be just the call to `std::verify_proof`, and that would be skipped on the first move (since there's no proof to verify) -- A `guessing` section, which is basically the logic part where the actual guessing happens - -In such a situation, and assuming Alice is first, she would skip the first part and try to guess Bob's number. Bob would then verify her proof on the first section of his run, and try to guess Alice's number on the second part, and so on. - -### Aggregating proofs - -In some one-way interaction situations, recursion would allow for aggregation of simple proofs that don't need to be immediately verified on-chain or elsewhere. - -To give a practical example, a barman wouldn't need to verify a "proof-of-age" on-chain every time he serves alcohol to a customer. Instead, the architecture would comprise two circuits: - -- A `main`, non-recursive circuit with some logic -- A `recursive` circuit meant to verify two proofs in one proof - -The customer's proofs would be intermediate, and made on their phones, and the barman could just verify them locally. He would then aggregate them into a final proof sent on-chain (or elsewhere) at the end of the day. - -### Recursively verifying different circuits - -Nothing prevents you from verifying different circuits in a recursive proof, for example: - -- A `circuit1` circuit -- A `circuit2` circuit -- A `recursive` circuit - -In this example, a regulator could verify that taxes were paid for a specific purchase by aggregating both a `payer` circuit (proving that a purchase was made and taxes were paid), and a `receipt` circuit (proving that the payment was received) - -## How fast is it - -At the time of writing, verifying recursive proofs is surprisingly fast. This is because most of the time is spent on generating the verification key that will be used to generate the next proof. So you are able to cache the verification key and reuse it later. - -Currently, Noir JS packages don't expose the functionality of loading proving and verification keys, but that feature exists in the underlying `bb.js` package. - -## How can I try it - -Learn more about using recursion in Nargo and NoirJS in the [how-to guide](../how_to/how-to-recursion.md) and see a full example in [noir-examples](https://github.com/noir-lang/noir-examples). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-writing-noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-writing-noir.md deleted file mode 100644 index 3ce4245dc45..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/explainers/explainer-writing-noir.md +++ /dev/null @@ -1,177 +0,0 @@ ---- -title: Thinking in Circuits -description: Considerations when writing Noir programs -keywords: [Noir, programming, rust] -tags: [Optimization] -sidebar_position: 0 ---- - - -This article intends to set you up with key concepts essential for writing more viable applications that use zero knowledge proofs, namely around efficient circuits. - -## Context - 'Efficient' is subjective - -When writing a web application for a performant computer with high-speed internet connection, writing efficient code sometimes is seen as an afterthought only if needed. Large multiplications running at the innermost of nested loops may not even be on a dev's radar. -When writing firmware for a battery-powered microcontroller, you think of cpu cycles as rations to keep within a product's power budget. - -> Code is written to create applications that perform specific tasks within specific constraints - -And these constraints differ depending on where the compiled code is execute. - -### The Ethereum Virtual Machine (EVM) - -In scenarios where extremely low gas costs are required for an Ethereum application to be viable/competitive, Ethereum smart contract developers get into what is colloquially known as: "*gas golfing*". Finding the lowest execution cost of their compiled code (EVM bytecode) to achieve a specific task. - -The equivalent optimization task when writing zk circuits is affectionately referred to as "*gate golfing*", finding the lowest gate representation of the compiled Noir code. - -### Coding for circuits - a paradigm shift - -In zero knowledge cryptography, code is compiled to "circuits" consisting of arithmetic gates, and gate count is the significant cost. Depending on the proving system this is linearly proportionate to proving time, and so from a product point this should be kept as low as possible. - -Whilst writing efficient code for web apps and Solidity has a few key differences, writing efficient circuits have a different set of considerations. It is a bit of a paradigm shift, like writing code for GPUs for the first time... - -For example, drawing a circle at (0, 0) of radius `r`: -- For a single CPU thread, -``` -for theta in 0..2*pi { - let x = r * cos(theta); - let y = r * sin(theta); - draw(x, y); -} // note: would do 0 - pi/2 and draw +ve/-ve x and y. -``` - -- For GPUs (simultaneous parallel calls with x, y across image), -``` -if (x^2 + y^2 = r^2) { - draw(x, y); -} -``` - -([Related](https://www.youtube.com/watch?v=-P28LKWTzrI)) - -Whilst this CPU -> GPU does not translate to circuits exactly, it is intended to exemplify the difference in intuition when coding for different machine capabilities/constraints. - -### Context Takeaway - -For those coming from a primarily web app background, this article will explain what you need to consider when writing circuits. Furthermore, for those experienced writing efficient machine code, prepare to shift what you think is efficient 😬 - -## Translating from Rust - -For some applications using Noir, existing code might be a convenient starting point to then proceed to optimize the gate count of. - -:::note -Many valuable functions and algorithms have been written in more established languages (C/C++), and converted to modern ones (like Rust). -::: - -Fortunately for Noir developers, when needing a particular function a Rust implementation can be readily compiled into Noir with some key changes. While the compiler does a decent amount of optimizations, it won't be able to change code that has been optimized for clock-cycles into code optimized for arithmetic gates. - -A few things to do when converting Rust code to Noir: -- `println!` is not a macro, use `println` function (same for `assert_eq`) -- No early `return` in function. Use constrain via assertion instead -- No passing by reference. Remove `&` operator to pass by value (copy) -- No boolean operators (`&&`, `||`). Use bitwise operators (`&`, `|`) with boolean values -- No type `usize`. Use types `u8`, `u32`, `u64`, ... -- `main` return must be public, `pub` -- No `const`, use `global` -- Noir's LSP is your friend, so error message should be informative enough to resolve syntax issues. - -## Writing efficient Noir for performant products - -The following points help refine our understanding over time. - -:::note -A Noir program makes a statement that can be verified. -::: - -It compiles to a structure that represents the calculation, and can assert results within the calculation at any stage (via the `constrain` keyword). - -A Noir program compiles to an Abstract Circuit Intermediate Representation which is: - - Conceptually a tree structure - - Leaves (inputs) are the `Field` type - - Nodes contain arithmetic operations to combine them (gates) - - The root is the final result (return value) - -:::tip -The command `nargo info` shows the programs circuit size, and is useful to compare the value of changes made. -You can dig deeper and use the `--print-acir` param to take a closer look at individual ACIR opcodes, and the proving backend to see its gate count (eg for barretenberg, `bb gates -b ./target/program.json`). -::: - -### Use the `Field` type - -Since the native type of values in circuits are `Field`s, using them for variables in Noir means less gates converting them under the hood. -Some things to be mindful of when using a Field type for a regular integer value: -- A variable of type `Field` can be cast `as` an integer type (eg `u8`, `u64`) - - Note: this retains only the bits of the integer type. Eg a Field value of 260 as a `u8` becomes 4 -- For Field types arithmetic operations meaningfully overflow/underflow, yet for integer types they are checked according to their size -- Comparisons and bitwise operations do not exist for `Field`s, cast to an appropriately sized integer type when you need to - -:::tip -Where possible, use `Field` type for values. Using smaller value types, and bit-packing strategies, will result in MORE gates -::: - - -### Use Arithmetic over non-arithmetic operations - -Since circuits are made of arithmetic gates, the cost of arithmetic operations tends to be one gate. Whereas for procedural code, they represent several clock cycles. - -Inversely, non-arithmetic operators are achieved with multiple gates, vs 1 clock cycle for procedural code. - -| (cost\op) | arithmetic
(`*`, `+`) | bit-wise ops
(eg `<`, `\|`, `>>`) | -| - | - | - | -| **cycles** | 10+ | 1 | -| **gates** | 1 | 10+ | - -Bit-wise operations (e.g. bit shifts `<<` and `>>`), albeit commonly used in general programming and especially for clock cycle optimizations, are on the contrary expensive in gates when performed within circuits. - -Translate away from bit shifts when writing constrained functions for the best performance. - -On the flip side, feel free to use bit shifts in unconstrained functions and tests if necessary, as they are executed outside of circuits and does not induce performance hits. - -### Use static over dynamic values - -Another general theme that manifests in different ways is that static reads are represented with less gates than dynamic ones. - -Reading from read-only memory (ROM) adds less gates than random-access memory (RAM), 2 vs ~3.25 due to the additional bounds checks. Arrays of fixed length (albeit used at a lower capacity), will generate less gates than dynamic storage. - -Related to this, if an index used to access an array is not known at compile time (ie unknown until run time), then ROM will be converted to RAM, expanding the gate count. - -:::tip -Use arrays and indices that are known at compile time where possible. -Using `assert_constant(i);` before an index, `i`, is used in an array will give a compile error if `i` is NOT known at compile time. -::: - -### Leverage unconstrained execution - -Constrained verification can leverage unconstrained execution, this is especially useful for operations that are represented by many gates. -Use an [unconstrained function](../noir/concepts/unconstrained.md) to perform gate-heavy calculations, then verify and constrain the result. - -Eg division generates more gates than multiplication, so calculating the quotient in an unconstrained function then constraining the product for the quotient and divisor (+ any remainder) equals the dividend will be more efficient. - -Use ` if is_unconstrained() { /`, to conditionally execute code if being called in an unconstrained vs constrained way. - -## Advanced - -Unless you're well into the depth of gate optimization, this advanced section can be ignored. - -### Combine arithmetic operations - -A Noir program can be honed further by combining arithmetic operators in a way that makes the most of each constraint of the backend proving system. This is in scenarios where the backend might not be doing this perfectly. - -Eg Barretenberg backend (current default for Noir) is a width-4 PLONKish constraint system -$ w_1*w_2*q_m + w_1*q_1 + w_2*q_2 + w_3*q_3 + w_4*q_4 + q_c = 0 $ - -Here we see there is one occurrence of witness 1 and 2 ($w_1$, $w_2$) being multiplied together, with addition to witnesses 1-4 ($w_1$ .. $w_4$) multiplied by 4 corresponding circuit constants ($q_1$ .. $q_4$) (plus a final circuit constant, $q_c$). - -Use `nargo info --print-acir`, to inspect the ACIR opcodes (and the proving system for gates), and it may present opportunities to amend the order of operations and reduce the number of constraints. - -#### Variable as witness vs expression - -If you've come this far and really know what you're doing at the equation level, a temporary lever (that will become unnecessary/useless over time) is: `std::as_witness`. This informs the compiler to save a variable as a witness not an expression. - -The compiler will mostly be correct and optimal, but this may help some near term edge cases that are yet to optimize. -Note: When used incorrectly it will create **less** efficient circuits (higher gate count). - -## References -- Guillaume's ["`Cryptdoku`" talk](https://www.youtube.com/watch?v=MrQyzuogxgg) (Jun'23) -- Tips from Tom, Jake and Zac. -- [Idiomatic Noir](https://www.vlayer.xyz/blog/idiomatic-noir-part-1-collections) blog post diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/project_breakdown.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/project_breakdown.md deleted file mode 100644 index e442e377040..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/project_breakdown.md +++ /dev/null @@ -1,159 +0,0 @@ ---- -title: Project Breakdown -description: - Learn about the anatomy of a Nargo project, including the purpose of the Prover TOML - file, and how to prove and verify your program. -keywords: - [Nargo, Nargo project, Prover.toml, proof verification, private asset transfer] -sidebar_position: 1 ---- - -This section breaks down our hello world program from the previous section. - -## Anatomy of a Nargo Project - -Upon creating a new project with `nargo new` and building the in/output files with `nargo check` -commands, you would get a minimal Nargo project of the following structure: - - - src - - Prover.toml - - Nargo.toml - -The source directory _src_ holds the source code for your Noir program. By default only a _main.nr_ -file will be generated within it. - -### Prover.toml - -_Prover.toml_ is used for specifying the input values for executing and proving the program. You can specify `toml` files with different names by using the `--prover-name` or `-p` flags, see the [Prover](#provertoml) section below. Optionally you may specify expected output values for prove-time checking as well. - -### Nargo.toml - -_Nargo.toml_ contains the environmental options of your project. It contains a "package" section and a "dependencies" section. - -Example Nargo.toml: - -```toml -[package] -name = "noir_starter" -type = "bin" -authors = ["Alice"] -compiler_version = "0.9.0" -description = "Getting started with Noir" -entry = "circuit/main.nr" -license = "MIT" - -[dependencies] -ecrecover = {tag = "v0.9.0", git = "https://github.com/colinnielsen/ecrecover-noir.git"} -``` - -Nargo.toml for a [workspace](../noir/modules_packages_crates/workspaces.md) will look a bit different. For example: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -#### Package section - -The package section defines a number of fields including: - -- `name` (**required**) - the name of the package -- `type` (**required**) - can be "bin", "lib", or "contract" to specify whether its a binary, library or Aztec contract -- `authors` (optional) - authors of the project -- `compiler_version` - specifies the version of the compiler to use. This is enforced by the compiler and follow's [Rust's versioning](https://doc.rust-lang.org/cargo/reference/manifest.html#the-version-field), so a `compiler_version = 0.18.0` will enforce Nargo version 0.18.0, `compiler_version = ^0.18.0` will enforce anything above 0.18.0 but below 0.19.0, etc. For more information, see how [Rust handles these operators](https://docs.rs/semver/latest/semver/enum.Op.html) -- `description` (optional) -- `entry` (optional) - a relative filepath to use as the entry point into your package (overrides the default of `src/lib.nr` or `src/main.nr`) -- `backend` (optional) -- `license` (optional) -- `expression_width` (optional) - Sets the default backend expression width. This field will override the default backend expression width specified by the Noir compiler (currently set to width 4). - -#### Dependencies section - -This is where you will specify any dependencies for your project. See the [Dependencies page](../noir/modules_packages_crates/dependencies.md) for more info. - -`./proofs/` and `./contract/` directories will not be immediately visible until you create a proof or -verifier contract respectively. - -### main.nr - -The _main.nr_ file contains a `main` method, this method is the entry point into your Noir program. - -In our sample program, _main.nr_ looks like this: - -```rust -fn main(x : Field, y : Field) { - assert(x != y); -} -``` - -The parameters `x` and `y` can be seen as the API for the program and must be supplied by the prover. Since neither `x` nor `y` is marked as public, the verifier does not supply any inputs, when verifying the proof. - -The prover supplies the values for `x` and `y` in the _Prover.toml_ file. - -As for the program body, `assert` ensures that the condition to be satisfied (e.g. `x != y`) is constrained by the proof of the execution of said program (i.e. if the condition was not met, the verifier would reject the proof as an invalid proof). - -### Prover.toml - -The _Prover.toml_ file is a file which the prover uses to supply the inputs to the Noir program (both private and public). - -In our hello world program the _Prover.toml_ file looks like this: - -```toml -x = "1" -y = "2" -``` - -When the command `nargo execute` is executed, nargo will execute the Noir program using the inputs specified in `Prover.toml`, aborting if it finds that these do not satisfy the constraints defined by `main`. In this example, `x` and `y` must satisfy the inequality constraint `assert(x != y)`. - -If an output name is specified such as `nargo execute foo`, the witness generated by this execution will be written to `./target/foo.gz`. This can then be used to generate a proof of the execution. - -#### Arrays of Structs - -The following code shows how to pass an array of structs to a Noir program to generate a proof. - -```rust -// main.nr -struct Foo { - bar: Field, - baz: Field, -} - -fn main(foos: [Foo; 3]) -> pub Field { - foos[2].bar + foos[2].baz -} -``` - -Prover.toml: - -```toml -[[foos]] # foos[0] -bar = 0 -baz = 0 - -[[foos]] # foos[1] -bar = 0 -baz = 0 - -[[foos]] # foos[2] -bar = 1 -baz = 2 -``` - -#### Custom toml files - -You can specify a `toml` file with a different name to use for execution by using the `--prover-name` or `-p` flags. - -This command looks for proof inputs in the default **Prover.toml** and generates the witness and saves it at `./target/foo.gz`: - -```bash -nargo execute foo -``` - -This command looks for proof inputs in the custom **OtherProver.toml** and generates the witness and saves it at `./target/bar.gz`: - -```bash -nargo execute -p OtherProver bar -``` - -Now that you understand the concepts, you'll probably want some editor feedback while you are writing more complex code. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/_category_.json deleted file mode 100644 index 23b560f610b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/_category_.json deleted file mode 100644 index cc2cbb1c253..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugging", - "position": 5, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/debugging_with_the_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/debugging_with_the_repl.md deleted file mode 100644 index 1d64dae3f37..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/debugging_with_the_repl.md +++ /dev/null @@ -1,164 +0,0 @@ ---- -title: Using the REPL Debugger -description: - Step-by-step guide on how to debug your Noir circuits with the REPL Debugger. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -#### Pre-requisites - -In order to use the REPL debugger, first you need to install recent enough versions of Nargo and vscode-noir. - -## Debugging a simple circuit - -Let's debug a simple circuit: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -To start the REPL debugger, using a terminal, go to a Noir circuit's home directory. Then: - -`$ nargo debug` - -You should be seeing this in your terminal: - -``` -[main] Starting debugger -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:9 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> -``` - -The debugger displays the current Noir code location, and it is now waiting for us to drive it. - -Let's first take a look at the available commands. For that we'll use the `help` command. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -Some commands operate only for unconstrained functions, such as `memory` and `memset`. If you try to use them while execution is paused at an ACIR opcode, the debugger will simply inform you that you are not executing unconstrained code: - -``` -> memory -Unconstrained VM memory not available -> -``` - -Before continuing, we can take a look at the initial witness map: - -``` -> witness -_0 = 1 -_1 = 2 -> -``` - -Cool, since `x==1`, `y==2`, and we want to check that `x != y`, our circuit should succeed. At this point we could intervene and use the witness setter command to change one of the witnesses. Let's set `y=3`, then back to 2, so we don't affect the expected result: - -``` -> witness -_0 = 1 -_1 = 2 -> witness 1 3 -_1 = 3 -> witness -_0 = 1 -_1 = 3 -> witness 1 2 -_1 = 2 -> witness -_0 = 1 -_1 = 2 -> -``` - -Now we can inspect the current state of local variables. For that we use the `vars` command. - -``` -> vars -> -``` - -We currently have no vars in context, since we are at the entry point of the program. Let's use `next` to execute until the next point in the program. - -``` -> vars -> next -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:20 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> vars -x:Field = 0x01 -``` - -As a result of stepping, the variable `x`, whose initial value comes from the witness map, is now in context and returned by `vars`. - -``` -> next - 1 fn main(x : Field, y : pub Field) { - 2 -> assert(x != y); - 3 } -> vars -y:Field = 0x02 -x:Field = 0x01 -``` - -Stepping again we can finally see both variables and their values. And now we can see that the next assertion should succeed. - -Let's continue to the end: - -``` -> continue -(Continuing execution...) -Finished execution -> q -[main] Circuit witness successfully solved -``` - -Upon quitting the debugger after a solved circuit, the resulting circuit witness gets saved, equivalent to what would happen if we had run the same circuit with `nargo execute`. - -We just went through the basics of debugging using Noir REPL debugger. For a comprehensive reference, check out [the reference page](../../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/debugging_with_vs_code.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/debugging_with_vs_code.md deleted file mode 100644 index a5858c1a5eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/debugger/debugging_with_vs_code.md +++ /dev/null @@ -1,68 +0,0 @@ ---- -title: Using the VS Code Debugger -description: - Step by step guide on how to debug your Noir circuits with the VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -This guide will show you how to use VS Code with the vscode-noir extension to debug a Noir project. - -#### Pre-requisites - -- Nargo -- vscode-noir -- A Noir project with a `Nargo.toml`, `Prover.toml` and at least one Noir (`.nr`) containing an entry point function (typically `main`). - -## Running the debugger - -The easiest way to start debugging is to open the file you want to debug, and press `F5`. This will cause the debugger to launch, using your `Prover.toml` file as input. - -You should see something like this: - -![Debugger launched](@site/static/img/debugger/1-started.png) - -Let's inspect the state of the program. For that, we open VS Code's _Debug pane_. Look for this icon: - -![Debug pane icon](@site/static/img/debugger/2-icon.png) - -You will now see two categories of variables: Locals and Witness Map. - -![Debug pane expanded](@site/static/img/debugger/3-debug-pane.png) - -1. **Locals**: variables of your program. At this point in execution this section is empty, but as we step through the code it will get populated by `x`, `result`, `digest`, etc. - -2. **Witness map**: these are initially populated from your project's `Prover.toml` file. In this example, they will be used to populate `x` and `result` at the beginning of the `main` function. - -Most of the time you will probably be focusing mostly on locals, as they represent the high level state of your program. - -You might be interested in inspecting the witness map in case you are trying to solve a really low level issue in the compiler or runtime itself, so this concerns mostly advanced or niche users. - -Let's step through the program, by using the debugger buttons or their corresponding keyboard shortcuts. - -![Debugger buttons](@site/static/img/debugger/4-debugger-buttons.png) - -Now we can see in the variables pane that there's values for `digest`, `result` and `x`. - -![Inspecting locals](@site/static/img/debugger/5-assert.png) - -We can also inspect the values of variables by directly hovering on them on the code. - -![Hover locals](@site/static/img/debugger/6-hover.png) - -Let's set a break point at the `keccak256` function, so we can continue execution up to the point when it's first invoked without having to go one step at a time. - -We just need to click the to the right of the line number 18. Once the breakpoint appears, we can click the `continue` button or use its corresponding keyboard shortcut (`F5` by default). - -![Breakpoint](@site/static/img/debugger/7-break.png) - -Now we are debugging the `keccak256` function, notice the _Call Stack pane_ at the lower right. This lets us inspect the current call stack of our process. - -That covers most of the current debugger functionalities. Check out [the reference](../../reference/debugger/debugger_vscode.md) for more details on how to configure the debugger. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-oracles.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-oracles.md deleted file mode 100644 index 08ce6ee5476..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-oracles.md +++ /dev/null @@ -1,275 +0,0 @@ ---- -title: How to use Oracles -description: Learn how to use oracles in your Noir program with examples in both Nargo and NoirJS. This guide also covers writing a JSON RPC server and providing custom foreign call handlers for NoirJS. -keywords: - - Noir Programming - - Oracles - - Nargo - - NoirJS - - JSON RPC Server - - Foreign Call Handlers -sidebar_position: 1 ---- - -This guide shows you how to use oracles in your Noir program. For the sake of clarity, it assumes that: - -- You have read the [explainer on Oracles](../explainers/explainer-oracle.md) and are comfortable with the concept. -- You have a Noir program to add oracles to. You can create one using the [vite-hardhat starter](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) as a boilerplate. -- You understand the concept of a JSON-RPC server. Visit the [JSON-RPC website](https://www.jsonrpc.org/) if you need a refresher. -- You are comfortable with server-side JavaScript (e.g. Node.js, managing packages, etc.). - -## Rundown - -This guide has 3 major steps: - -1. How to modify our Noir program to make use of oracle calls as unconstrained functions -2. How to write a JSON RPC Server to resolve these oracle calls with Nargo -3. How to use them in Nargo and how to provide a custom resolver in NoirJS - -## Step 1 - Modify your Noir program - -An oracle is defined in a Noir program by defining two methods: - -- An unconstrained method - This tells the compiler that it is executing an [unconstrained functions](../noir/concepts//unconstrained.md). -- A decorated oracle method - This tells the compiler that this method is an RPC call. - -An example of an oracle that returns a `Field` would be: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(number: Field) -> Field { } - -unconstrained fn get_sqrt(number: Field) -> Field { - sqrt(number) -} -``` - -In this example, we're wrapping our oracle function in an unconstrained method, and decorating it with `oracle(getSqrt)`. We can then call the unconstrained function as we would call any other function: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); -} -``` - -In the next section, we will make this `getSqrt` (defined on the `sqrt` decorator) be a method of the RPC server Noir will use. - -:::danger - -As explained in the [Oracle Explainer](../explainers/explainer-oracle.md), this `main` function is unsafe unless you constrain its return value. For example: - -```rust -fn main(input: Field) { - let sqrt = get_sqrt(input); - assert(sqrt.pow_32(2) as u64 == input as u64); // <---- constrain the return of an oracle! -} -``` - -::: - -:::info - -Currently, oracles only work with single params or array params. For example: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt([Field; 2]) -> [Field; 2] { } -``` - -::: - -## Step 2 - Write an RPC server - -Brillig will call *one* RPC server. Most likely you will have to write your own, and you can do it in whatever language you prefer. In this guide, we will do it in Javascript. - -Let's use the above example of an oracle that consumes an array with two `Field` and returns their square roots: - -```rust -#[oracle(getSqrt)] -unconstrained fn sqrt(input: [Field; 2]) -> [Field; 2] { } - -unconstrained fn get_sqrt(input: [Field; 2]) -> [Field; 2] { - sqrt(input) -} - -fn main(input: [Field; 2]) { - let sqrt = get_sqrt(input); - assert(sqrt[0].pow_32(2) as u64 == input[0] as u64); - assert(sqrt[1].pow_32(2) as u64 == input[1] as u64); -} - -#[test] -fn test() { - let input = [4, 16]; - main(input); -} -``` - -:::info - -Why square root? - -In general, computing square roots is computationally more expensive than multiplications, which takes a toll when speaking about ZK applications. In this case, instead of calculating the square root in Noir, we are using our oracle to offload that computation to be made in plain. In our circuit we can simply multiply the two values. - -::: - -Now, we should write the correspondent RPC server, starting with the [default JSON-RPC 2.0 boilerplate](https://www.npmjs.com/package/json-rpc-2.0#example): - -```js -import { JSONRPCServer } from "json-rpc-2.0"; -import express from "express"; -import bodyParser from "body-parser"; - -const app = express(); -app.use(bodyParser.json()); - -const server = new JSONRPCServer(); -app.post("/", (req, res) => { - const jsonRPCRequest = req.body; - server.receive(jsonRPCRequest).then((jsonRPCResponse) => { - if (jsonRPCResponse) { - res.json(jsonRPCResponse); - } else { - res.sendStatus(204); - } - }); -}); - -app.listen(5555); -``` - -Now, we will add our `getSqrt` method, as expected by the `#[oracle(getSqrt)]` decorator in our Noir code. It maps through the params array and returns their square roots: - -```js -server.addMethod("resolve_foreign_call", async (params) => { - if (params[0].function !== "getSqrt") { - throw Error("Unexpected foreign call") - }; - const values = params[0].inputs[0].map((field) => { - return `${Math.sqrt(parseInt(field, 16))}`; - }); - return { values: [values] }; -}); -``` - -If you're using Typescript, the following types may be helpful in understanding the expected return value and making sure they're easy to follow: - -```js -export type ForeignCallSingle = string; - -export type ForeignCallArray = string[]; - -export type ForeignCallResult = { - values: (ForeignCallSingle | ForeignCallArray)[]; -}; -``` - -:::info Multidimensional Arrays - -If the Oracle function is returning an array containing other arrays, such as `[['1','2],['3','4']]`, you need to provide the values in JSON as flattened values. In the previous example, it would be `['1', '2', '3', '4']`. In the Noir program, the Oracle signature can use a nested type, the flattened values will be automatically converted to the nested type. - -::: - -## Step 3 - Usage with Nargo - -Using the [`nargo` CLI tool](../reference/nargo_commands.md), you can use oracles in the `nargo test` and `nargo execute` commands by passing a value to `--oracle-resolver`. For example: - -```bash -nargo test --oracle-resolver http://localhost:5555 -``` - -This tells `nargo` to use your RPC Server URL whenever it finds an oracle decorator. - -## Step 4 - Usage with NoirJS - -In a JS environment, an RPC server is not strictly necessary, as you may want to resolve your oracles without needing any JSON call at all. NoirJS simply expects that you pass a callback function when you generate proofs, and that callback function can be anything. - -For example, if your Noir program expects the host machine to provide CPU pseudo-randomness, you could simply pass it as the `foreignCallHandler`. You don't strictly need to create an RPC server to serve pseudo-randomness, as you may as well get it directly in your app: - -```js -const foreignCallHandler = (name, inputs) => crypto.randomBytes(16) // etc - -await noir.execute(inputs, foreignCallHandler) -``` - -As one can see, in NoirJS, the [`foreignCallHandler`](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) function simply means "a callback function that returns a value of type [`ForeignCallOutput`](../reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md). It doesn't have to be an RPC call like in the case for Nargo. - -:::tip - -Does this mean you don't have to write an RPC server like in [Step #2](#step-2---write-an-rpc-server)? - -You don't technically have to, but then how would you run `nargo test`? To use both `Nargo` and `NoirJS` in your development flow, you will have to write a JSON RPC server. - -::: - -In this case, let's make `foreignCallHandler` call the JSON RPC Server we created in [Step #2](#step-2---write-an-rpc-server), by making it a JSON RPC Client. - -For example, using the same `getSqrt` program in [Step #1](#step-1---modify-your-noir-program) (comments in the code): - -```js -import { JSONRPCClient } from "json-rpc-2.0"; - -// declaring the JSONRPCClient -const client = new JSONRPCClient((jsonRPCRequest) => { -// hitting the same JSON RPC Server we coded above - return fetch("http://localhost:5555", { - method: "POST", - headers: { - "content-type": "application/json", - }, - body: JSON.stringify(jsonRPCRequest), - }).then((response) => { - if (response.status === 200) { - return response - .json() - .then((jsonRPCResponse) => client.receive(jsonRPCResponse)); - } else if (jsonRPCRequest.id !== undefined) { - return Promise.reject(new Error(response.statusText)); - } - }); -}); - -// declaring a function that takes the name of the foreign call (getSqrt) and the inputs -const foreignCallHandler = async (name, input) => { - const inputs = input[0].map((i) => i.toString("hex")) - // notice that the "inputs" parameter contains *all* the inputs - // in this case we to make the RPC request with the first parameter "numbers", which would be input[0] - const oracleReturn = await client.request("resolve_foreign_call", [ - { - function: name, - inputs: [inputs] - }, - ]); - return [oracleReturn.values[0]]; -}; - -// the rest of your NoirJS code -const input = { input: [4, 16] }; -const { witness } = await noir.execute(input, foreignCallHandler); -``` - -:::tip - -If you're in a NoirJS environment running your RPC server together with a frontend app, you'll probably hit a familiar problem in full-stack development: requests being blocked by [CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/CORS) policy. For development only, you can simply install and use the [`cors` npm package](https://www.npmjs.com/package/cors) to get around the problem: - -```bash -yarn add cors -``` - -and use it as a middleware: - -```js -import cors from "cors"; - -const app = express(); -app.use(cors()) -``` - -::: - -## Conclusion - -Hopefully by the end of this guide, you should be able to: - -- Write your own logic around Oracles and how to write a JSON RPC server to make them work with your Nargo commands. -- Provide custom foreign call handlers for NoirJS. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-recursion.md deleted file mode 100644 index fac79a9cf49..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-recursion.md +++ /dev/null @@ -1,172 +0,0 @@ ---- -title: How to use recursion on NoirJS -description: Learn how to implement recursion with NoirJS, a powerful tool for creating smart contracts on the EVM blockchain. This guide assumes familiarity with NoirJS, solidity verifiers, and the Barretenberg proving backend. Discover how to generate both final and intermediate proofs using `noir_js` and `bb.js`. -keywords: - [ - "NoirJS", - "EVM blockchain", - "smart contracts", - "recursion", - "solidity verifiers", - "Barretenberg backend", - "noir_js", - "intermediate proofs", - "final proofs", - "nargo compile", - "json import", - "recursive circuit", - "recursive app" - ] -sidebar_position: 1 ---- - -This guide shows you how to use recursive proofs in your NoirJS app. For the sake of clarity, it is assumed that: - -- You already have a NoirJS app. If you don't, please visit the [NoirJS tutorial](../tutorials/noirjs_app.md) and the [reference](../reference/NoirJS/noir_js/index.md). -- You are familiar with what are recursive proofs and you have read the [recursion explainer](../explainers/explainer-recursion.md) -- You already built a recursive circuit following [the reference](../noir/standard_library/recursion.mdx), and understand how it works. - -It is also assumed that you're not using `noir_wasm` for compilation, and instead you've used [`nargo compile`](../reference/nargo_commands.md) to generate the `json` you're now importing into your project. However, the guide should work just the same if you're using `noir_wasm`. - -:::info - -As you've read in the [explainer](../explainers/explainer-recursion.md), a recursive proof is an intermediate proof. This means that it doesn't necessarily generate the final step that makes it verifiable in a smart contract. However, it is easy to verify within another circuit. - -::: - -In a standard recursive app, you're also dealing with at least two circuits. For the purpose of this guide, we will assume the following: - -- `main`: a circuit of type `assert(x != y)`, where `main` is marked with a `#[recursive]` attribute. This attribute states that the backend should generate proofs that are friendly for verification within another circuit. -- `recursive`: a circuit that verifies `main` - -For a full example of how recursive proofs work, please refer to the [noir-examples](https://github.com/noir-lang/noir-examples) repository. We will *not* be using it as a reference for this guide. - -## Step 1: Setup - -In a common NoirJS app, you need to instantiate a backend with something like `const backend = new Backend(circuit)`. Then you feed it to the `noir_js` interface. - -For recursion, this doesn't happen, and the only need for `noir_js` is only to `execute` a circuit and get its witness and return value. Everything else is not interfaced, so it needs to happen on the `backend` object. - -It is also recommended that you instantiate the backend with as many threads as possible, to allow for maximum concurrency: - -```js -const backend = new Backend(circuit, { threads: 8 }) -``` - -:::tip -You can use the [`os.cpus()`](https://nodejs.org/api/os.html#oscpus) object in `nodejs` or [`navigator.hardwareConcurrency`](https://developer.mozilla.org/en-US/docs/Web/API/Navigator/hardwareConcurrency) on the browser to make the most out of those glorious cpu cores -::: - -## Step 2: Generating the witness and the proof for `main` - -After instantiating the backend, you should also instantiate `noir_js`. We will use it to execute the circuit and get the witness. - -```js -const noir = new Noir(circuit) -const { witness } = noir.execute(input) -``` - -With this witness, you are now able to generate the intermediate proof for the main circuit: - -```js -const { proof, publicInputs } = await backend.generateProof(witness) -``` - -:::warning - -Always keep in mind what is actually happening on your development process, otherwise you'll quickly become confused about what circuit we are actually running and why! - -In this case, you can imagine that Alice (running the `main` circuit) is proving something to Bob (running the `recursive` circuit), and Bob is verifying her proof within his proof. - -With this in mind, it becomes clear that our intermediate proof is the one *meant to be verified within another circuit*, so it must be Alice's. Actually, the only final proof in this theoretical scenario would be the last one, sent on-chain. - -::: - -## Step 3 - Verification and proof artifacts - -Optionally, you are able to verify the intermediate proof: - -```js -const verified = await backend.verifyProof({ proof, publicInputs }) -``` - -This can be useful to make sure our intermediate proof was correctly generated. But the real goal is to do it within another circuit. For that, we need to generate recursive proof artifacts that will be passed to the circuit that is verifying the proof we just generated. Instead of passing the proof and verification key as a byte array, we pass them as fields which makes it cheaper to verify in a circuit: - -```js -const { proofAsFields, vkAsFields, vkHash } = await backend.generateRecursiveProofArtifacts( { publicInputs, proof }, publicInputsCount) -``` - -This call takes the public inputs and the proof, but also the public inputs count. While this is easily retrievable by simply counting the `publicInputs` length, the backend interface doesn't currently abstract it away. - -:::info - -The `proofAsFields` has a constant size `[Field; 93]` and verification keys in Barretenberg are always `[Field; 114]`. - -::: - -:::warning - -One common mistake is to forget *who* makes this call. - -In a situation where Alice is generating the `main` proof, if she generates the proof artifacts and sends them to Bob, which gladly takes them as true, this would mean Alice could prove anything! - -Instead, Bob needs to make sure *he* extracts the proof artifacts, using his own instance of the `main` circuit backend. This way, Alice has to provide a valid proof for the correct `main` circuit. - -::: - -## Step 4 - Recursive proof generation - -With the artifacts, generating a recursive proof is no different from a normal proof. You simply use the `backend` (with the recursive circuit) to generate it: - -```js -const recursiveInputs = { - verification_key: vkAsFields, // array of length 114 - proof: proofAsFields, // array of length 93 + size of public inputs - publicInputs: [mainInput.y], // using the example above, where `y` is the only public input - key_hash: vkHash, -} - -const { witness, returnValue } = noir.execute(recursiveInputs) // we're executing the recursive circuit now! -const { proof, publicInputs } = backend.generateProof(witness) -const verified = backend.verifyProof({ proof, publicInputs }) -``` - -You can obviously chain this proof into another proof. In fact, if you're using recursive proofs, you're probably interested of using them this way! - -:::tip - -Managing circuits and "who does what" can be confusing. To make sure your naming is consistent, you can keep them in an object. For example: - -```js -const circuits = { - main: mainJSON, - recursive: recursiveJSON -} -const backends = { - main: new BarretenbergBackend(circuits.main), - recursive: new BarretenbergBackend(circuits.recursive) -} -const noir_programs = { - main: new Noir(circuits.main), - recursive: new Noir(circuits.recursive) -} -``` - -This allows you to neatly call exactly the method you want without conflicting names: - -```js -// Alice runs this 👇 -const { witness: mainWitness } = await noir_programs.main.execute(input) -const proof = await backends.main.generateProof(mainWitness) - -// Bob runs this 👇 -const verified = await backends.main.verifyProof(proof) -const { proofAsFields, vkAsFields, vkHash } = await backends.main.generateRecursiveProofArtifacts( - proof, - numPublicInputs, -); -const { witness: recursiveWitness } = await noir_programs.recursive.execute(recursiveInputs) -const recursiveProof = await backends.recursive.generateProof(recursiveWitness); -``` - -::: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/merkle-proof.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/merkle-proof.mdx deleted file mode 100644 index 0a128adb2de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/merkle-proof.mdx +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Prove Merkle Tree Membership -description: - Learn how to use merkle membership proof in Noir to prove that a given leaf is a member of a - merkle tree with a specified root, at a given index. -keywords: - [merkle proof, merkle membership proof, Noir, rust, hash function, Pedersen, sha256, merkle tree] -sidebar_position: 4 ---- - -Let's walk through an example of a merkle membership proof in Noir that proves that a given leaf is -in a merkle tree. - -```rust - -fn main(message : [Field; 62], index : Field, hashpath : [Field; 40], root : Field) { - let leaf = std::hash::hash_to_field(message.as_slice()); - let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); - assert(merkle_root == root); -} - -``` - -The message is hashed using `hash_to_field`. The specific hash function that is being used is chosen -by the backend. The only requirement is that this hash function can heuristically be used as a -random oracle. If only collision resistance is needed, then one can call `std::hash::pedersen_hash` -instead. - -```rust -let leaf = std::hash::hash_to_field(message.as_slice()); -``` - -The leaf is then passed to a compute_merkle_root function with the root, index and hashpath. The returned root can then be asserted to be the same as the provided root. - -```rust -let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); -assert (merkle_root == root); -``` - -> **Note:** It is possible to re-implement the merkle tree implementation without standard library. -> However, for most usecases, it is enough. In general, the standard library will always opt to be -> as conservative as possible, while striking a balance with efficiency. - -An example, the merkle membership proof, only requires a hash function that has collision -resistance, hence a hash function like Pedersen is allowed, which in most cases is more efficient -than the even more conservative sha256. - -[View an example on the starter repo](https://github.com/noir-lang/noir-examples/blob/3ea09545cabfa464124ec2f3ea8e60c608abe6df/stealthdrop/circuits/src/main.nr#L20) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/using-devcontainers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/using-devcontainers.mdx deleted file mode 100644 index 727ec6ca667..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/using-devcontainers.mdx +++ /dev/null @@ -1,110 +0,0 @@ ---- -title: Developer Containers and Codespaces -description: "Learn how to set up a devcontainer in your GitHub repository for a seamless coding experience with Codespaces. Follow our easy 8-step guide to create your own Noir environment without installing Nargo locally." -keywords: ["Devcontainer", "Codespaces", "GitHub", "Noir Environment", "Docker Image", "Development Environment", "Remote Coding", "GitHub Codespaces", "Noir Programming", "Nargo", "VSCode Extensions", "Noirup"] -sidebar_position: 1 ---- - -Adding a developer container configuration file to your Noir project is one of the easiest way to unlock coding in browser. - -## What's a devcontainer after all? - -A [Developer Container](https://containers.dev/) (devcontainer for short) is a Docker image that comes preloaded with tools, extensions, and other tools you need to quickly get started or continue a project, without having to install Nargo locally. Think of it as a development environment in a box. - -There are many advantages to this: - -- It's platform and architecture agnostic -- You don't need to have an IDE installed, or Nargo, or use a terminal at all -- It's safer for using on a public machine or public network - -One of the best ways of using devcontainers is... not using your machine at all, for maximum control, performance, and ease of use. -Enter Codespaces. - -## Codespaces - -If a devcontainer is just a Docker image, then what stops you from provisioning a `p3dn.24xlarge` AWS EC2 instance with 92 vCPUs and 768 GiB RAM and using it to prove your 10-gate SNARK proof? - -Nothing! Except perhaps the 30-40$ per hour it will cost you. - -The problem is that provisioning takes time, and I bet you don't want to see the AWS console every time you want to code something real quick. - -Fortunately, there's an easy and free way to get a decent remote machine ready and loaded in less than 2 minutes: Codespaces. [Codespaces is a Github feature](https://github.com/features/codespaces) that allows you to code in a remote machine by using devcontainers, and it's pretty cool: - -- You can start coding Noir in less than a minute -- It uses the resources of a remote machine, so you can code on your grandma's phone if needed be -- It makes it easy to share work with your frens -- It's fully reusable, you can stop and restart whenever you need to - -:::info - -Don't take out your wallet just yet. Free GitHub accounts get about [15-60 hours of coding](https://github.com/features/codespaces) for free per month, depending on the size of your provisioned machine. - -::: - -## Tell me it's _actually_ easy - -It is! - -Github comes with a default codespace and you can use it to code your own devcontainer. That's exactly what we will be doing in this guide. - - - -8 simple steps: - -#### 1. Create a new repository on GitHub. - -#### 2. Click "Start coding with Codespaces". This will use the default image. - -#### 3. Create a folder called `.devcontainer` in the root of your repository. - -#### 4. Create a Dockerfile in that folder, and paste the following code: - -```docker -FROM --platform=linux/amd64 node:lts-bookworm-slim -SHELL ["/bin/bash", "-c"] -RUN apt update && apt install -y curl bash git tar gzip libc++-dev -RUN curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -ENV PATH="/root/.nargo/bin:$PATH" -RUN noirup -ENTRYPOINT ["nargo"] -``` -#### 5. Create a file called `devcontainer.json` in the same folder, and paste the following code: - -```json -{ - "name": "Noir on Codespaces", - "build": { - "context": ".", - "dockerfile": "Dockerfile" - }, - "customizations": { - "vscode": { - "extensions": ["noir-lang.vscode-noir"] - } - } -} -``` -#### 6. Commit and push your changes - -This will pull the new image and build it, so it could take a minute or so - -#### 8. Done! -Just wait for the build to finish, and there's your easy Noir environment. - - -Refer to [noir-starter](https://github.com/noir-lang/noir-starter/) as an example of how devcontainers can be used together with codespaces. - - - -## How do I use it? - -Using the codespace is obviously much easier than setting it up. -Just navigate to your repository and click "Code" -> "Open with Codespaces". It should take a few seconds to load, and you're ready to go. - -:::info - -If you really like the experience, you can add a badge to your readme, links to existing codespaces, and more. -Check out the [official docs](https://docs.github.com/en/codespaces/setting-up-your-project-for-codespaces/setting-up-your-repository/facilitating-quick-creation-and-resumption-of-codespaces) for more info. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/index.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/index.mdx deleted file mode 100644 index 5c116a73b3f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/index.mdx +++ /dev/null @@ -1,76 +0,0 @@ ---- -title: Noir Lang -hide_title: true -description: - Learn about the public alpha release of Noir, a domain specific language heavily influenced by Rust that compiles to - an intermediate language which can be compiled to an arithmetic circuit or a rank-1 constraint system. -keywords: - [Noir, - Domain Specific Language, - Rust, - Intermediate Language, - Arithmetic Circuit, - Rank-1 Constraint System, - Ethereum Developers, - Protocol Developers, - Blockchain Developers, - Proving System, - Smart Contract Language] -sidebar_position: 0 ---- - -import Tabs from '@theme/Tabs'; -import TabItem from '@theme/TabItem'; -import ThemedImage from '@theme/ThemedImage'; -import useBaseUrl from '@docusaurus/useBaseUrl'; - - - -Noir is an open-source Domain-Specific Language for safe and seamless construction of privacy-preserving Zero-Knowledge programs, requiring no previous knowledge on the underlying mathematics or cryptography. - -ZK programs are programs that can generate short proofs of statements without revealing all inputs to the statements. You can read more about Zero-Knowledge Proofs [here](https://dev.to/spalladino/a-beginners-intro-to-coding-zero-knowledge-proofs-c56). - -## What's new about Noir? - -Noir works differently from most ZK languages by taking a two-pronged path. First, it compiles the program to an adaptable intermediate language known as ACIR. From there, depending on a given project's needs, ACIR can be further compiled into an arithmetic circuit for integration with the proving backend. - -:::info - -Noir is backend agnostic, which means it makes no assumptions on which proving backend powers the ZK proof. Being the language that powers [Aztec Contracts](https://docs.aztec.network/developers/contracts/main), it defaults to Aztec's Barretenberg proving backend. - -However, the ACIR output can be transformed to be compatible with other PLONK-based backends, or into a [rank-1 constraint system](https://www.rareskills.io/post/rank-1-constraint-system) suitable for backends such as Arkwork's Marlin. - -::: - -## Who is Noir for? - -Noir can be used both in complex cloud-based backends and in user's smartphones, requiring no knowledge on the underlying math or cryptography. From authorization systems that keep a password in the user's device, to complex on-chain verification of recursive proofs, Noir is designed to abstract away complexity without any significant overhead. Here are some examples of situations where Noir can be used: - - - - Noir Logo - - Aztec Contracts leverage Noir to allow for the storage and execution of private information. Writing an Aztec Contract is as easy as writing Noir, and Aztec developers can easily interact with the network storage and execution through the [Aztec.nr](https://docs.aztec.network/developers/contracts/main) library. - - - Soliditry Verifier Example - Noir can auto-generate Solidity verifier contracts that verify Noir proofs. This allows for non-interactive verification of proofs containing private information in an immutable system. This feature powers a multitude of use-case scenarios, from P2P chess tournaments, to [Aztec Layer-2 Blockchain](https://docs.aztec.network/) - - - Aztec Labs developed NoirJS, an easy interface to generate and verify Noir proofs in a Javascript environment. This allows for Noir to be used in webpages, mobile apps, games, and any other environment supporting JS execution in a standalone manner. - - - - -## Libraries - -Noir is meant to be easy to extend by simply importing Noir libraries just like in Rust. -The [awesome-noir repo](https://github.com/noir-lang/awesome-noir#libraries) is a collection of libraries developed by the Noir community. -Writing a new library is easy and makes code be composable and easy to reuse. See the section on [dependencies](noir/modules_packages_crates/dependencies.md) for more information. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/migration_notes.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/migration_notes.md deleted file mode 100644 index 6bd740024e5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/migration_notes.md +++ /dev/null @@ -1,105 +0,0 @@ ---- -title: Migration notes -description: Read about migration notes from previous versions, which could solve problems while updating -keywords: [Noir, notes, migration, updating, upgrading] ---- - -Noir is in full-speed development. Things break fast, wild, and often. This page attempts to leave some notes on errors you might encounter when upgrading and how to resolve them until proper patches are built. - -### `backend encountered an error: libc++.so.1` - -Depending on your OS, you may encounter the following error when running `nargo prove` for the first time: - -```text -The backend encountered an error: "/home/codespace/.nargo/backends/acvm-backend-barretenberg/backend_binary: error while loading shared libraries: libc++.so.1: cannot open shared object file: No such file or directory\n" -``` - -Install the `libc++-dev` library with: - -```bash -sudo apt install libc++-dev -``` - -## ≥0.19 - -### Enforcing `compiler_version` - -From this version on, the compiler will check for the `compiler_version` field in `Nargo.toml`, and will error if it doesn't match the current Nargo version in use. - -To update, please make sure this field in `Nargo.toml` matches the output of `nargo --version`. - -## ≥0.14 - -The index of the [for loops](noir/concepts/control_flow.md#loops) is now of type `u64` instead of `Field`. An example refactor would be: - -```rust -for i in 0..10 { - let i = i as Field; -} -``` - -## ≥v0.11.0 and Nargo backend - -From this version onwards, Nargo starts managing backends through the `nargo backend` command. Upgrading to the versions per usual steps might lead to: - -### `backend encountered an error` - -This is likely due to the existing locally installed version of proving backend (e.g. barretenberg) is incompatible with the version of Nargo in use. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo prove -``` - -with your Noir program. - -This will trigger the download and installation of the latest version of barretenberg compatible with your Nargo in use. - -### `backend encountered an error: illegal instruction` - -On certain Intel-based systems, an `illegal instruction` error may arise due to incompatibility of barretenberg with certain CPU instructions. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo backend install acvm-backend-barretenberg https://github.com/noir-lang/barretenberg-js-binary/raw/master/run-bb.tar.gz -``` - -This downloads and installs a specific bb.js based version of barretenberg binary from GitHub. - -The gzipped file is running [this bash script](https://github.com/noir-lang/barretenberg-js-binary/blob/master/run-bb-js.sh), where we need to gzip it as the Nargo currently expect the backend to be zipped up. - -Then run: - -``` -DESIRED_BINARY_VERSION=0.8.1 nargo info -``` - -This overrides the bb native binary with a bb.js node application instead, which should be compatible with most if not all hardware. This does come with the drawback of being generally slower than native binary. - -0.8.1 indicates bb.js version 0.8.1, so if you change that it will update to a different version or the default version in the script if none was supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/_category_.json deleted file mode 100644 index 7da08f8a8c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Concepts", - "position": 0, - "collapsible": true, - "collapsed": true -} \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/assert.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/assert.md deleted file mode 100644 index 2132de42072..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/assert.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Assert Function -description: - Learn about the `assert` and `static_assert` functions in Noir, which can be used to explicitly - constrain the predicate or comparison expression that follows to be true, and what happens if - the expression is false at runtime or compile-time, respectively. -keywords: [Noir programming language, assert statement, predicate expression, comparison expression] -sidebar_position: 4 ---- - -Noir includes a special `assert` function which will explicitly constrain the predicate/comparison -expression that follows to be true. If this expression is false at runtime, the program will fail to -be proven. Example: - -```rust -fn main(x : Field, y : Field) { - assert(x == y); -} -``` - -> Assertions only work for predicate operations, such as `==`. If there's any ambiguity on the operation, the program will fail to compile. For example, it is unclear if `assert(x + y)` would check for `x + y == 0` or simply would return `true`. - -You can optionally provide a message to be logged when the assertion fails: - -```rust -assert(x == y, "x and y are not equal"); -``` - -Aside string literals, the optional message can be a format string or any other type supported as input for Noir's [print](../standard_library/logging.md) functions. This feature lets you incorporate runtime variables into your failed assertion logs: - -```rust -assert(x == y, f"Expected x == y, but got {x} == {y}"); -``` - -Using a variable as an assertion message directly: - -```rust -struct myStruct { - myField: Field -} - -let s = myStruct { myField: y }; -assert(s.myField == x, s); -``` - -There is also a special `static_assert` function that behaves like `assert`, -but that runs at compile-time. - -```rust -fn main(xs: [Field; 3]) { - let x = 2 + 2; - let y = 4; - static_assert(x == y, "expected 2 + 2 to equal 4"); - - // This passes since the length of `xs` is known at compile-time - static_assert(xs.len() == 3, "expected the input to have 3 elements"); -} -``` - -This function fails when passed a dynamic (run-time) argument: - -```rust -fn main(x : Field, y : Field) { - // this fails because `x` is not known at compile-time - static_assert(x == 2, "expected x to be known at compile-time and equal to 2"); - - let mut example_slice = &[]; - if y == 4 { - example_slice = example_slice.push_back(0); - } - - // This fails because the length of `example_slice` is not known at - // compile-time - let error_message = "expected an empty slice, known at compile-time"; - static_assert(example_slice.len() == 0, error_message); -} -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/comments.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/comments.md deleted file mode 100644 index b51a85f5c94..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/comments.md +++ /dev/null @@ -1,33 +0,0 @@ ---- -title: Comments -description: - Learn how to write comments in Noir programming language. A comment is a line of code that is - ignored by the compiler, but it can be read by programmers. Single-line and multi-line comments - are supported in Noir. -keywords: [Noir programming language, comments, single-line comments, multi-line comments] -sidebar_position: 10 ---- - -A comment is a line in your codebase which the compiler ignores, however it can be read by -programmers. - -Here is a single line comment: - -```rust -// This is a comment and is ignored -``` - -`//` is used to tell the compiler to ignore the rest of the line. - -Noir also supports multi-line block comments. Start a block comment with `/*` and end the block with `*/`. - -Noir does not natively support doc comments. You may be able to use [Rust doc comments](https://doc.rust-lang.org/reference/comments.html) in your code to leverage some Rust documentation build tools with Noir code. - -```rust -/* - This is a block comment describing a complex function. -*/ -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/comptime.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/comptime.md deleted file mode 100644 index 2ceb030c7e1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/comptime.md +++ /dev/null @@ -1,445 +0,0 @@ ---- -title: Compile-time Code & Metaprogramming -description: Learn how to use metaprogramming in Noir to create macros or derive your own traits -keywords: [Noir, comptime, compile-time, metaprogramming, macros, quote, unquote] -sidebar_position: 15 ---- - -## Overview - -Metaprogramming in Noir is comprised of three parts: -1. `comptime` code -2. Quoting and unquoting -3. The metaprogramming API in `std::meta` - -Each of these are explained in more detail in the next sections but the wide picture is that -`comptime` allows us to write code which runs at compile-time. In this `comptime` code we -can quote and unquote snippets of the program, manipulate them, and insert them in other -parts of the program. Comptime functions which do this are said to be macros. Additionally, -there's a compile-time API of built-in types and functions provided by the compiler which allows -for greater analysis and modification of programs. - ---- - -## Comptime - -`comptime` is a new keyword in Noir which marks an item as executing or existing at compile-time. It can be used in several ways: - -- `comptime fn` to define functions which execute exclusively during compile-time. -- `comptime global` to define a global variable which is evaluated at compile-time. - - Unlike runtime globals, `comptime global`s can be mutable. -- `comptime { ... }` to execute a block of statements during compile-time. -- `comptime let` to define a variable whose value is evaluated at compile-time. -- `comptime for` to run a for loop at compile-time. Syntax sugar for `comptime { for .. }`. - -### Scoping - -Note that while in a `comptime` context, any runtime variables _local to the current function_ are never visible. - -### Evaluating - -Evaluation rules of `comptime` follows the normal unconstrained evaluation rules for other Noir code. There are a few things to note though: - -- Certain built-in functions may not be available, although more may be added over time. -- Evaluation order of global items is currently unspecified. For example, given the following two functions we can't guarantee -which `println` will execute first. The ordering of the two printouts will be arbitrary, but should be stable across multiple compilations with the same `nargo` version as long as the program is also unchanged. - -```rust -fn one() { - comptime { println("one"); } -} - -fn two() { - comptime { println("two"); } -} -``` - -- Since evaluation order is unspecified, care should be taken when using mutable globals so that they do not rely on a particular ordering. -For example, using globals to generate unique ids should be fine but relying on certain ids always being produced (especially after edits to the program) should be avoided. -- Although most ordering of globals is unspecified, two are: - - Dependencies of a crate will always be evaluated before the dependent crate. - - Any annotations on a function will be run before the function itself is resolved. This is to allow the annotation to modify the function if necessary. Note that if the - function itself was called at compile-time previously, it will already be resolved and cannot be modified. To prevent accidentally calling functions you wish to modify - at compile-time, it may be helpful to sort your `comptime` annotation functions into a different crate along with any dependencies they require. - -### Lowering - -When a `comptime` value is used in runtime code it must be lowered into a runtime value. This means replacing the expression with the literal that it evaluated to. For example, the code: - -```rust -struct Foo { array: [Field; 2], len: u32 } - -fn main() { - println(comptime { - let mut foo = std::mem::zeroed::(); - foo.array[0] = 4; - foo.len = 1; - foo - }); -} -``` - -will be converted to the following after `comptime` expressions are evaluated: - -```rust -struct Foo { array: [Field; 2], len: u32 } - -fn main() { - println(Foo { array: [4, 0], len: 1 }); -} -``` - -Not all types of values can be lowered. For example, `Type`s and `TypeDefinition`s (among other types) cannot be lowered at all. - -```rust -fn main() { - // There's nothing we could inline here to create a Type value at runtime - // let _ = get_type!(); -} - -comptime fn get_type() -> Type { ... } -``` - ---- - -## (Quasi) Quote - -Macros in Noir are `comptime` functions which return code as a value which is inserted into the call site when it is lowered there. -A code value in this case is of type `Quoted` and can be created by a `quote { ... }` expression. -More specifically, the code value `quote` creates is a token stream - a representation of source code as a series of words, numbers, string literals, or operators. -For example, the expression `quote { Hi "there reader"! }` would quote three tokens: the word "hi", the string "there reader", and an exclamation mark. -You'll note that snippets that would otherwise be invalid syntax can still be quoted. - -When a `Quoted` value is used in runtime code, it is lowered into a `quote { ... }` expression. Since this expression is only valid -in compile-time code however, we'd get an error if we tried this. Instead, we can use macro insertion to insert each token into the -program at that point, and parse it as an expression. To do this, we have to add a `!` after the function name returning the `Quoted` value. -If the value was created locally and there is no function returning it, `std::meta::unquote!(_)` can be used instead. -Calling such a function at compile-time without `!` will just return the `Quoted` value to be further manipulated. For example: - -```rust title="quote-example" showLineNumbers -comptime fn quote_one() -> Quoted { - quote { 1 } - } - - #[test] - fn returning_versus_macro_insertion() { - comptime { - // let _a: Quoted = quote { 1 }; - let _a: Quoted = quote_one(); - - // let _b: Field = 1; - let _b: Field = quote_one!(); - - // Since integers default to fields, if we - // want a different type we have to explicitly cast - // let _c: i32 = 1 as i32; - let _c: i32 = quote_one!() as i32; - } - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L120-L140 - - -For those familiar with quoting from other languages (primarily lisps), Noir's `quote` is actually a _quasiquote_. -This means we can escape the quoting by using the unquote operator to splice values in the middle of quoted code. - -## Unquote - -The unquote operator `$` is usable within a `quote` expression. -It takes a variable as an argument, evaluates the variable, and splices the resulting value into the quoted token stream at that point. For example, - -```rust -comptime { - let x = 1 + 2; - let y = quote { $x + 4 }; -} -``` - -The value of `y` above will be the token stream containing `3`, `+`, and `4`. We can also use this to combine `Quoted` values into larger token streams: - -```rust -comptime { - let x = quote { 1 + 2 }; - let y = quote { $x + 4 }; -} -``` - -The value of `y` above is now the token stream containing five tokens: `1 + 2 + 4`. - -Note that to unquote something, a variable name _must_ follow the `$` operator in a token stream. -If it is an expression (even a parenthesized one), it will do nothing. Most likely a parse error will be given when the macro is later unquoted. - -Unquoting can also be avoided by escaping the `$` with a backslash: - -``` -comptime { - let x = quote { 1 + 2 }; - - // y contains the four tokens: `$x + 4` - let y = quote { \$x + 4 }; -} -``` - ---- - -## Annotations - -Annotations provide a way to run a `comptime` function on an item in the program. -When you use an annotation, the function with the same name will be called with that item as an argument: - -```rust -#[my_struct_annotation] -struct Foo {} - -comptime fn my_struct_annotation(s: StructDefinition) { - println("Called my_struct_annotation!"); -} - -#[my_function_annotation] -fn foo() {} - -comptime fn my_function_annotation(f: FunctionDefinition) { - println("Called my_function_annotation!"); -} -``` - -Anything returned from one of these functions will be inserted at top-level along with the original item. -Note that expressions are not valid at top-level so you'll get an error trying to return `3` or similar just as if you tried to write a program containing `3; struct Foo {}`. -You can insert other top-level items such as trait impls, structs, or functions this way though. -For example, this is the mechanism used to insert additional trait implementations into the program when deriving a trait impl from a struct: - -```rust title="derive-field-count-example" showLineNumbers -trait FieldCount { - fn field_count() -> u32; - } - - #[derive_field_count] - struct Bar { - x: Field, - y: [Field; 2], - } - - comptime fn derive_field_count(s: StructDefinition) -> Quoted { - let typ = s.as_type(); - let field_count = s.fields().len(); - quote { - impl FieldCount for $typ { - fn field_count() -> u32 { - $field_count - } - } - } - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L142-L164 - - -### Calling annotations with additional arguments - -Arguments may optionally be given to annotations. -When this is done, these additional arguments are passed to the annotation function after the item argument. - -```rust title="annotation-arguments-example" showLineNumbers -#[assert_field_is_type(quote { i32 }.as_type())] - struct MyStruct { - my_field: i32, - } - - comptime fn assert_field_is_type(s: StructDefinition, typ: Type) { - // Assert the first field in `s` has type `typ` - let fields = s.fields(); - assert_eq(fields[0].1, typ); - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L166-L177 - - -We can also take any number of arguments by adding the `varargs` annotation: - -```rust title="annotation-varargs-example" showLineNumbers -#[assert_three_args(1, 2, 3)] - struct MyOtherStruct { - my_other_field: u32, - } - - #[varargs] - comptime fn assert_three_args(_s: StructDefinition, args: [Field]) { - assert_eq(args.len(), 3); - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L179-L189 - - ---- - -## Comptime API - -Although `comptime`, `quote`, and unquoting provide a flexible base for writing macros, -Noir's true metaprogramming ability comes from being able to interact with the compiler through a compile-time API. -This API can be accessed through built-in functions in `std::meta` as well as on methods of several `comptime` types. - -The following is an incomplete list of some `comptime` types along with some useful methods on them. You can see more in the standard library [Metaprogramming section](../standard_library/meta). - -- `Quoted`: A token stream -- `Type`: The type of a Noir type - - `fn implements(self, constraint: TraitConstraint) -> bool` - - Returns true if `self` implements the given trait constraint -- `Expr`: A syntactically valid expression. Can be used to recur on a program's parse tree to inspect how it is structured. - - Methods: - - `fn as_function_call(self) -> Option<(Expr, [Expr])>` - - If this is a function call expression, return `(function, arguments)` - - `fn as_block(self) -> Option<[Expr]>` - - If this is a block, return each statement in the block -- `FunctionDefinition`: A function definition - - Methods: - - `fn parameters(self) -> [(Quoted, Type)]` - - Returns a slice of `(name, type)` pairs for each parameter -- `StructDefinition`: A struct definition - - Methods: - - `fn as_type(self) -> Type` - - Returns this `StructDefinition` as a `Type`. Any generics are kept as-is - - `fn generics(self) -> [Quoted]` - - Return the name of each generic on this struct - - `fn fields(self) -> [(Quoted, Type)]` - - Return the name and type of each field -- `TraitConstraint`: A trait constraint such as `From` -- `TypedExpr`: A type-checked expression. -- `UnresolvedType`: A syntactic notation that refers to a Noir type that hasn't been resolved yet - -There are many more functions available by exploring the `std::meta` module and its submodules. -Using these methods is the key to writing powerful metaprogramming libraries. - -### `#[use_callers_scope]` - -Since certain functions such as `Quoted::as_type`, `Expression::as_type`, or `Quoted::as_trait_constraint` will attempt -to resolve their contents in a particular scope - it can be useful to change the scope they resolve in. By default -these functions will resolve in the current function's scope which is usually the attribute function they are called in. -If you're working on a library however, this may be a completely different module or crate to the item you're trying to -use the attribute on. If you want to be able to use `Quoted::as_type` to refer to types local to the caller's scope for -example, you can annotate your attribute function with `#[use_callers_scope]`. This will ensure your attribute, and any -closures it uses, can refer to anything in the caller's scope. `#[use_callers_scope]` also works recursively. So if both -your attribute function and a helper function it calls use it, then they can both refer to the same original caller. - ---- - -## Example: Derive - -Using all of the above, we can write a `derive` macro that behaves similarly to Rust's but is not built into the language. -From the user's perspective it will look like this: - -```rust -// Example usage -#[derive(Default, Eq, Ord)] -struct MyStruct { my_field: u32 } -``` - -To implement `derive` we'll have to create a `comptime` function that accepts -a variable amount of traits. - -```rust title="derive_example" showLineNumbers -// These are needed for the unconstrained hashmap we're using to store derive functions -use crate::collections::umap::UHashMap; -use crate::hash::BuildHasherDefault; -use crate::hash::poseidon2::Poseidon2Hasher; - -// A derive function is one that given a struct definition can -// create us a quoted trait impl from it. -pub type DeriveFunction = fn(StructDefinition) -> Quoted; - -// We'll keep a global HANDLERS map to keep track of the derive handler for each trait -comptime mut global HANDLERS: UHashMap> = - UHashMap::default(); - -// Given a struct and a slice of traits to derive, create trait impls for each. -// This function is as simple as iterating over the slice, checking if we have a trait -// handler registered for the given trait, calling it, and appending the result. -#[varargs] -pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted { - let mut result = quote {}; - - for trait_to_derive in traits { - let handler = unsafe { HANDLERS.get(trait_to_derive) }; - assert(handler.is_some(), f"No derive function registered for `{trait_to_derive}`"); - - let trait_impl = handler.unwrap()(s); - result = quote { $result $trait_impl }; - } - - result -} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L31-L64 - - -Registering a derive function could be done as follows: - -```rust title="derive_via" showLineNumbers -// To register a handler for a trait, just add it to our handlers map -pub comptime fn derive_via(t: TraitDefinition, f: DeriveFunction) { - HANDLERS.insert(t, f); -} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L66-L73 - - -```rust title="big-derive-usage-example" showLineNumbers -// Finally, to register a handler we call the above function as an annotation - // with our handler function. - #[derive_via(derive_do_nothing)] - trait DoNothing { - fn do_nothing(self); - } - - comptime fn derive_do_nothing(s: StructDefinition) -> Quoted { - // This is simplified since we don't handle generics or where clauses! - // In a real example we'd likely also need to introduce each of - // `s.generics()` as well as a trait constraint for each generic - // to ensure they also implement the trait. - let typ = s.as_type(); - quote { - impl DoNothing for $typ { - fn do_nothing(self) { - // Traits can't tell us what to do - println("something"); - } - } - } - } - - // Since `DoNothing` is a simple trait which: - // 1. Only has one method - // 2. Does not have any generics on the trait itself - // We can use `std::meta::make_trait_impl` to help us out. - // This helper function will generate our impl for us along with any - // necessary where clauses and still provides a flexible interface - // for us to work on each field on the struct. - comptime fn derive_do_nothing_alt(s: StructDefinition) -> Quoted { - let trait_name = quote { DoNothing }; - let method_signature = quote { fn do_nothing(self) }; - - // Call `do_nothing` recursively on each field in the struct - let for_each_field = |field_name| quote { self.$field_name.do_nothing(); }; - - // Some traits like Eq want to join each field expression with something like `&`. - // We don't need that here - let join_fields_with = quote {}; - - // The body function is a spot to insert any extra setup/teardown needed. - // We'll insert our println here. Since we recur on each field, we should see - // one println for the struct itself, followed by a println for every field (recursively). - let body = |body| quote { - println("something"); - $body - }; - crate::meta::make_trait_impl( - s, - trait_name, - method_signature, - for_each_field, - join_fields_with, - body, - ) - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L191-L249 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/control_flow.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/control_flow.md deleted file mode 100644 index b365bb22728..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/control_flow.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Control Flow -description: - Learn how to use loops and if expressions in the Noir programming language. Discover the syntax - and examples for for loops and if-else statements. -keywords: [Noir programming language, loops, for loop, if-else statements, Rust syntax] -sidebar_position: 2 ---- - -## If Expressions - -Noir supports `if-else` statements. The syntax is most similar to Rust's where it is not required -for the statement's conditional to be surrounded by parentheses. - -```rust -let a = 0; -let mut x: u32 = 0; - -if a == 0 { - if a != 0 { - x = 6; - } else { - x = 2; - } -} else { - x = 5; - assert(x == 5); -} -assert(x == 2); -``` - -## Loops - -Noir has one kind of loop: the `for` loop. `for` loops allow you to repeat a block of code multiple -times. - -The following block of code between the braces is run 10 times. - -```rust -for i in 0..10 { - // do something -} -``` - -Alternatively, `start..=end` can be used for a range that is inclusive on both ends. - -The index for loops is of type `u64`. - -### Break and Continue - -In unconstrained code, `break` and `continue` are also allowed in `for` loops. These are only allowed -in unconstrained code since normal constrained code requires that Noir knows exactly how many iterations -a loop may have. `break` and `continue` can be used like so: - -```rust -for i in 0 .. 10 { - println("Iteration start") - - if i == 2 { - continue; - } - - if i == 5 { - break; - } - - println(i); -} -println("Loop end") -``` - -When used, `break` will end the current loop early and jump to the statement after the for loop. In the example -above, the `break` will stop the loop and jump to the `println("Loop end")`. - -`continue` will stop the current iteration of the loop, and jump to the start of the next iteration. In the example -above, `continue` will jump to `println("Iteration start")` when used. Note that the loop continues as normal after this. -The iteration variable `i` is still increased by one as normal when `continue` is used. - -`break` and `continue` cannot currently be used to jump out of more than a single loop at a time. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_bus.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_bus.mdx deleted file mode 100644 index e55e58622ce..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_bus.mdx +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: Data Bus -sidebar_position: 13 ---- -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -The data bus is an optimization that the backend can use to make recursion more efficient. -In order to use it, you must define some inputs of the program entry points (usually the `main()` -function) with the `call_data` modifier, and the return values with the `return_data` modifier. -These modifiers are incompatible with `pub` and `mut` modifiers. - -## Example - -```rust -fn main(mut x: u32, y: call_data u32, z: call_data [u32;4] ) -> return_data u32 { - let a = z[x]; - a+y -} -``` - -As a result, both call_data and return_data will be treated as private inputs and encapsulated into a read-only array each, for the backend to process. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/arrays.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/arrays.md deleted file mode 100644 index 289145a8c4d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/arrays.md +++ /dev/null @@ -1,276 +0,0 @@ ---- -title: Arrays -description: - Dive into the Array data type in Noir. Grasp its methods, practical examples, and best practices for efficiently using Arrays in your Noir code. -keywords: - [ - noir, - array type, - methods, - examples, - indexing, - ] -sidebar_position: 4 ---- - -An array is one way of grouping together values into one compound type. Array types can be inferred -or explicitly specified via the syntax `[; ]`: - -```rust -fn main(x : Field, y : Field) { - let my_arr = [x, y]; - let your_arr: [Field; 2] = [x, y]; -} -``` - -Here, both `my_arr` and `your_arr` are instantiated as an array containing two `Field` elements. - -Array elements can be accessed using indexing: - -```rust -fn main() { - let a = [1, 2, 3, 4, 5]; - - let first = a[0]; - let second = a[1]; -} -``` - -All elements in an array must be of the same type (i.e. homogeneous). That is, an array cannot group -a `Field` value and a `u8` value together for example. - -You can write mutable arrays, like: - -```rust -fn main() { - let mut arr = [1, 2, 3, 4, 5]; - assert(arr[0] == 1); - - arr[0] = 42; - assert(arr[0] == 42); -} -``` - -You can instantiate a new array of a fixed size with the same value repeated for each element. The following example instantiates an array of length 32 where each element is of type Field and has the value 0. - -```rust -let array: [Field; 32] = [0; 32]; -``` - -Like in Rust, arrays in Noir are a fixed size. However, if you wish to convert an array to a [slice](./slices.mdx), you can just call `as_slice` on your array: - -```rust -let array: [Field; 32] = [0; 32]; -let sl = array.as_slice() -``` - -You can define multidimensional arrays: - -```rust -let array : [[Field; 2]; 2]; -let element = array[0][0]; -``` - -However, multidimensional slices are not supported. For example, the following code will error at compile time: - -```rust -let slice : [[Field]] = &[]; -``` - -## Types - -You can create arrays of primitive types or structs. There is not yet support for nested arrays -(arrays of arrays) or arrays of structs that contain arrays. - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for arrays. -Each of these functions are located within the generic impl `impl [T; N] {`. -So anywhere `self` appears, it refers to the variable `self: [T; N]`. - -### len - -Returns the length of an array - -```rust -fn len(self) -> Field -``` - -example - -```rust -fn main() { - let array = [42, 42]; - assert(array.len() == 2); -} -``` - -### sort - -Returns a new sorted array. The original array remains untouched. Notice that this function will -only work for arrays of fields or integers, not for any arbitrary type. This is because the sorting -logic it uses internally is optimized specifically for these values. If you need a sort function to -sort any type, you should use the function `sort_via` described below. - -```rust -fn sort(self) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32]; - let sorted = arr.sort(); - assert(sorted == [32, 42]); -} -``` - -### sort_via - -Sorts the array with a custom comparison function. The ordering function must return true if the first argument should be sorted to be before the second argument or is equal to the second argument. - -Using this method with an operator like `<` that does not return `true` for equal values will result in an assertion failure for arrays with equal elements. - -```rust -fn sort_via(self, ordering: fn(T, T) -> bool) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32] - let sorted_ascending = arr.sort_via(|a, b| a <= b); - assert(sorted_ascending == [32, 42]); // verifies - - let sorted_descending = arr.sort_via(|a, b| a >= b); - assert(sorted_descending == [32, 42]); // does not verify -} -``` - -### map - -Applies a function to each element of the array, returning a new array containing the mapped elements. - -```rust -fn map(self, f: fn(T) -> U) -> [U; N] -``` - -example - -```rust -let a = [1, 2, 3]; -let b = a.map(|a| a * 2); // b is now [2, 4, 6] -``` - -### fold - -Applies a function to each element of the array, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn(U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the array, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = [1]; -let a2 = [1, 2]; -let a3 = [1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let arr = [2, 2, 2, 2, 2]; - let folded = arr.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -Requires `self` to be non-empty. - -```rust -fn reduce(self, f: fn(T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let reduced = arr.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let all = arr.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 5]; - let any = arr.any(|a| a == 5); - assert(any); -} -``` - -### as_str_unchecked - -Converts a byte array of type `[u8; N]` to a string. Note that this performs no UTF-8 validation - -the given array is interpreted as-is as a string. - -```rust -impl [u8; N] { - pub fn as_str_unchecked(self) -> str -} -``` - -example: - -```rust -fn main() { - let hi = [104, 105].as_str_unchecked(); - assert_eq(hi, "hi"); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/booleans.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/booleans.md deleted file mode 100644 index 2507af710e7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/booleans.md +++ /dev/null @@ -1,28 +0,0 @@ ---- -title: Booleans -description: - Delve into the Boolean data type in Noir. Understand its methods, practical examples, and best practices for using Booleans in your Noir programs. -keywords: - [ - noir, - boolean type, - methods, - examples, - logical operations, - ] -sidebar_position: 2 ---- - - -The `bool` type in Noir has two possible values: `true` and `false`: - -```rust -fn main() { - let t = true; - let f: bool = false; -} -``` - -The boolean type is most commonly used in conditionals like `if` expressions and `assert` -statements. More about conditionals is covered in the [Control Flow](../control_flow.md) and -[Assert Function](../assert.md) sections. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/function_types.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/function_types.md deleted file mode 100644 index f6121af17e2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/function_types.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Function types -sidebar_position: 10 ---- - -Noir supports higher-order functions. The syntax for a function type is as follows: - -```rust -fn(arg1_type, arg2_type, ...) -> return_type -``` - -Example: - -```rust -fn assert_returns_100(f: fn() -> Field) { // f takes no args and returns a Field - assert(f() == 100); -} - -fn main() { - assert_returns_100(|| 100); // ok - assert_returns_100(|| 150); // fails -} -``` - -A function type also has an optional capture environment - this is necessary to support closures. -See [Lambdas](../lambdas.md) for more details. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/index.md deleted file mode 100644 index 0f2db2b2d75..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/index.md +++ /dev/null @@ -1,126 +0,0 @@ ---- -title: Data Types -description: - Get a clear understanding of the two categories of Noir data types - primitive types and compound - types. Learn about their characteristics, differences, and how to use them in your Noir - programming. -keywords: - [ - noir, - data types, - primitive types, - compound types, - private types, - public types, - ] ---- - -Every value in Noir has a type, which determines which operations are valid for it. - -All values in Noir are fundamentally composed of `Field` elements. For a more approachable -developing experience, abstractions are added on top to introduce different data types in Noir. - -Noir has two category of data types: primitive types (e.g. `Field`, integers, `bool`) and compound -types that group primitive types (e.g. arrays, tuples, structs). Each value can either be private or -public. - -## Private & Public Types - -A **private value** is known only to the Prover, while a **public value** is known by both the -Prover and Verifier. Mark values as `private` when the value should only be known to the prover. All -primitive types (including individual fields of compound types) in Noir are private by default, and -can be marked public when certain values are intended to be revealed to the Verifier. - -> **Note:** For public values defined in Noir programs paired with smart contract verifiers, once -> the proofs are verified on-chain the values can be considered known to everyone that has access to -> that blockchain. - -Public data types are treated no differently to private types apart from the fact that their values -will be revealed in proofs generated. Simply changing the value of a public type will not change the -circuit (where the same goes for changing values of private types as well). - -_Private values_ are also referred to as _witnesses_ sometimes. - -> **Note:** The terms private and public when applied to a type (e.g. `pub Field`) have a different -> meaning than when applied to a function (e.g. `pub fn foo() {}`). -> -> The former is a visibility modifier for the Prover to interpret if a value should be made known to -> the Verifier, while the latter is a visibility modifier for the compiler to interpret if a -> function should be made accessible to external Noir programs like in other languages. - -### pub Modifier - -All data types in Noir are private by default. Types are explicitly declared as public using the -`pub` modifier: - -```rust -fn main(x : Field, y : pub Field) -> pub Field { - x + y -} -``` - -In this example, `x` is **private** while `y` and `x + y` (the return value) are **public**. Note -that visibility is handled **per variable**, so it is perfectly valid to have one input that is -private and another that is public. - -> **Note:** Public types can only be declared through parameters on `main`. - -## Type Aliases - -A type alias is a new name for an existing type. Type aliases are declared with the keyword `type`: - -```rust -type Id = u8; - -fn main() { - let id: Id = 1; - let zero: u8 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can also be used with [generics](../generics.md): - -```rust -type Id = Size; - -fn main() { - let id: Id = 1; - let zero: u32 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can even refer to other aliases. An error will be issued if they form a cycle: - -```rust -// Ok! -type A = B; -type B = Field; - -type Bad1 = Bad2; - -// error: Dependency cycle found -type Bad2 = Bad1; -// ^^^^^^^^^^^ 'Bad2' recursively depends on itself: Bad2 -> Bad1 -> Bad2 -``` - -By default, like functions, type aliases are private to the module they exist in. You can use `pub` -to make the type alias public or `pub(crate)` to make it public to just its crate: - -```rust -// This type alias is now public -pub type Id = u8; -``` - -## Wildcard Type -Noir can usually infer the type of the variable from the context, so specifying the type of a variable is only required when it cannot be inferred. However, specifying a complex type can be tedious, especially when it has multiple generic arguments. Often some of the generic types can be inferred from the context, and Noir only needs a hint to properly infer the other types. We can partially specify a variable's type by using `_` as a marker, indicating where we still want the compiler to infer the type. - -```rust -let a: [_; 4] = foo(b); -``` - - -### BigInt - -You can achieve BigInt functionality using the [Noir BigInt](https://github.com/shuklaayush/noir-bigint) library. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/integers.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/integers.md deleted file mode 100644 index a1d59bf3166..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/integers.md +++ /dev/null @@ -1,156 +0,0 @@ ---- -title: Integers -description: Explore the Integer data type in Noir. Learn about its methods, see real-world examples, and grasp how to efficiently use Integers in your Noir code. -keywords: [noir, integer types, methods, examples, arithmetic] -sidebar_position: 1 ---- - -An integer type is a range constrained field type. -The Noir frontend supports both unsigned and signed integer types. -The allowed sizes are 1, 8, 16, 32 and 64 bits. - -:::info - -When an integer is defined in Noir without a specific type, it will default to `Field`. - -The one exception is for loop indices which default to `u64` since comparisons on `Field`s are not possible. - -::: - -## Unsigned Integers - -An unsigned integer type is specified first with the letter `u` (indicating its unsigned nature) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: u8 = 1; - let y: u8 = 1; - let z = x + y; - assert (z == 2); -} -``` - -The bit size determines the maximum value the integer type can store. For example, a `u8` variable can store a value in the range of 0 to 255 (i.e. $\\2^{8}-1\\$). - -## Signed Integers - -A signed integer type is specified first with the letter `i` (which stands for integer) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: i8 = -1; - let y: i8 = -1; - let z = x + y; - assert (z == -2); -} -``` - -The bit size determines the maximum and minimum range of value the integer type can store. For example, an `i8` variable can store a value in the range of -128 to 127 (i.e. $\\-2^{7}\\$ to $\\2^{7}-1\\$). - -## 128 bits Unsigned Integers - -The built-in structure `U128` allows you to use 128-bit unsigned integers almost like a native integer type. However, there are some differences to keep in mind: -- You cannot cast between a native integer and `U128` -- There is a higher performance cost when using `U128`, compared to a native type. - -Conversion between unsigned integer types and U128 are done through the use of `from_integer` and `to_integer` functions. `from_integer` also accepts the `Field` type as input. - -```rust -fn main() { - let x = U128::from_integer(23); - let y = U128::from_hex("0x7"); - let z = x + y; - assert(z.to_integer() == 30); -} -``` - -`U128` is implemented with two 64 bits limbs, representing the low and high bits, which explains the performance cost. You should expect `U128` to be twice more costly for addition and four times more costly for multiplication. -You can construct a U128 from its limbs: -```rust -fn main(x: u64, y: u64) { - let x = U128::from_u64s_be(x,y); - assert(z.hi == x as Field); - assert(z.lo == y as Field); -} -``` - -Note that the limbs are stored as Field elements in order to avoid unnecessary conversions. -Apart from this, most operations will work as usual: - -```rust -fn main(x: U128, y: U128) { - // multiplication - let c = x * y; - // addition and subtraction - let c = c - x + y; - // division - let c = x / y; - // bit operation; - let c = x & y | y; - // bit shift - let c = x << y; - // comparisons; - let c = x < y; - let c = x == y; -} -``` - -## Overflows - -Computations that exceed the type boundaries will result in overflow errors. This happens with both signed and unsigned integers. For example, attempting to prove: - -```rust -fn main(x: u8, y: u8) { - let z = x + y; -} -``` - -With: - -```toml -x = "255" -y = "1" -``` - -Would result in: - -``` -$ nargo execute -error: Assertion failed: 'attempt to add with overflow' -┌─ ~/src/main.nr:9:13 -│ -│ let z = x + y; -│ ----- -│ -= Call stack: - ... -``` - -A similar error would happen with signed integers: - -```rust -fn main() { - let x: i8 = -118; - let y: i8 = -11; - let z = x + y; -} -``` - -### Wrapping methods - -Although integer overflow is expected to error, some use-cases rely on wrapping. For these use-cases, the standard library provides `wrapping` variants of certain common operations: - -```rust -fn wrapping_add(x: T, y: T) -> T; -fn wrapping_sub(x: T, y: T) -> T; -fn wrapping_mul(x: T, y: T) -> T; -``` - -Example of how it is used: - -```rust - -fn main(x: u8, y: u8) -> pub u8 { - std::wrapping_add(x, y) -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/references.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/references.md deleted file mode 100644 index a5293d11cfb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/references.md +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: References -sidebar_position: 9 ---- - -Noir supports first-class references. References are a bit like pointers: they point to a specific address that can be followed to access the data stored at that address. You can use Rust-like syntax to use pointers in Noir: the `&` operator references the variable, the `*` operator dereferences it. - -Example: - -```rust -fn main() { - let mut x = 2; - - // you can reference x as &mut and pass it to multiplyBy2 - multiplyBy2(&mut x); -} - -// you can access &mut here -fn multiplyBy2(x: &mut Field) { - // and dereference it with * - *x = *x * 2; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/slices.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/slices.mdx deleted file mode 100644 index cfee564a302..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/slices.mdx +++ /dev/null @@ -1,358 +0,0 @@ ---- -title: Slices -description: Explore the Slice data type in Noir. Understand its methods, see real-world examples, and learn how to effectively use Slices in your Noir programs. -keywords: [noir, slice type, methods, examples, subarrays] -sidebar_position: 5 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A slice is a dynamically-sized view into a sequence of elements. They can be resized at runtime, but because they don't own the data, they cannot be returned from a circuit. You can treat slices as arrays without a constrained size. - -```rust -fn main() -> pub u32 { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -To write a slice literal, use a preceding ampersand as in: `&[0; 2]` or -`&[1, 2, 3]`. - -It is important to note that slices are not references to arrays. In Noir, -`&[..]` is more similar to an immutable, growable vector. - -View the corresponding test file [here][test-file]. - -[test-file]: https://github.com/noir-lang/noir/blob/f387ec1475129732f72ba294877efdf6857135ac/crates/nargo_cli/tests/test_data_ssa_refactor/slices/src/main.nr - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for slices: - -### push_back - -Pushes a new element to the end of the slice, returning a new slice with a length one greater than the original unmodified slice. - -```rust -fn push_back(_self: [T], _elem: T) -> [T] -``` - -example: - -```rust -fn main() -> pub Field { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -View the corresponding test file [here][test-file]. - -### push_front - -Returns a new array with the specified element inserted at index 0. The existing elements indexes are incremented by 1. - -```rust -fn push_front(_self: Self, _elem: T) -> Self -``` - -Example: - -```rust -let mut new_slice: [Field] = &[]; -new_slice = new_slice.push_front(20); -assert(new_slice[0] == 20); // returns true -``` - -View the corresponding test file [here][test-file]. - -### pop_front - -Returns a tuple of two items, the first element of the array and the rest of the array. - -```rust -fn pop_front(_self: Self) -> (T, Self) -``` - -Example: - -```rust -let (first_elem, rest_of_slice) = slice.pop_front(); -``` - -View the corresponding test file [here][test-file]. - -### pop_back - -Returns a tuple of two items, the beginning of the array with the last element omitted and the last element. - -```rust -fn pop_back(_self: Self) -> (Self, T) -``` - -Example: - -```rust -let (popped_slice, last_elem) = slice.pop_back(); -``` - -View the corresponding test file [here][test-file]. - -### append - -Loops over a slice and adds it to the end of another. - -```rust -fn append(mut self, other: Self) -> Self -``` - -Example: - -```rust -let append = &[1, 2].append(&[3, 4, 5]); -``` - -### insert - -Inserts an element at a specified index and shifts all following elements by 1. - -```rust -fn insert(_self: Self, _index: Field, _elem: T) -> Self -``` - -Example: - -```rust -new_slice = rest_of_slice.insert(2, 100); -assert(new_slice[2] == 100); -``` - -View the corresponding test file [here][test-file]. - -### remove - -Remove an element at a specified index, shifting all elements after it to the left, returning the altered slice and the removed element. - -```rust -fn remove(_self: Self, _index: Field) -> (Self, T) -``` - -Example: - -```rust -let (remove_slice, removed_elem) = slice.remove(3); -``` - -### len - -Returns the length of a slice - -```rust -fn len(self) -> Field -``` - -Example: - -```rust -fn main() { - let slice = &[42, 42]; - assert(slice.len() == 2); -} -``` - -### as_array - -Converts this slice into an array. - -Make sure to specify the size of the resulting array. -Panics if the resulting array length is different than the slice's length. - -```rust -fn as_array(self) -> [T; N] -``` - -Example: - -```rust -fn main() { - let slice = &[5, 6]; - - // Always specify the length of the resulting array! - let array: [Field; 2] = slice.as_array(); - - assert(array[0] == slice[0]); - assert(array[1] == slice[1]); -} -``` - -### map - -Applies a function to each element of the slice, returning a new slice containing the mapped elements. - -```rust -fn map(self, f: fn[Env](T) -> U) -> [U] -``` - -example - -```rust -let a = &[1, 2, 3]; -let b = a.map(|a| a * 2); // b is now &[2, 4, 6] -``` - -### fold - -Applies a function to each element of the slice, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn[Env](U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the slice, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = &[1]; -let a2 = &[1, 2]; -let a3 = &[1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let folded = slice.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn[Env](T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let reduced = slice.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### filter - -Returns a new slice containing only elements for which the given predicate returns true. - -```rust -fn filter(self, f: fn[Env](T) -> bool) -> Self -``` - -example: - -```rust -fn main() { - let slice = &[1, 2, 3, 4, 5]; - let odds = slice.filter(|x| x % 2 == 1); - assert_eq(odds, &[1, 3, 5]); -} -``` - -### join - -Flatten each element in the slice into one value, separated by `separator`. - -Note that although slices implement `Append`, `join` cannot be used on slice -elements since nested slices are prohibited. - -```rust -fn join(self, separator: T) -> T where T: Append -``` - -example: - -```rust -struct Accumulator { - total: Field, -} - -// "Append" two accumulators by adding them -impl Append for Accumulator { - fn empty() -> Self { - Self { total: 0 } - } - - fn append(self, other: Self) -> Self { - Self { total: self.total + other.total } - } -} - -fn main() { - let slice = &[1, 2, 3, 4, 5].map(|total| Accumulator { total }); - - let result = slice.join(Accumulator::empty()); - assert_eq(result, Accumulator { total: 15 }); - - // We can use a non-empty separator to insert additional elements to sum: - let separator = Accumulator { total: 10 }; - let result = slice.join(separator); - assert_eq(result, Accumulator { total: 55 }); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let all = slice.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 5]; - let any = slice.any(|a| a == 5); - assert(any); -} - -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/strings.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/strings.md deleted file mode 100644 index 1fdee42425e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/strings.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Strings -description: - Discover the String data type in Noir. Learn about its methods, see real-world examples, and understand how to effectively manipulate and use Strings in Noir. -keywords: - [ - noir, - string type, - methods, - examples, - concatenation, - ] -sidebar_position: 3 ---- - - -The string type is a fixed length value defined with `str`. - -You can use strings in `assert()` functions or print them with -`println()`. See more about [Logging](../../standard_library/logging.md). - -```rust - -fn main(message : pub str<11>, hex_as_string : str<4>) { - println(message); - assert(message == "hello world"); - assert(hex_as_string == "0x41"); -} -``` - -You can convert a `str` to a byte array by calling `as_bytes()` -or a vector by calling `as_bytes_vec()`. - -```rust -fn main() { - let message = "hello world"; - let message_bytes = message.as_bytes(); - let mut message_vec = message.as_bytes_vec(); - assert(message_bytes.len() == 11); - assert(message_bytes[0] == 104); - assert(message_bytes[0] == message_vec.get(0)); -} -``` - -## Escape characters - -You can use escape characters for your strings: - -| Escape Sequence | Description | -|-----------------|-----------------| -| `\r` | Carriage Return | -| `\n` | Newline | -| `\t` | Tab | -| `\0` | Null Character | -| `\"` | Double Quote | -| `\\` | Backslash | - -Example: - -```rust -let s = "Hello \"world" // prints "Hello "world" -let s = "hey \tyou"; // prints "hey you" -``` - -## Raw strings - -A raw string begins with the letter `r` and is optionally delimited by a number of hashes `#`. - -Escape characters are *not* processed within raw strings. All contents are interpreted literally. - -Example: - -```rust -let s = r"Hello world"; -let s = r#"Simon says "hello world""#; - -// Any number of hashes may be used (>= 1) as long as the string also terminates with the same number of hashes -let s = r#####"One "#, Two "##, Three "###, Four "####, Five will end the string."#####; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/structs.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/structs.md deleted file mode 100644 index 29951ae843a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/structs.md +++ /dev/null @@ -1,96 +0,0 @@ ---- -title: Structs -description: - Explore the Struct data type in Noir. Learn about its methods, see real-world examples, and grasp how to effectively define and use Structs in your Noir programs. -keywords: - [ - noir, - struct type, - methods, - examples, - data structures, - ] -sidebar_position: 8 ---- - -A struct also allows for grouping multiple values of different types. Unlike tuples, we can also -name each field. - -> **Note:** The usage of _field_ here refers to each element of the struct and is unrelated to the -> field type of Noir. - -Defining a struct requires giving it a name and listing each field within as `: ` pairs: - -```rust -struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` - -An instance of a struct can then be created with actual values in `: ` pairs in any -order. Struct fields are accessible using their given names: - -```rust -fn main() { - let legs = 4; - - let dog = Animal { - eyes: 2, - hands: 0, - legs, - }; - - let zero = dog.hands; -} -``` - -Structs can also be destructured in a pattern, binding each field to a new variable: - -```rust -fn main() { - let Animal { hands, legs: feet, eyes } = get_octopus(); - - let ten = hands + feet + eyes as u8; -} - -fn get_octopus() -> Animal { - let octopus = Animal { - hands: 0, - legs: 8, - eyes: 2, - }; - - octopus -} -``` - -The new variables can be bound with names different from the original struct field names, as -showcased in the `legs --> feet` binding in the example above. - -### Visibility - -By default, like functions, structs are private to the module they exist in. You can use `pub` -to make the struct public or `pub(crate)` to make it public to just its crate: - -```rust -// This struct is now public -pub struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` - -The same applies to struct fields: by default they are private to the module they exist in, -but they can be made `pub` or `pub(crate)`: - -```rust -// This struct is now public -pub struct Animal { - hands: Field, // private to its module - pub(crate) legs: Field, // accessible from the entire crate - pub eyes: u8, // accessible from anywhere -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/tuples.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/tuples.md deleted file mode 100644 index 2ec5c9c4113..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/tuples.md +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Tuples -description: - Dive into the Tuple data type in Noir. Understand its methods, practical examples, and best practices for efficiently using Tuples in your Noir code. -keywords: - [ - noir, - tuple type, - methods, - examples, - multi-value containers, - ] -sidebar_position: 7 ---- - -A tuple collects multiple values like an array, but with the added ability to collect values of -different types: - -```rust -fn main() { - let tup: (u8, u64, Field) = (255, 500, 1000); -} -``` - -One way to access tuple elements is via destructuring using pattern matching: - -```rust -fn main() { - let tup = (1, 2); - - let (one, two) = tup; - - let three = one + two; -} -``` - -Another way to access tuple elements is via direct member access, using a period (`.`) followed by -the index of the element we want to access. Index `0` corresponds to the first tuple element, `1` to -the second and so on: - -```rust -fn main() { - let tup = (5, 6, 7, 8); - - let five = tup.0; - let eight = tup.3; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/functions.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/functions.md deleted file mode 100644 index f656cdfd97a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/functions.md +++ /dev/null @@ -1,226 +0,0 @@ ---- -title: Functions -description: - Learn how to declare functions and methods in Noir, a programming language with Rust semantics. - This guide covers parameter declaration, return types, call expressions, and more. -keywords: [Noir, Rust, functions, methods, parameter declaration, return types, call expressions] -sidebar_position: 1 ---- - -Functions in Noir follow the same semantics of Rust, though Noir does not support early returns. - -To declare a function the `fn` keyword is used. - -```rust -fn foo() {} -``` - -By default, functions are visible only within the package they are defined. To make them visible outside of that package (for example, as part of a [library](../modules_packages_crates/crates_and_packages.md#libraries)), you should mark them as `pub`: - -```rust -pub fn foo() {} -``` - -You can also restrict the visibility of the function to only the crate it was defined in, by specifying `pub(crate)`: - -```rust -pub(crate) fn foo() {} //foo can only be called within its crate -``` - -All parameters in a function must have a type and all types are known at compile time. The parameter -is pre-pended with a colon and the parameter type. Multiple parameters are separated using a comma. - -```rust -fn foo(x : Field, y : Field){} -``` - -The return type of a function can be stated by using the `->` arrow notation. The function below -states that the foo function must return a `Field`. If the function returns no value, then the arrow -is omitted. - -```rust -fn foo(x : Field, y : Field) -> Field { - x + y -} -``` - -Note that a `return` keyword is unneeded in this case - the last expression in a function's body is -returned. - -## Main function - -If you're writing a binary, the `main` function is the starting point of your program. You can pass all types of expressions to it, as long as they have a fixed size at compile time: - -```rust -fn main(x : Field) // this is fine: passing a Field -fn main(x : [Field; 2]) // this is also fine: passing a Field with known size at compile-time -fn main(x : (Field, bool)) // 👌: passing a (Field, bool) tuple means size 2 -fn main(x : str<5>) // this is fine, as long as you pass a string of size 5 - -fn main(x : Vec) // can't compile, has variable size -fn main(x : [Field]) // can't compile, has variable size -fn main(....// i think you got it by now -``` - -Keep in mind [tests](../../tooling/testing.md) don't differentiate between `main` and any other function. The following snippet passes tests, but won't compile or prove: - -```rust -fn main(x : [Field]) { - assert(x[0] == 1); -} - -#[test] -fn test_one() { - main(&[1, 2]); -} -``` - -```bash -$ nargo test -[testing] Running 1 test functions -[testing] Testing test_one... ok -[testing] All tests passed - -$ nargo check -The application panicked (crashed). -Message: Cannot have variable sized arrays as a parameter to main -``` - -## Call Expressions - -Calling a function in Noir is executed by using the function name and passing in the necessary -arguments. - -Below we show how to call the `foo` function from the `main` function using a call expression: - -```rust -fn main(x : Field, y : Field) { - let z = foo(x); -} - -fn foo(x : Field) -> Field { - x + x -} -``` - -## Methods - -You can define methods in Noir on any struct type in scope. - -```rust -struct MyStruct { - foo: Field, - bar: Field, -} - -impl MyStruct { - fn new(foo: Field) -> MyStruct { - MyStruct { - foo, - bar: 2, - } - } - - fn sum(self) -> Field { - self.foo + self.bar - } -} - -fn main() { - let s = MyStruct::new(40); - assert(s.sum() == 42); -} -``` - -Methods are just syntactic sugar for functions, so if we wanted to we could also call `sum` as -follows: - -```rust -assert(MyStruct::sum(s) == 42); -``` - -It is also possible to specialize which method is chosen depending on the [generic](./generics.md) type that is used. In this example, the `foo` function returns different values depending on its type: - -```rust -struct Foo {} - -impl Foo { - fn foo(self) -> Field { 1 } -} - -impl Foo { - fn foo(self) -> Field { 2 } -} - -fn main() { - let f1: Foo = Foo{}; - let f2: Foo = Foo{}; - assert(f1.foo() + f2.foo() == 3); -} -``` - -Also note that impls with the same method name defined in them cannot overlap. For example, if we already have `foo` defined for `Foo` and `Foo` like we do above, we cannot also define `foo` in an `impl Foo` since it would be ambiguous which version of `foo` to choose. - -```rust -// Including this impl in the same project as the above snippet would -// cause an overlapping impls error -impl Foo { - fn foo(self) -> Field { 3 } -} -``` - -## Lambdas - -Lambdas are anonymous functions. They follow the syntax of Rust - `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -See [Lambdas](./lambdas.md) for more details. - -## Attributes - -Attributes are metadata that can be applied to a function, using the following syntax: `#[attribute(value)]`. - -Supported attributes include: - -- **builtin**: the function is implemented by the compiler, for efficiency purposes. -- **deprecated**: mark the function as _deprecated_. Calling the function will generate a warning: `warning: use of deprecated function` -- **field**: Used to enable conditional compilation of code depending on the field size. See below for more details -- **oracle**: mark the function as _oracle_; meaning it is an external unconstrained function, implemented in noir_js. See [Unconstrained](./unconstrained.md) and [NoirJS](../../reference/NoirJS/noir_js/index.md) for more details. -- **test**: mark the function as unit tests. See [Tests](../../tooling/testing.md) for more details - -### Field Attribute - -The field attribute defines which field the function is compatible for. The function is conditionally compiled, under the condition that the field attribute matches the Noir native field. -The field can be defined implicitly, by using the name of the elliptic curve usually associated to it - for instance bn254, bls12_381 - or explicitly by using the field (prime) order, in decimal or hexadecimal form. -As a result, it is possible to define multiple versions of a function with each version specialized for a different field attribute. This can be useful when a function requires different parameters depending on the underlying elliptic curve. - -Example: we define the function `foo()` three times below. Once for the default Noir bn254 curve, once for the field $\mathbb F_{23}$, which will normally never be used by Noir, and once again for the bls12_381 curve. - -```rust -#[field(bn254)] -fn foo() -> u32 { - 1 -} - -#[field(23)] -fn foo() -> u32 { - 2 -} - -// This commented code would not compile as foo would be defined twice because it is the same field as bn254 -// #[field(21888242871839275222246405745257275088548364400416034343698204186575808495617)] -// fn foo() -> u32 { -// 2 -// } - -#[field(bls12_381)] -fn foo() -> u32 { - 3 -} -``` - -If the field name is not known to Noir, it will discard the function. Field names are case insensitive. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/generics.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/generics.md deleted file mode 100644 index c180a0ce7e6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/generics.md +++ /dev/null @@ -1,251 +0,0 @@ ---- -title: Generics -description: Learn how to use Generics in Noir -keywords: [Noir, Rust, generics, functions, structs] -sidebar_position: 7 ---- - -Generics allow you to use the same functions with multiple different concrete data types. You can -read more about the concept of generics in the Rust documentation -[here](https://doc.rust-lang.org/book/ch10-01-syntax.html). - -Here is a trivial example showing the identity function that supports any type. In Rust, it is -common to refer to the most general type as `T`. We follow the same convention in Noir. - -```rust -fn id(x: T) -> T { - x -} -``` - -## Numeric Generics - -If we want to be generic over array lengths (which are type-level integers), we can use numeric -generics. Using these looks similar to using regular generics, but introducing them into scope -requires declaring them with `let MyGenericName: IntegerType`. This can be done anywhere a normal -generic is declared. Instead of types, these generics resolve to integers at compile-time. -Here's an example of a struct that is generic over the size of the array it contains internally: - -```rust -struct BigInt { - limbs: [u32; N], -} - -impl BigInt { - // `N` is in scope of all methods in the impl - fn first(first: BigInt, second: BigInt) -> Self { - assert(first.limbs != second.limbs); - first - - fn second(first: BigInt, second: Self) -> Self { - assert(first.limbs != second.limbs); - second - } -} -``` - -## In Structs - -Generics are useful for specifying types in structs. For example, we can specify that a field in a -struct will be of a certain generic type. In this case `value` is of type `T`. - -```rust -struct RepeatedValue { - value: T, - count: Field, -} - -impl RepeatedValue { - fn print(self) { - for _i in 0 .. self.count { - println(self.value); - } - } -} - -fn main() { - let repeated = RepeatedValue { value: "Hello!", count: 2 }; - repeated.print(); -} -``` - -The `print` function will print `Hello!` an arbitrary number of times, twice in this case. - -## Calling functions on generic parameters - -Since a generic type `T` can represent any type, how can we call functions on the underlying type? -In other words, how can we go from "any type `T`" to "any type `T` that has certain methods available?" - -This is what [traits](../concepts/traits.md) are for in Noir. Here's an example of a function generic over -any type `T` that implements the `Eq` trait for equality: - -```rust -fn first_element_is_equal(array1: [T; N], array2: [T; N]) -> bool - where T: Eq -{ - if (array1.len() == 0) | (array2.len() == 0) { - true - } else { - array1[0] == array2[0] - } -} - -fn main() { - assert(first_element_is_equal([1, 2, 3], [1, 5, 6])); - - // We can use first_element_is_equal for arrays of any type - // as long as we have an Eq impl for the types we pass in - let array = [MyStruct::new(), MyStruct::new()]; - assert(array_eq(array, array, MyStruct::eq)); -} - -impl Eq for MyStruct { - fn eq(self, other: MyStruct) -> bool { - self.foo == other.foo - } -} -``` - -You can find more details on traits and trait implementations on the [traits page](../concepts/traits.md). - -## Manually Specifying Generics with the Turbofish Operator - -There are times when the compiler cannot reasonably infer what type should be used for a generic, or when the developer themselves may want to manually distinguish generic type parameters. This is where the `::<>` turbofish operator comes into play. - -The `::<>` operator can follow a variable or path and can be used to manually specify generic arguments within the angle brackets. -The name "turbofish" comes from that `::<>` looks like a little fish. - -Examples: -```rust -fn main() { - let mut slice = []; - slice = slice.push_back(1); - slice = slice.push_back(2); - // Without turbofish a type annotation would be needed on the left hand side - let array = slice.as_array::<2>(); -} -``` - - -```rust -trait MyTrait { - fn ten() -> Self; -} - -impl MyTrait for Field { - fn ten() -> Self { 10 } -} - -struct Foo { - inner: T -} - -impl Foo { - fn generic_method(_self: Self) -> U where U: MyTrait { - U::ten() - } -} - -fn example() { - let foo: Foo = Foo { inner: 1 }; - // Using a type other than `Field` here (e.g. u32) would fail as - // there is no matching impl for `u32: MyTrait`. - // - // Substituting the `10` on the left hand side of this assert - // with `10 as u32` would also fail with a type mismatch as we - // are expecting a `Field` from the right hand side. - assert(10 as u32 == foo.generic_method::()); -} -``` - -## Arithmetic Generics - -In addition to numeric generics, Noir also allows a limited form of arithmetic on generics. -When you have a numeric generic such as `N`, you can use the following operators on it in a -type position: `+`, `-`, `*`, `/`, and `%`. - -Note that type checking arithmetic generics is a best effort guess from the compiler and there -are many cases of types that are equal that the compiler may not see as such. For example, -we know that `T * (N + M)` should be equal to `T*N + T*M` but the compiler does not currently -apply the distributive law and thus sees these as different types. - -Even with this limitation though, the compiler can handle common cases decently well: - -```rust -trait Serialize { - fn serialize(self) -> [Field; N]; -} - -impl Serialize<1> for Field { - fn serialize(self) -> [Field; 1] { - [self] - } -} - -impl Serialize for [T; N] - where T: Serialize { .. } - -impl Serialize for (T, U) - where T: Serialize, U: Serialize { .. } - -fn main() { - let data = (1, [2, 3, 4]); - assert_eq(data.serialize().len(), 4); -} -``` - -Note that if there is any over or underflow the types will fail to unify: - -```rust title="underflow-example" showLineNumbers -fn pop(array: [Field; N]) -> [Field; N - 1] { - let mut result: [Field; N - 1] = std::mem::zeroed(); - for i in 0..N - 1 { - result[i] = array[i]; - } - result -} - -fn main() { - // error: Could not determine array length `(0 - 1)` - pop([]); -} -``` -> Source code: test_programs/compile_failure/arithmetic_generics_underflow/src/main.nr#L1-L14 - - -This also applies if there is underflow in an intermediate calculation: - -```rust title="intermediate-underflow-example" showLineNumbers -fn main() { - // From main it looks like there's nothing sketchy going on - seems_fine([]); -} - -// Since `seems_fine` says it can receive and return any length N -fn seems_fine(array: [Field; N]) -> [Field; N] { - // But inside `seems_fine` we pop from the array which - // requires the length to be greater than zero. - - // error: Could not determine array length `(0 - 1)` - push_zero(pop(array)) -} - -fn pop(array: [Field; N]) -> [Field; N - 1] { - let mut result: [Field; N - 1] = std::mem::zeroed(); - for i in 0..N - 1 { - result[i] = array[i]; - } - result -} - -fn push_zero(array: [Field; N]) -> [Field; N + 1] { - let mut result: [Field; N + 1] = std::mem::zeroed(); - for i in 0..N { - result[i] = array[i]; - } - // index N is already zeroed - result -} -``` -> Source code: test_programs/compile_failure/arithmetic_generics_intermediate_underflow/src/main.nr#L1-L32 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/globals.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/globals.md deleted file mode 100644 index 6b8314399a2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/globals.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: Global Variables -description: - Learn about global variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, globals, global variables, constants] -sidebar_position: 8 ---- - -## Globals - - -Noir supports global variables. The global's type can be inferred by the compiler entirely: - -```rust -global N = 5; // Same as `global N: Field = 5` - -global TUPLE = (3, 2); - -fn main() { - assert(N == 5); - assert(N == TUPLE.0 + TUPLE.1); -} -``` - -:::info - -Globals can be defined as any expression, so long as they don't depend on themselves - otherwise there would be a dependency cycle! For example: - -```rust -global T = foo(T); // dependency error -``` - -::: - - -If they are initialized to a literal integer, globals can be used to specify an array's length: - -```rust -global N: u32 = 2; - -fn main(y : [Field; N]) { - assert(y[0] == y[1]) -} -``` - -A global from another module can be imported or referenced externally like any other name: - -```rust -global N = 20; - -fn main() { - assert(my_submodule::N != N); -} - -mod my_submodule { - global N: Field = 10; -} -``` - -When a global is used, Noir replaces the name with its definition on each occurrence. -This means globals defined using function calls will repeat the call each time they're used: - -```rust -global RESULT = foo(); - -fn foo() -> [Field; 100] { ... } -``` - -This is usually fine since Noir will generally optimize any function call that does not -refer to a program input into a constant. It should be kept in mind however, if the called -function performs side-effects like `println`, as these will still occur on each use. - -### Visibility - -By default, like functions, globals are private to the module they exist in. You can use `pub` -to make the global public or `pub(crate)` to make it public to just its crate: - -```rust -// This global is now public -pub global N = 5; -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/lambdas.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/lambdas.md deleted file mode 100644 index be3c7e0b5ca..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/lambdas.md +++ /dev/null @@ -1,81 +0,0 @@ ---- -title: Lambdas -description: Learn how to use anonymous functions in Noir programming language. -keywords: [Noir programming language, lambda, closure, function, anonymous function] -sidebar_position: 9 ---- - -## Introduction - -Lambdas are anonymous functions. The syntax is `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -A block can be used as the body of a lambda, allowing you to declare local variables inside it: - -```rust -let cool = || { - let x = 100; - let y = 100; - x + y -} - -assert(cool() == 200); -``` - -## Closures - -Inside the body of a lambda, you can use variables defined in the enclosing function. Such lambdas are called **closures**. In this example `x` is defined inside `main` and is accessed from within the lambda: - -```rust -fn main() { - let x = 100; - let closure = || x + 150; - assert(closure() == 250); -} -``` - -## Passing closures to higher-order functions - -It may catch you by surprise that the following code fails to compile: - -```rust -fn foo(f: fn () -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // error :( -} -``` - -The reason is that the closure's capture environment affects its type - we have a closure that captures two Fields and `foo` -expects a regular function as an argument - those are incompatible. -:::note - -Variables contained within the `||` are the closure's parameters, and the expression that follows it is the closure's body. The capture environment is comprised of any variables used in the closure's body that are not parameters. - -E.g. in |x| x + y, y would be a captured variable, but x would not be, since it is a parameter of the closure. - -::: -The syntax for the type of a closure is `fn[env](args) -> ret_type`, where `env` is the capture environment of the closure - -in this example that's `(Field, Field)`. - -The best solution in our case is to make `foo` generic over the environment type of its parameter, so that it can be called -with closures with any environment, as well as with regular functions: - -```rust -fn foo(f: fn[Env]() -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // compiles fine - assert(foo(|| 60) == 60); // compiles fine -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/mutability.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/mutability.md deleted file mode 100644 index fdeef6a87c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/mutability.md +++ /dev/null @@ -1,121 +0,0 @@ ---- -title: Mutability -description: - Learn about mutable variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, mutability in noir, mutable variables] -sidebar_position: 8 ---- - -Variables in noir can be declared mutable via the `mut` keyword. Mutable variables can be reassigned -to via an assignment expression. - -```rust -let x = 2; -x = 3; // error: x must be mutable to be assigned to - -let mut y = 3; -let y = 4; // OK -``` - -The `mut` modifier can also apply to patterns: - -```rust -let (a, mut b) = (1, 2); -a = 11; // error: a must be mutable to be assigned to -b = 12; // OK - -let mut (c, d) = (3, 4); -c = 13; // OK -d = 14; // OK - -// etc. -let MyStruct { x: mut y } = MyStruct { x: a }; -// y is now in scope -``` - -Note that mutability in noir is local and everything is passed by value, so if a called function -mutates its parameters then the parent function will keep the old value of the parameters. - -```rust -fn main() -> pub Field { - let x = 3; - helper(x); - x // x is still 3 -} - -fn helper(mut x: i32) { - x = 4; -} -``` - -## Non-local mutability - -Non-local mutability can be achieved through the mutable reference type `&mut T`: - -```rust -fn set_to_zero(x: &mut Field) { - *x = 0; -} - -fn main() { - let mut y = 42; - set_to_zero(&mut y); - assert(*y == 0); -} -``` - -When creating a mutable reference, the original variable being referred to (`y` in this -example) must also be mutable. Since mutable references are a reference type, they must -be explicitly dereferenced via `*` to retrieve the underlying value. Note that this yields -a copy of the value, so mutating this copy will not change the original value behind the -reference: - -```rust -fn main() { - let mut x = 1; - let x_ref = &mut x; - - let mut y = *x_ref; - let y_ref = &mut y; - - x = 2; - *x_ref = 3; - - y = 4; - *y_ref = 5; - - assert(x == 3); - assert(*x_ref == 3); - assert(y == 5); - assert(*y_ref == 5); -} -``` - -Note that types in Noir are actually deeply immutable so the copy that occurs when -dereferencing is only a conceptual copy - no additional constraints will occur. - -Mutable references can also be stored within structs. Note that there is also -no lifetime parameter on these unlike rust. This is because the allocated memory -always lasts the entire program - as if it were an array of one element. - -```rust -struct Foo { - x: &mut Field -} - -impl Foo { - fn incr(mut self) { - *self.x += 1; - } -} - -fn main() { - let foo = Foo { x: &mut 0 }; - foo.incr(); - assert(*foo.x == 1); -} -``` - -In general, you should avoid non-local & shared mutability unless it is needed. Sticking -to only local mutability will improve readability and potentially improve compiler optimizations as well. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/ops.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/ops.md deleted file mode 100644 index c35c36c38a9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/ops.md +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: Logical Operations -description: - Learn about the supported arithmetic and logical operations in the Noir programming language. - Discover how to perform operations on private input types, integers, and booleans. -keywords: - [ - Noir programming language, - supported operations, - arithmetic operations, - logical operations, - predicate operators, - bitwise operations, - short-circuiting, - backend, - ] -sidebar_position: 3 ---- - -# Operations - -## Table of Supported Operations - -| Operation | Description | Requirements | -| :-------- | :------------------------------------------------------------: | -------------------------------------: | -| + | Adds two private input types together | Types must be private input | -| - | Subtracts two private input types together | Types must be private input | -| \* | Multiplies two private input types together | Types must be private input | -| / | Divides two private input types together | Types must be private input | -| ^ | XOR two private input types together | Types must be integer | -| & | AND two private input types together | Types must be integer | -| \| | OR two private input types together | Types must be integer | -| \<\< | Left shift an integer by another integer amount | Types must be integer, shift must be u8 | -| >> | Right shift an integer by another integer amount | Types must be integer, shift must be u8 | -| ! | Bitwise not of a value | Type must be integer or boolean | -| \< | returns a bool if one value is less than the other | Upper bound must have a known bit size | -| \<= | returns a bool if one value is less than or equal to the other | Upper bound must have a known bit size | -| > | returns a bool if one value is more than the other | Upper bound must have a known bit size | -| >= | returns a bool if one value is more than or equal to the other | Upper bound must have a known bit size | -| == | returns a bool if one value is equal to the other | Both types must not be constants | -| != | returns a bool if one value is not equal to the other | Both types must not be constants | - -### Predicate Operators - -`<,<=, !=, == , >, >=` are known as predicate/comparison operations because they compare two values. -This differs from the operations such as `+` where the operands are used in _computation_. - -### Bitwise Operations Example - -```rust -fn main(x : Field) { - let y = x as u32; - let z = y & y; -} -``` - -`z` is implicitly constrained to be the result of `y & y`. The `&` operand is used to denote bitwise -`&`. - -> `x & x` would not compile as `x` is a `Field` and not an integer type. - -### Logical Operators - -Noir has no support for the logical operators `||` and `&&`. This is because encoding the -short-circuiting that these operators require can be inefficient for Noir's backend. Instead you can -use the bitwise operators `|` and `&` which operate identically for booleans, just without the -short-circuiting. - -```rust -let my_val = 5; - -let mut flag = 1; -if (my_val > 6) | (my_val == 0) { - flag = 0; -} -assert(flag == 1); - -if (my_val != 10) & (my_val < 50) { - flag = 0; -} -assert(flag == 0); -``` - -### Shorthand operators - -Noir shorthand operators for most of the above operators, namely `+=, -=, *=, /=, %=, &=, |=, ^=, <<=`, and `>>=`. These allow for more concise syntax. For example: - -```rust -let mut i = 0; -i = i + 1; -``` - -could be written as: - -```rust -let mut i = 0; -i += 1; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/oracles.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/oracles.mdx deleted file mode 100644 index 77a2ac1550a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/oracles.mdx +++ /dev/null @@ -1,29 +0,0 @@ ---- -title: Oracles -description: Dive into how Noir supports Oracles via RPC calls, and learn how to declare an Oracle in Noir with our comprehensive guide. -keywords: - - Noir - - Oracles - - RPC Calls - - Unconstrained Functions - - Programming - - Blockchain -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -Noir has support for Oracles via RPC calls. This means Noir will make an RPC call and use the return value for proof generation. - -Since Oracles are not resolved by Noir, they are [`unconstrained` functions](./unconstrained.md) - -You can declare an Oracle through the `#[oracle()]` flag. Example: - -```rust -#[oracle(get_number_sequence)] -unconstrained fn get_number_sequence(_size: Field) -> [Field] {} -``` - -The timeout for when using an external RPC oracle resolver can be set with the `NARGO_FOREIGN_CALL_TIMEOUT` environment variable. This timeout is in units of milliseconds. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/shadowing.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/shadowing.md deleted file mode 100644 index 5ce6130d201..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/shadowing.md +++ /dev/null @@ -1,44 +0,0 @@ ---- -title: Shadowing -sidebar_position: 12 ---- - -Noir allows for inheriting variables' values and re-declaring them with the same name similar to Rust, known as shadowing. - -For example, the following function is valid in Noir: - -```rust -fn main() { - let x = 5; - - { - let x = x * 2; - assert (x == 10); - } - - assert (x == 5); -} -``` - -In this example, a variable x is first defined with the value 5. - -The local scope that follows shadows the original x, i.e. creates a local mutable x based on the value of the original x. It is given a value of 2 times the original x. - -When we return to the main scope, x once again refers to just the original x, which stays at the value of 5. - -## Temporal mutability - -One way that shadowing is useful, in addition to ergonomics across scopes, is for temporarily mutating variables. - -```rust -fn main() { - let age = 30; - // age = age + 5; // Would error as `age` is immutable by default. - - let mut age = age + 5; // Temporarily mutates `age` with a new value. - - let age = age; // Locks `age`'s mutability again. - - assert (age == 35); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/traits.md deleted file mode 100644 index 9da00a77587..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/traits.md +++ /dev/null @@ -1,501 +0,0 @@ ---- -title: Traits -description: - Traits in Noir can be used to abstract out a common interface for functions across - several data types. -keywords: [noir programming language, traits, interfaces, generic, protocol] -sidebar_position: 14 ---- - -## Overview - -Traits in Noir are a useful abstraction similar to interfaces or protocols in other languages. Each trait defines -the interface of several methods contained within the trait. Types can then implement this trait by providing -implementations for these methods. For example in the program: - -```rust -struct Rectangle { - width: Field, - height: Field, -} - -impl Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -fn log_area(r: Rectangle) { - println(r.area()); -} -``` - -We have a function `log_area` to log the area of a `Rectangle`. Now how should we change the program if we want this -function to work on `Triangle`s as well?: - -```rust -struct Triangle { - width: Field, - height: Field, -} - -impl Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Making `log_area` generic over all types `T` would be invalid since not all types have an `area` method. Instead, we can -introduce a new `Area` trait and make `log_area` generic over all types `T` that implement `Area`: - -```rust -trait Area { - fn area(self) -> Field; -} - -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -We also need to explicitly implement `Area` for `Rectangle` and `Triangle`. We can do that by changing their existing -impls slightly. Note that the parameter types and return type of each of our `area` methods must match those defined -by the `Area` trait. - -```rust -impl Area for Rectangle { - fn area(self) -> Field { - self.width * self.height - } -} - -impl Area for Triangle { - fn area(self) -> Field { - self.width * self.height / 2 - } -} -``` - -Now we have a working program that is generic over any type of Shape that is used! Others can even use this program -as a library with their own types - such as `Circle` - as long as they also implement `Area` for these types. - -## Where Clauses - -As seen in `log_area` above, when we want to create a function or method that is generic over any type that implements -a trait, we can add a where clause to the generic function. - -```rust -fn log_area(shape: T) where T: Area { - println(shape.area()); -} -``` - -It is also possible to apply multiple trait constraints on the same variable at once by combining traits with the `+` -operator. Similarly, we can have multiple trait constraints by separating each with a comma: - -```rust -fn foo(elements: [T], thing: U) where - T: Default + Add + Eq, - U: Bar, -{ - let mut sum = T::default(); - - for element in elements { - sum += element; - } - - if sum == T::default() { - thing.bar(); - } -} -``` - -## Generic Implementations - -You can add generics to a trait implementation by adding the generic list after the `impl` keyword: - -```rust -trait Second { - fn second(self) -> Field; -} - -impl Second for (T, Field) { - fn second(self) -> Field { - self.1 - } -} -``` - -You can also implement a trait for every type this way: - -```rust -trait Debug { - fn debug(self); -} - -impl Debug for T { - fn debug(self) { - println(self); - } -} - -fn main() { - 1.debug(); -} -``` - -### Generic Trait Implementations With Where Clauses - -Where clauses can be placed on trait implementations themselves to restrict generics in a similar way. -For example, while `impl Foo for T` implements the trait `Foo` for every type, `impl Foo for T where T: Bar` -will implement `Foo` only for types that also implement `Bar`. This is often used for implementing generic types. -For example, here is the implementation for array equality: - -```rust -impl Eq for [T; let N: u32] where T: Eq { - // Test if two arrays have the same elements. - // Because both arrays must have length N, we know their lengths already match. - fn eq(self, other: Self) -> bool { - let mut result = true; - - for i in 0 .. self.len() { - // The T: Eq constraint is needed to call == on the array elements here - result &= self[i] == other[i]; - } - - result - } -} -``` - -Where clauses can also be placed on struct implementations. -For example, here is a method utilizing a generic type that implements the equality trait. - -```rust -struct Foo { - a: u32, - b: T, -} - -impl Foo where T: Eq { - fn eq(self, other: Self) -> bool { - (self.a == other.a) & self.b.eq(other.b) - } -} -``` - -## Generic Traits - -Traits themselves can also be generic by placing the generic arguments after the trait name. These generics are in -scope of every item within the trait. - -```rust -trait Into { - // Convert `self` to type `T` - fn into(self) -> T; -} -``` - -When implementing generic traits the generic arguments of the trait must be specified. This is also true anytime -when referencing a generic trait (e.g. in a `where` clause). - -```rust -struct MyStruct { - array: [Field; 2], -} - -impl Into<[Field; 2]> for MyStruct { - fn into(self) -> [Field; 2] { - self.array - } -} - -fn as_array(x: T) -> [Field; 2] - where T: Into<[Field; 2]> -{ - x.into() -} - -fn main() { - let array = [1, 2]; - let my_struct = MyStruct { array }; - - assert_eq(as_array(my_struct), array); -} -``` - -### Associated Types and Constants - -Traits also support associated types and constraints which can be thought of as additional generics that are referred to by name. - -Here's an example of a trait with an associated type `Foo` and a constant `Bar`: - -```rust -trait MyTrait { - type Foo; - - let Bar: u32; -} -``` - -Now when we're implementing `MyTrait` we also have to provide values for `Foo` and `Bar`: - -```rust -impl MyTrait for Field { - type Foo = i32; - - let Bar: u32 = 11; -} -``` - -Since associated constants can also be used in a type position, its values are limited to only other -expression kinds allowed in numeric generics. - -Note that currently all associated types and constants must be explicitly specified in a trait constraint. -If we leave out any, we'll get an error that we're missing one: - -```rust -// Error! Constraint is missing associated constant for `Bar` -fn foo(x: T) where T: MyTrait { - ... -} -``` - -Because all associated types and constants must be explicitly specified, they are essentially named generics, -although this is set to change in the future. Future versions of Noir will allow users to elide associated types -in trait constraints similar to Rust. When this is done, you may still refer to their value with the `::AssociatedType` -syntax: - -```rust -// Only valid in future versions of Noir: -fn foo(x: T) where T: MyTrait { - let _: ::Foo = ...; -} -``` - -The type as trait syntax is possible in Noir today but is less useful when each type must be explicitly specified anyway: - -```rust -fn foo(x: T) where T: MyTrait { - // Works, but could just use F directly - let _: >::Foo = ...; - - let _: F = ...; -} -``` - -## Trait Methods With No `self` - -A trait can contain any number of methods, each of which have access to the `Self` type which represents each type -that eventually implements the trait. Similarly, the `self` variable is available as well but is not required to be used. -For example, we can define a trait to create a default value for a type. This trait will need to return the `Self` type -but doesn't need to take any parameters: - -```rust -trait Default { - fn default() -> Self; -} -``` - -Implementing this trait can be done similarly to any other trait: - -```rust -impl Default for Field { - fn default() -> Field { - 0 - } -} - -struct MyType {} - -impl Default for MyType { - fn default() -> Field { - MyType {} - } -} -``` - -However, since there is no `self` parameter, we cannot call it via the method call syntax `object.method()`. -Instead, we'll need to refer to the function directly. This can be done either by referring to the -specific impl `MyType::default()` or referring to the trait itself `Default::default()`. In the later -case, type inference determines the impl that is selected. - -```rust -let my_struct = MyStruct::default(); - -let x: Field = Default::default(); -let result = x + Default::default(); -``` - -:::warning - -```rust -let _ = Default::default(); -``` - -If type inference cannot select which impl to use because of an ambiguous `Self` type, an impl will be -arbitrarily selected. This occurs most often when the result of a trait function call with no parameters -is unused. To avoid this, when calling a trait function with no `self` or `Self` parameters or return type, -always refer to it via the implementation type's namespace - e.g. `MyType::default()`. -This is set to change to an error in future Noir versions. - -::: - -## Default Method Implementations - -A trait can also have default implementations of its methods by giving a body to the desired functions. -Note that this body must be valid for all types that may implement the trait. As a result, the only -valid operations on `self` will be operations valid for any type or other operations on the trait itself. - -```rust -trait Numeric { - fn add(self, other: Self) -> Self; - - // Default implementation of double is (self + self) - fn double(self) -> Self { - self.add(self) - } -} -``` - -When implementing a trait with default functions, a type may choose to implement only the required functions: - -```rust -impl Numeric for Field { - fn add(self, other: Field) -> Field { - self + other - } -} -``` - -Or it may implement the optional methods as well: - -```rust -impl Numeric for u32 { - fn add(self, other: u32) -> u32 { - self + other - } - - fn double(self) -> u32 { - self * 2 - } -} -``` - -## Impl Specialization - -When implementing traits for a generic type it is possible to implement the trait for only a certain combination -of generics. This can be either as an optimization or because those specific generics are required to implement the trait. - -```rust -trait Sub { - fn sub(self, other: Self) -> Self; -} - -struct NonZero { - value: T, -} - -impl Sub for NonZero { - fn sub(self, other: Self) -> Self { - let value = self.value - other.value; - assert(value != 0); - NonZero { value } - } -} -``` - -## Overlapping Implementations - -Overlapping implementations are disallowed by Noir to ensure Noir's decision on which impl to select is never ambiguous. -This means if a trait `Foo` is already implemented -by a type `Bar` for all `T`, then we cannot also have a separate impl for `Bar` (or any other -type argument). Similarly, if there is an impl for all `T` such as `impl Debug for T`, we cannot create -any more impls to `Debug` for other types since it would be ambiguous which impl to choose for any given -method call. - -```rust -trait Trait {} - -// Previous impl defined here -impl Trait for (A, B) {} - -// error: Impl for type `(Field, Field)` overlaps with existing impl -impl Trait for (Field, Field) {} -``` - -## Trait Coherence - -Another restriction on trait implementations is coherence. This restriction ensures other crates cannot create -impls that may overlap with other impls, even if several unrelated crates are used as dependencies in the same -program. - -The coherence restriction is: to implement a trait, either the trait itself or the object type must be declared -in the crate the impl is in. - -In practice this often comes up when using types provided by libraries. If a library provides a type `Foo` that does -not implement a trait in the standard library such as `Default`, you may not `impl Default for Foo` in your own crate. -While restrictive, this prevents later issues or silent changes in the program if the `Foo` library later added its -own impl for `Default`. If you are a user of the `Foo` library in this scenario and need a trait not implemented by the -library your choices are to either submit a patch to the library or use the newtype pattern. - -### The Newtype Pattern - -The newtype pattern gets around the coherence restriction by creating a new wrapper type around the library type -that we cannot create `impl`s for. Since the new wrapper type is defined in our current crate, we can create -impls for any trait we need on it. - -```rust -struct Wrapper { - foo: some_library::Foo, -} - -impl Default for Wrapper { - fn default() -> Wrapper { - Wrapper { - foo: some_library::Foo::new(), - } - } -} -``` - -Since we have an impl for our own type, the behavior of this code will not change even if `some_library` is updated -to provide its own `impl Default for Foo`. The downside of this pattern is that it requires extra wrapping and -unwrapping of values when converting to and from the `Wrapper` and `Foo` types. - -### Trait Inheritance - -Sometimes, you might need one trait to use another trait’s functionality (like "inheritance" in some other languages). In this case, you can specify this relationship by listing any child traits after the parent trait's name and a colon. Now, whenever the parent trait is implemented it will require the child traits to be implemented as well. A parent trait is also called a "super trait." - -```rust -trait Person { - fn name(self) -> String; -} - -// Person is a supertrait of Student. -// Implementing Student requires you to also impl Person. -trait Student: Person { - fn university(self) -> String; -} - -trait Programmer { - fn fav_language(self) -> String; -} - -// CompSciStudent (computer science student) is a subtrait of both Programmer -// and Student. Implementing CompSciStudent requires you to impl both supertraits. -trait CompSciStudent: Programmer + Student { - fn git_username(self) -> String; -} -``` - -### Visibility - -By default, like functions, traits are private to the module they exist in. You can use `pub` -to make the trait public or `pub(crate)` to make it public to just its crate: - -```rust -// This trait is now public -pub trait Trait {} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/unconstrained.md deleted file mode 100644 index b5221b8d2dd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/unconstrained.md +++ /dev/null @@ -1,104 +0,0 @@ ---- -title: Unconstrained Functions -description: "Learn about what unconstrained functions in Noir are, how to use them and when you'd want to." - -keywords: [Noir programming language, unconstrained, open] -sidebar_position: 5 ---- - -Unconstrained functions are functions which do not constrain any of the included computation and allow for non-deterministic computation. - -## Why? - -Zero-knowledge (ZK) domain-specific languages (DSL) enable developers to generate ZK proofs from their programs by compiling code down to the constraints of an NP complete language (such as R1CS or PLONKish languages). However, the hard bounds of a constraint system can be very limiting to the functionality of a ZK DSL. - -Enabling a circuit language to perform unconstrained execution is a powerful tool. Said another way, unconstrained execution lets developers generate witnesses from code that does not generate any constraints. Being able to execute logic outside of a circuit is critical for both circuit performance and constructing proofs on information that is external to a circuit. - -Fetching information from somewhere external to a circuit can also be used to enable developers to improve circuit efficiency. - -A ZK DSL does not just prove computation, but proves that some computation was handled correctly. Thus, it is necessary that when we switch from performing some operation directly inside of a circuit to inside of an unconstrained environment that the appropriate constraints are still laid down elsewhere in the circuit. - -## Example - -An in depth example might help drive the point home. This example comes from the excellent [post](https://discord.com/channels/1113924620781883405/1124022445054111926/1128747641853972590) by Tom in the Noir Discord. - -Let's look at how we can optimize a function to turn a `u72` into an array of `u8`s. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u72 & 0xff) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 91 -Backend circuit size: 3619 -``` - -A lot of the operations in this function are optimized away by the compiler (all the bit-shifts turn into divisions by constants). However we can save a bunch of gates by casting to u8 a bit earlier. This automatically truncates the bit-shifted value to fit in a u8 which allows us to remove the AND against 0xff. This saves us ~480 gates in total. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 75 -Backend circuit size: 3143 -``` - -Those are some nice savings already but we can do better. This code is all constrained so we're proving every step of calculating out using num, but we don't actually care about how we calculate this, just that it's correct. This is where brillig comes in. - -It turns out that truncating a u72 into a u8 is hard to do inside a snark, each time we do as u8 we lay down 4 ACIR opcodes which get converted into multiple gates. It's actually much easier to calculate num from out than the other way around. All we need to do is multiply each element of out by a constant and add them all together, both relatively easy operations inside a snark. - -We can then run `u72_to_u8` as unconstrained brillig code in order to calculate out, then use that result in our constrained function and assert that if we were to do the reverse calculation we'd get back num. This looks a little like the below: - -```rust -fn main(num: u72) -> pub [u8; 8] { - let out = unsafe { - u72_to_u8(num) - }; - - let mut reconstructed_num: u72 = 0; - for i in 0..8 { - reconstructed_num += (out[i] as u72 << (56 - (8 * i))); - } - assert(num == reconstructed_num); - out -} - -unconstrained fn u72_to_u8(num: u72) -> [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8))) as u8; - } - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 78 -Backend circuit size: 2902 -``` - -This ends up taking off another ~250 gates from our circuit! We've ended up with more ACIR opcodes than before but they're easier for the backend to prove (resulting in fewer gates). - -Note that in order to invoke unconstrained functions we need to wrap them in an `unsafe` block, -to make it clear that the call is unconstrained. - -Generally we want to use brillig whenever there's something that's easy to verify but hard to compute within the circuit. For example, if you wanted to calculate a square root of a number it'll be a much better idea to calculate this in brillig and then assert that if you square the result you get back your number. - -## Break and Continue - -In addition to loops over runtime bounds, `break` and `continue` are also available in unconstrained code. See [break and continue](../concepts/control_flow.md#break-and-continue) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/_category_.json deleted file mode 100644 index 1debcfe7675..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Modules, Packages and Crates", - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/crates_and_packages.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/crates_and_packages.md deleted file mode 100644 index 95ee9f52ab2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/crates_and_packages.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Crates and Packages -description: Learn how to use Crates and Packages in your Noir project -keywords: [Nargo, dependencies, package management, crates, package] -sidebar_position: 0 ---- - -## Crates - -A crate is the smallest amount of code that the Noir compiler considers at a time. -Crates can contain modules, and the modules may be defined in other files that get compiled with the crate, as we’ll see in the coming sections. - -### Crate Types - -A Noir crate can come in several forms: binaries, libraries or contracts. - -#### Binaries - -_Binary crates_ are programs which you can compile to an ACIR circuit which you can then create proofs against. Each must have a function called `main` that defines the ACIR circuit which is to be proved. - -#### Libraries - -_Library crates_ don't have a `main` function and they don't compile down to ACIR. Instead they define functionality intended to be shared with multiple projects, and eventually included in a binary crate. - -#### Contracts - -Contract crates are similar to binary crates in that they compile to ACIR which you can create proofs against. They are different in that they do not have a single `main` function, but are a collection of functions to be deployed to the [Aztec network](https://aztec.network). You can learn more about the technical details of Aztec in the [monorepo](https://github.com/AztecProtocol/aztec-packages) or contract [examples](https://github.com/AztecProtocol/aztec-packages/tree/master/noir-projects/noir-contracts/contracts). - -### Crate Root - -Every crate has a root, which is the source file that the compiler starts, this is also known as the root module. The Noir compiler does not enforce any conditions on the name of the file which is the crate root, however if you are compiling via Nargo the crate root must be called `lib.nr` or `main.nr` for library or binary crates respectively. - -## Packages - -A Nargo _package_ is a collection of one of more crates that provides a set of functionality. A package must include a Nargo.toml file. - -A package _must_ contain either a library or a binary crate, but not both. - -### Differences from Cargo Packages - -One notable difference between Rust's Cargo and Noir's Nargo is that while Cargo allows a package to contain an unlimited number of binary crates and a single library crate, Nargo currently only allows a package to contain a single crate. - -In future this restriction may be lifted to allow a Nargo package to contain both a binary and library crate or multiple binary crates. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/dependencies.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/dependencies.md deleted file mode 100644 index 24e02de08fe..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/dependencies.md +++ /dev/null @@ -1,124 +0,0 @@ ---- -title: Dependencies -description: - Learn how to specify and manage dependencies in Nargo, allowing you to upload packages to GitHub - and use them easily in your project. -keywords: [Nargo, dependencies, GitHub, package management, versioning] -sidebar_position: 1 ---- - -Nargo allows you to upload packages to GitHub and use them as dependencies. - -## Specifying a dependency - -Specifying a dependency requires a tag to a specific commit and the git url to the url containing -the package. - -Currently, there are no requirements on the tag contents. If requirements are added, it would follow -semver 2.0 guidelines. - -> Note: Without a `tag` , there would be no versioning and dependencies would change each time you -> compile your project. - -For example, to add the [ecrecover-noir library](https://github.com/colinnielsen/ecrecover-noir) to your project, add it to `Nargo.toml`: - -```toml -# Nargo.toml - -[dependencies] -ecrecover = {tag = "v0.8.0", git = "https://github.com/colinnielsen/ecrecover-noir"} -``` - -If the module is in a subdirectory, you can define a subdirectory in your git repository, for example: - -```toml -# Nargo.toml - -[dependencies] -easy_private_token_contract = {tag ="v0.1.0-alpha62", git = "https://github.com/AztecProtocol/aztec-packages", directory = "noir-contracts/contracts/easy_private_token_contract"} -``` - -## Specifying a local dependency - -You can also specify dependencies that are local to your machine. - -For example, this file structure has a library and binary crate - -```tree -├── binary_crate -│   ├── Nargo.toml -│   └── src -│   └── main.nr -└── lib_a - ├── Nargo.toml - └── src - └── lib.nr -``` - -Inside of the binary crate, you can specify: - -```toml -# Nargo.toml - -[dependencies] -lib_a = { path = "../lib_a" } -``` - -## Importing dependencies - -You can import a dependency to a Noir file using the following syntax. For example, to import the -ecrecover-noir library and local lib_a referenced above: - -```rust -use ecrecover; -use lib_a; -``` - -You can also import only the specific parts of dependency that you want to use, like so: - -```rust -use std::hash::sha256; -use std::scalar_mul::fixed_base_embedded_curve; -``` - -Lastly, as demonstrated in the -[elliptic curve example](../standard_library/cryptographic_primitives/ec_primitives.md#examples), you -can import multiple items in the same line by enclosing them in curly braces: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; -``` - -We don't have a way to consume libraries from inside a [workspace](./workspaces.md) as external dependencies right now. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -## Dependencies of Dependencies - -Note that when you import a dependency, you also get access to all of the dependencies of that package. - -For example, the [phy_vector](https://github.com/resurgencelabs/phy_vector) library imports an [fraction](https://github.com/resurgencelabs/fraction) library. If you're importing the phy_vector library, then you can access the functions in fractions library like so: - -```rust -use phy_vector; - -fn main(x : Field, y : pub Field) { - //... - let f = phy_vector::fraction::toFraction(true, 2, 1); - //... -} -``` - -## Available Libraries - -Noir does not currently have an official package manager. You can find a list of available Noir libraries in the [awesome-noir repo here](https://github.com/noir-lang/awesome-noir#libraries). - -Some libraries that are available today include: - -- [Standard Library](https://github.com/noir-lang/noir/tree/master/noir_stdlib) - the Noir Standard Library -- [Ethereum Storage Proof Verification](https://github.com/aragonzkresearch/noir-trie-proofs) - a library that contains the primitives necessary for RLP decoding (in the form of look-up table construction) and Ethereum state and storage proof verification (or verification of any trie proof involving 32-byte long keys) -- [BigInt](https://github.com/shuklaayush/noir-bigint) - a library that provides a custom BigUint56 data type, allowing for computations on large unsigned integers -- [ECrecover](https://github.com/colinnielsen/ecrecover-noir/tree/main) - a library to verify an ECDSA signature and return the source Ethereum address -- [Sparse Merkle Tree Verifier](https://github.com/vocdoni/smtverifier-noir/tree/main) - a library for verification of sparse Merkle trees -- [Signed Int](https://github.com/resurgencelabs/signed_int) - a library for accessing a custom Signed Integer data type, allowing access to negative numbers on Noir -- [Fraction](https://github.com/resurgencelabs/fraction) - a library for accessing fractional number data type in Noir, allowing results that aren't whole numbers diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/modules.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/modules.md deleted file mode 100644 index 14aa1f0579a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/modules.md +++ /dev/null @@ -1,221 +0,0 @@ ---- -title: Modules -description: - Learn how to organize your files using modules in Noir, following the same convention as Rust's - module system. Examples included. -keywords: [Noir, Rust, modules, organizing files, sub-modules] -sidebar_position: 2 ---- - -Noir's module system follows the same convention as the _newer_ version of Rust's module system. - -## Purpose of Modules - -Modules are used to organize files. Without modules all of your code would need to live in a single -file. In Noir, the compiler does not automatically scan all of your files to detect modules. This -must be done explicitly by the developer. - -## Examples - -### Importing a module in the crate root - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo.nr` - -```rust -fn from_foo() {} -``` - -In the above snippet, the crate root is the `src/main.nr` file. The compiler sees the module -declaration `mod foo` which prompts it to look for a foo.nr file. - -Visually this module hierarchy looks like the following : - -``` -crate - ├── main - │ - └── foo - └── from_foo - -``` - -The module filename may also be the name of the module as a directory with the contents in a -file named `mod.nr` within that directory. The above example can alternatively be expressed like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -fn from_foo() {} -``` - -Note that it's an error to have both files `src/foo.nr` and `src/foo/mod.nr` in the filesystem. - -### Importing a module throughout the tree - -All modules are accessible from the `crate::` namespace. - -``` -crate - ├── bar - ├── foo - └── main - -``` - -In the above snippet, if `bar` would like to use functions in `foo`, it can do so by `use crate::foo::function_name`. - -### Sub-modules - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -fn from_bar() {} -``` - -In the above snippet, we have added an extra module to the module tree; `bar`. `bar` is a submodule -of `foo` hence we declare bar in `foo.nr` with `mod bar`. Since `foo` is not the crate root, the -compiler looks for the file associated with the `bar` module in `src/foo/bar.nr` - -Visually the module hierarchy looks as follows: - -``` -crate - ├── main - │ - └── foo - ├── from_foo - └── bar - └── from_bar -``` - -Similar to importing a module in the crate root, modules can be placed in a `mod.nr` file, like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar/mod.nr` - -```rust -fn from_bar() {} -``` - -### Referencing a parent module - -Given a submodule, you can refer to its parent module using the `super` keyword. - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; - -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -// Same as bar::from_foo -use super::from_foo; - -fn from_bar() { - from_foo(); // invokes super::from_foo(), which is bar::from_foo() - super::from_foo(); // also invokes bar::from_foo() -} -``` - -### `use` visibility - -`use` declarations are private to the containing module, by default. However, like functions, -they can be marked as `pub` or `pub(crate)`. Such a use declaration serves to _re-export_ a name. -A public `use` declaration can therefore redirect some public name to a different target definition: -even a definition with a private canonical path, inside a different module. - -An example of re-exporting: - -```rust -mod some_module { - pub use foo::{bar, baz}; - mod foo { - pub fn bar() {} - pub fn baz() {} - } -} - -fn main() { - some_module::bar(); - some_module::baz(); -} -``` - -In this example, the module `some_module` re-exports two public names defined in `foo`. - -### Visibility - -By default, like functions, modules are private to the module (or crate) they exist in. You can use `pub` -to make the module public or `pub(crate)` to make it public to just its crate: - -```rust -// This module is now public and can be seen by other crates. -pub mod foo; -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/workspaces.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/workspaces.md deleted file mode 100644 index 513497f12bf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/modules_packages_crates/workspaces.md +++ /dev/null @@ -1,42 +0,0 @@ ---- -title: Workspaces -sidebar_position: 3 ---- - -Workspaces are a feature of nargo that allow you to manage multiple related Noir packages in a single repository. A workspace is essentially a group of related projects that share common build output directories and configurations. - -Each Noir project (with it's own Nargo.toml file) can be thought of as a package. Each package is expected to contain exactly one "named circuit", being the "name" defined in Nargo.toml with the program logic defined in `./src/main.nr`. - -For a project with the following structure: - -```tree -├── crates -│ ├── a -│ │ ├── Nargo.toml -│ │ └── Prover.toml -│ │ └── src -│ │ └── main.nr -│ └── b -│ ├── Nargo.toml -│ └── Prover.toml -│ └── src -│ └── main.nr -│ -└── Nargo.toml -``` - -You can define a workspace in Nargo.toml like so: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -`members` indicates which packages are included in the workspace. As such, all member packages of a workspace will be processed when the `--workspace` flag is used with various commands or if a `default-member` is not specified. - -`default-member` indicates which package various commands process by default. - -Libraries can be defined in a workspace. Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/_category_.json deleted file mode 100644 index af04c0933fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Standard Library", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/bigint.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/bigint.md deleted file mode 100644 index 05c3011634f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/bigint.md +++ /dev/null @@ -1,127 +0,0 @@ ---- -title: Big Integers -description: How to use big integers from Noir standard library -keywords: - [ - Big Integer, - Noir programming language, - Noir libraries, - ] ---- - -The BigInt module in the standard library exposes some class of integers which do not fit (well) into a Noir native field. It implements modulo arithmetic, modulo a 'big' prime number. - -:::note - -The module can currently be considered as `Field`s with fixed modulo sizes used by a set of elliptic curves, in addition to just the native curve. [More work](https://github.com/noir-lang/noir/issues/510) is needed to achieve arbitrarily sized big integers. - -:::note - -`nargo` can be built with `--profile release-pedantic` to enable extra overflow checks which may affect `BigInt` results in some cases. -Consider the [`noir-bignum`](https://github.com/noir-lang/noir-bignum) library for an optimized alternative approach. - -::: - -Currently 6 classes of integers (i.e 'big' prime numbers) are available in the module, namely: - -- BN254 Fq: Bn254Fq -- BN254 Fr: Bn254Fr -- Secp256k1 Fq: Secpk1Fq -- Secp256k1 Fr: Secpk1Fr -- Secp256r1 Fr: Secpr1Fr -- Secp256r1 Fq: Secpr1Fq - -Where XXX Fq and XXX Fr denote respectively the order of the base and scalar field of the (usual) elliptic curve XXX. -For instance the big integer 'Secpk1Fq' in the standard library refers to integers modulo $2^{256}-2^{32}-977$. - -Feel free to explore the source code for the other primes: - -```rust title="big_int_definition" showLineNumbers -pub struct BigInt { - pointer: u32, - modulus: u32, -} -``` -> Source code: noir_stdlib/src/bigint.nr#L28-L33 - - -## Example usage - -A common use-case is when constructing a big integer from its bytes representation, and performing arithmetic operations on it: - -```rust title="big_int_example" showLineNumbers -fn big_int_example(x: u8, y: u8) { - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - let b = Secpk1Fq::from_le_bytes(&[y, x, 9]); - let c = (a + b) * b / a; - let d = c.to_le_bytes(); - println(d[0]); -} -``` -> Source code: test_programs/execution_success/bigint/src/main.nr#L74-L82 - - -## Methods - -The available operations for each big integer are: - -### from_le_bytes - -Construct a big integer from its little-endian bytes representation. Example: - -```rust - // Construct a big integer from a slice of bytes - let a = Secpk1Fq::from_le_bytes(&[x, y, 0, 45, 2]); - // Construct a big integer from an array of 32 bytes - let a = Secpk1Fq::from_le_bytes_32([1;32]); - ``` - -Sure, here's the formatted version of the remaining methods: - -### to_le_bytes - -Return the little-endian bytes representation of a big integer. Example: - -```rust -let bytes = a.to_le_bytes(); -``` - -### add - -Add two big integers. Example: - -```rust -let sum = a + b; -``` - -### sub - -Subtract two big integers. Example: - -```rust -let difference = a - b; -``` - -### mul - -Multiply two big integers. Example: - -```rust -let product = a * b; -``` - -### div - -Divide two big integers. Note that division is field division and not euclidean division. Example: - -```rust -let quotient = a / b; -``` - -### eq - -Compare two big integers. Example: - -```rust -let are_equal = a == b; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/black_box_fns.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/black_box_fns.md deleted file mode 100644 index d6079ab182c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/black_box_fns.md +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Black Box Functions -description: Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. -keywords: [noir, black box functions] ---- - -Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. This makes certain zk-snark unfriendly computations cheaper than if they were implemented in Noir. - -The ACVM spec defines a set of blackbox functions which backends will be expected to implement. This allows backends to use optimized implementations of these constraints if they have them, however they may also fallback to less efficient naive implementations if not. - -## Function list - -Here is a list of the current black box functions: - -- [AES128](./cryptographic_primitives/ciphers.mdx#aes128) -- [SHA256](./cryptographic_primitives/hashes.mdx#sha256) -- [Schnorr signature verification](./cryptographic_primitives/schnorr.mdx) -- [Blake2s](./cryptographic_primitives/hashes.mdx#blake2s) -- [Blake3](./cryptographic_primitives/hashes.mdx#blake3) -- [Pedersen Hash](./cryptographic_primitives/hashes.mdx#pedersen_hash) -- [Pedersen Commitment](./cryptographic_primitives/hashes.mdx#pedersen_commitment) -- [ECDSA signature verification](./cryptographic_primitives/ecdsa_sig_verification.mdx) -- [Embedded curve operations (MSM, addition, ...)](./cryptographic_primitives/embedded_curve_ops.mdx) -- AND -- XOR -- RANGE -- [Keccak256](./cryptographic_primitives/hashes.mdx#keccak256) -- [Recursive proof verification](./recursion.mdx) - -Most black box functions are included as part of the Noir standard library, however `AND`, `XOR` and `RANGE` are used as part of the Noir language syntax. For instance, using the bitwise operator `&` will invoke the `AND` black box function. - -You can view the black box functions defined in the ACVM code [here](https://github.com/noir-lang/noir/blob/master/acvm-repo/acir/src/circuit/black_box_functions.rs). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/bn254.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/bn254.md deleted file mode 100644 index 3294f005dbb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/bn254.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Bn254 Field Library ---- - -Noir provides a module in standard library with some optimized functions for bn254 Fr in `std::field::bn254`. - -## decompose - -```rust -fn decompose(x: Field) -> (Field, Field) {} -``` - -Decomposes a single field into two fields, low and high. The low field contains the lower 16 bytes of the input field and the high field contains the upper 16 bytes of the input field. Both field results are range checked to 128 bits. - - -## assert_gt - -```rust -fn assert_gt(a: Field, b: Field) {} -``` - -Asserts that a > b. This will generate less constraints than using `assert(gt(a, b))`. - -## assert_lt - -```rust -fn assert_lt(a: Field, b: Field) {} -``` - -Asserts that a < b. This will generate less constraints than using `assert(lt(a, b))`. - -## gt - -```rust -fn gt(a: Field, b: Field) -> bool {} -``` - -Returns true if a > b. - -## lt - -```rust -fn lt(a: Field, b: Field) -> bool {} -``` - -Returns true if a < b. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/boundedvec.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/boundedvec.md deleted file mode 100644 index 509b214bf3a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/boundedvec.md +++ /dev/null @@ -1,419 +0,0 @@ ---- -title: Bounded Vectors -keywords: [noir, vector, bounded vector, slice] -sidebar_position: 1 ---- - -A `BoundedVec` is a growable storage similar to a `Vec` except that it -is bounded with a maximum possible length. Unlike `Vec`, `BoundedVec` is not implemented -via slices and thus is not subject to the same restrictions slices are (notably, nested -slices - and thus nested vectors as well - are disallowed). - -Since a BoundedVec is backed by a normal array under the hood, growing the BoundedVec by -pushing an additional element is also more efficient - the length only needs to be increased -by one. - -For these reasons `BoundedVec` should generally be preferred over `Vec` when there -is a reasonable maximum bound that can be placed on the vector. - -Example: - -```rust -let mut vector: BoundedVec = BoundedVec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -assert(vector.max_len() == 10); -``` - -## Methods - -### new - -```rust -pub fn new() -> Self -``` - -Creates a new, empty vector of length zero. - -Since this container is backed by an array internally, it still needs an initial value -to give each element. To resolve this, each element is zeroed internally. This value -is guaranteed to be inaccessible unless `get_unchecked` is used. - -Example: - -```rust -let empty_vector: BoundedVec = BoundedVec::new(); -assert(empty_vector.len() == 0); -``` - -Note that whenever calling `new` the maximum length of the vector should always be specified -via a type signature: - -```rust title="new_example" showLineNumbers -fn good() -> BoundedVec { - // Ok! MaxLen is specified with a type annotation - let v1: BoundedVec = BoundedVec::new(); - let v2 = BoundedVec::new(); - - // Ok! MaxLen is known from the type of `good`'s return value - v2 -} - -fn bad() { - // Error: Type annotation needed - // The compiler can't infer `MaxLen` from this code. - let mut v3 = BoundedVec::new(); - v3.push(5); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L11-L27 - - -This defaulting of `MaxLen` (and numeric generics in general) to zero may change in future noir versions -but for now make sure to use type annotations when using bounded vectors. Otherwise, you will receive a constraint failure at runtime when the vec is pushed to. - -### get - -```rust -pub fn get(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this -will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - let last = v.get(v.len() - 1); - assert(first != last); -} -``` - -### get_unchecked - -```rust -pub fn get_unchecked(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero, without -performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, -it is unsafe! Use at your own risk! - -Example: - -```rust title="get_unchecked_example" showLineNumbers -fn sum_of_first_three(v: BoundedVec) -> u32 { - // Always ensure the length is larger than the largest - // index passed to get_unchecked - assert(v.len() > 2); - let first = v.get_unchecked(0); - let second = v.get_unchecked(1); - let third = v.get_unchecked(2); - first + second + third -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L54-L64 - - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - assert(first != 42); - v.set(0, 42); - let new_first = v.get(0); - assert(new_first == 42); -} -``` - -### set_unchecked - -```rust -pub fn set_unchecked(&mut self: Self, index: u64, value: T) -> T { -``` - -Writes an element to the vector at the given index, starting from zero, without performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, it is unsafe! Use at your own risk! - -Example: - -```rust title="set_unchecked_example" showLineNumbers -fn set_unchecked_example() { - let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([1, 2]); - - // Here we're safely writing within the valid range of `vec` - // `vec` now has the value [42, 2] - vec.set_unchecked(0, 42); - - // We can then safely read this value back out of `vec`. - // Notice that we use the checked version of `get` which would prevent reading unsafe values. - assert_eq(vec.get(0), 42); - - // We've now written past the end of `vec`. - // As this index is still within the maximum potential length of `v`, - // it won't cause a constraint failure. - vec.set_unchecked(2, 42); - println(vec); - - // This will write past the end of the maximum potential length of `vec`, - // it will then trigger a constraint failure. - vec.set_unchecked(5, 42); - println(vec); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L67-L91 - - - -### push - -```rust -pub fn push(&mut self, elem: T) { -``` - -Pushes an element to the end of the vector. This increases the length -of the vector by one. - -Panics if the new length of the vector will be greater than the max length. - -Example: - -```rust title="bounded-vec-push-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - v.push(1); - v.push(2); - - // Panics with failed assertion "push out of bounds" - v.push(3); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L95-L103 - - -### pop - -```rust -pub fn pop(&mut self) -> T -``` - -Pops the element at the end of the vector. This will decrease the length -of the vector by one. - -Panics if the vector is empty. - -Example: - -```rust title="bounded-vec-pop-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.push(1); - v.push(2); - - let two = v.pop(); - let one = v.pop(); - - assert(two == 2); - assert(one == 1); - // error: cannot pop from an empty vector - // let _ = v.pop(); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L108-L120 - - -### len - -```rust -pub fn len(self) -> u64 { -``` - -Returns the current length of this vector - -Example: - -```rust title="bounded-vec-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - assert(v.len() == 0); - - v.push(100); - assert(v.len() == 1); - - v.push(200); - v.push(300); - v.push(400); - assert(v.len() == 4); - - let _ = v.pop(); - let _ = v.pop(); - assert(v.len() == 2); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L125-L140 - - -### max_len - -```rust -pub fn max_len(_self: BoundedVec) -> u64 { -``` - -Returns the maximum length of this vector. This is always -equal to the `MaxLen` parameter this vector was initialized with. - -Example: - -```rust title="bounded-vec-max-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.max_len() == 5); - v.push(10); - assert(v.max_len() == 5); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L145-L151 - - -### storage - -```rust -pub fn storage(self) -> [T; MaxLen] { -``` - -Returns the internal array within this vector. -Since arrays in Noir are immutable, mutating the returned storage array will not mutate -the storage held internally by this vector. - -Note that uninitialized elements may be zeroed out! - -Example: - -```rust title="bounded-vec-storage-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.storage() == [0, 0, 0, 0, 0]); - - v.push(57); - assert(v.storage() == [57, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L156-L163 - - -### extend_from_array - -```rust -pub fn extend_from_array(&mut self, array: [T; Len]) -``` - -Pushes each element from the given array to this vector. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-array-example" showLineNumbers -let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([2, 4]); - - assert(vec.len == 2); - assert(vec.get(0) == 2); - assert(vec.get(1) == 4); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L168-L175 - - -### extend_from_bounded_vec - -```rust -pub fn extend_from_bounded_vec(&mut self, vec: BoundedVec) -``` - -Pushes each element from the other vector to this vector. The length of -the other vector is left unchanged. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-bounded-vec-example" showLineNumbers -let mut v1: BoundedVec = BoundedVec::new(); - let mut v2: BoundedVec = BoundedVec::new(); - - v2.extend_from_array([1, 2, 3]); - v1.extend_from_bounded_vec(v2); - - assert(v1.storage() == [1, 2, 3, 0, 0]); - assert(v2.storage() == [1, 2, 3, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L180-L189 - - -### from_array - -```rust -pub fn from_array(array: [T; Len]) -> Self -``` - -Creates a new vector, populating it with values derived from an array input. -The maximum length of the vector is determined based on the type signature. - -Example: -```rust -let bounded_vec: BoundedVec = BoundedVec::from_array([1, 2, 3]) -``` - -### map - -```rust -pub fn map(self, f: fn[Env](T) -> U) -> BoundedVec -``` - -Creates a new vector of equal size by calling a closure on each element in this vector. - -Example: - -```rust title="bounded-vec-map-example" showLineNumbers -let vec: BoundedVec = BoundedVec::from_array([1, 2, 3, 4]); - let result = vec.map(|value| value * 2); -``` -> Source code: noir_stdlib/src/collections/bounded_vec.nr#L495-L498 - - -### any - -```rust -pub fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -Returns true if the given predicate returns true for any element -in this vector. - -Example: - -```rust title="bounded-vec-any-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.extend_from_array([2, 4, 6]); - - let all_even = !v.any(|elem: u32| elem % 2 != 0); - assert(all_even); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L256-L262 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/hashmap.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/hashmap.md deleted file mode 100644 index 395cc312705..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/hashmap.md +++ /dev/null @@ -1,587 +0,0 @@ ---- -title: HashMap -keywords: [noir, map, hash, hashmap] -sidebar_position: 1 ---- - -`HashMap` is used to efficiently store and look up key-value pairs. - -`HashMap` is a bounded type which can store anywhere from zero to `MaxLen` total elements. -Note that due to hash collisions, the actual maximum number of elements stored by any particular -hashmap is likely lower than `MaxLen`. This is true even with cryptographic hash functions since -every hash value will be performed modulo `MaxLen`. - -Example: - -```rust -// Create a mapping from Fields to u32s with a maximum length of 12 -// using a poseidon2 hasher -use std::hash::poseidon2::Poseidon2Hasher; -let mut map: HashMap> = HashMap::default(); - -map.insert(1, 2); -map.insert(3, 4); - -let two = map.get(1).unwrap(); -``` - -## Methods - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default, -{ - /// Constructs an empty HashMap. - /// - /// Example: - /// - /// ```noir - /// let hashmap: HashMap> = HashMap::default(); - /// assert(hashmap.is_empty()); - /// ``` - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L681-L696 - - -Creates a fresh, empty HashMap. - -When using this function, always make sure to specify the maximum size of the hash map. - -This is the same `default` from the `Default` implementation given further below. It is -repeated here for convenience since it is the recommended way to create a hashmap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L207-L210 - - -Because `HashMap` has so many generic arguments that are likely to be the same throughout -your program, it may be helpful to create a type alias: - -```rust title="type_alias" showLineNumbers -type MyMap = HashMap>; -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L203 - - -### with_hasher - -```rust title="with_hasher" showLineNumbers -pub fn with_hasher(_build_hasher: B) -> Self - where - B: BuildHasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L103-L108 - - -Creates a hashmap with an existing `BuildHasher`. This can be used to ensure multiple -hashmaps are created with the same hasher instance. - -Example: - -```rust title="with_hasher_example" showLineNumbers -let my_hasher: BuildHasherDefault = Default::default(); - let hashmap: HashMap> = - HashMap::with_hasher(my_hasher); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L211-L216 - - -### get - -```rust title="get" showLineNumbers -pub fn get(self, key: K) -> Option - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L465-L472 - - -Retrieves a value from the hashmap, returning `Option::none()` if it was not found. - -Example: - -```rust title="get_example" showLineNumbers -fn get_example(map: HashMap>) { - let x = map.get(12); - - if x.is_some() { - assert(x.unwrap() == 42); - } -} -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L296-L304 - - -### insert - -```rust title="insert" showLineNumbers -pub fn insert(&mut self, key: K, value: V) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L507-L514 - - -Inserts a new key-value pair into the map. If the key was already in the map, its -previous value will be overridden with the newly provided one. - -Example: - -```rust title="insert_example" showLineNumbers -let mut map: HashMap> = HashMap::default(); - map.insert(12, 42); - assert(map.len() == 1); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L217-L221 - - -### remove - -```rust title="remove" showLineNumbers -pub fn remove(&mut self, key: K) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L563-L570 - - -Removes the given key-value pair from the map. If the key was not already present -in the map, this does nothing. - -Example: - -```rust title="remove_example" showLineNumbers -map.remove(12); - assert(map.is_empty()); - - // If a key was not present in the map, remove does nothing - map.remove(12); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L224-L231 - - -### is_empty - -```rust title="is_empty" showLineNumbers -pub fn is_empty(self) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L167-L169 - - -True if the length of the hash map is empty. - -Example: - -```rust title="is_empty_example" showLineNumbers -assert(map.is_empty()); - - map.insert(1, 2); - assert(!map.is_empty()); - - map.remove(1); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L232-L240 - - -### len - -```rust title="len" showLineNumbers -pub fn len(self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L424-L426 - - -Returns the current length of this hash map. - -Example: - -```rust title="len_example" showLineNumbers -// This is equivalent to checking map.is_empty() - assert(map.len() == 0); - - map.insert(1, 2); - map.insert(3, 4); - map.insert(5, 6); - assert(map.len() == 3); - - // 3 was already present as a key in the hash map, so the length is unchanged - map.insert(3, 7); - assert(map.len() == 3); - - map.remove(1); - assert(map.len() == 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L241-L256 - - -### capacity - -```rust title="capacity" showLineNumbers -pub fn capacity(_self: Self) -> u32 { -``` -> Source code: noir_stdlib/src/collections/map.nr#L446-L448 - - -Returns the maximum capacity of this hashmap. This is always equal to the capacity -specified in the hashmap's type. - -Unlike hashmaps in general purpose programming languages, hashmaps in Noir have a -static capacity that does not increase as the map grows larger. Thus, this capacity -is also the maximum possible element count that can be inserted into the hashmap. -Due to hash collisions (modulo the hashmap length), it is likely the actual maximum -element count will be lower than the full capacity. - -Example: - -```rust title="capacity_example" showLineNumbers -let empty_map: HashMap> = - HashMap::default(); - assert(empty_map.len() == 0); - assert(empty_map.capacity() == 42); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L257-L262 - - -### clear - -```rust title="clear" showLineNumbers -pub fn clear(&mut self) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L123-L125 - - -Clears the hashmap, removing all key-value pairs from it. - -Example: - -```rust title="clear_example" showLineNumbers -assert(!map.is_empty()); - map.clear(); - assert(map.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L263-L267 - - -### contains_key - -```rust title="contains_key" showLineNumbers -pub fn contains_key(self, key: K) -> bool - where - K: Hash + Eq, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L143-L150 - - -True if the hashmap contains the given key. Unlike `get`, this will not also return -the value associated with the key. - -Example: - -```rust title="contains_key_example" showLineNumbers -if map.contains_key(7) { - let value = map.get(7); - assert(value.is_some()); - } else { - println("No value for key 7!"); - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L268-L275 - - -### entries - -```rust title="entries" showLineNumbers -pub fn entries(self) -> BoundedVec<(K, V), N> { -``` -> Source code: noir_stdlib/src/collections/map.nr#L191-L193 - - -Returns a vector of each key-value pair present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="entries_example" showLineNumbers -let entries = map.entries(); - - // The length of a hashmap may not be compile-time known, so we - // need to loop over its capacity instead - for i in 0..map.capacity() { - if i < entries.len() { - let (key, value) = entries.get(i); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L307-L318 - - -### keys - -```rust title="keys" showLineNumbers -pub fn keys(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L227-L229 - - -Returns a vector of each key present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="keys_example" showLineNumbers -let keys = map.keys(); - - for i in 0..keys.max_len() { - if i < keys.len() { - let key = keys.get_unchecked(i); - let value = map.get(key).unwrap_unchecked(); - println(f"{key} -> {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L319-L329 - - -### values - -```rust title="values" showLineNumbers -pub fn values(self) -> BoundedVec { -``` -> Source code: noir_stdlib/src/collections/map.nr#L262-L264 - - -Returns a vector of each value present in the hashmap. - -The length of the returned vector is always equal to the length of the hashmap. - -Example: - -```rust title="values_example" showLineNumbers -let values = map.values(); - - for i in 0..values.max_len() { - if i < values.len() { - let value = values.get_unchecked(i); - println(f"Found value {value}"); - } - } -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L330-L339 - - -### iter_mut - -```rust title="iter_mut" showLineNumbers -pub fn iter_mut(&mut self, f: fn(K, V) -> (K, V)) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L297-L304 - - -Iterates through each key-value pair of the HashMap, setting each key-value pair to the -result returned from the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If this is not desired, use `iter_values_mut` if only values need to be mutated, -or `entries` if neither keys nor values need to be mutated. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_mut_example" showLineNumbers -// Add 1 to each key in the map, and double the value associated with that key. - map.iter_mut(|k, v| (k + 1, v * 2)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L343-L346 - - -### iter_keys_mut - -```rust title="iter_keys_mut" showLineNumbers -pub fn iter_keys_mut(&mut self, f: fn(K) -> K) - where - K: Eq + Hash, - B: BuildHasher, - H: Hasher, - { -``` -> Source code: noir_stdlib/src/collections/map.nr#L335-L342 - - -Iterates through the HashMap, mutating each key to the result returned from -the given function. - -Note that since keys can be mutated, the HashMap needs to be rebuilt as it is iterated -through. If only iteration is desired and the keys are not intended to be mutated, -prefer using `entries` instead. - -The iteration order is left unspecified. As a result, if two keys are mutated to become -equal, which of the two values that will be present for the key in the resulting map is also unspecified. - -Example: - -```rust title="iter_keys_mut_example" showLineNumbers -// Double each key, leaving the value associated with that key untouched - map.iter_keys_mut(|k| k * 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L347-L350 - - -### iter_values_mut - -```rust title="iter_values_mut" showLineNumbers -pub fn iter_values_mut(&mut self, f: fn(V) -> V) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L367-L369 - - -Iterates through the HashMap, applying the given function to each value and mutating the -value to equal the result. This function is more efficient than `iter_mut` and `iter_keys_mut` -because the keys are untouched and the underlying hashmap thus does not need to be reordered. - -Example: - -```rust title="iter_values_mut_example" showLineNumbers -// Halve each value - map.iter_values_mut(|v| v / 2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L351-L354 - - -### retain - -```rust title="retain" showLineNumbers -pub fn retain(&mut self, f: fn(K, V) -> bool) { -``` -> Source code: noir_stdlib/src/collections/map.nr#L388-L390 - - -Retains only the key-value pairs for which the given function returns true. -Any key-value pairs for which the function returns false will be removed from the map. - -Example: - -```rust title="retain_example" showLineNumbers -map.retain(|k, v| (k != 0) & (v != 0)); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L279-L281 - - -## Trait Implementations - -### default - -```rust title="default" showLineNumbers -impl Default for HashMap -where - B: BuildHasher + Default, - H: Hasher + Default, -{ - /// Constructs an empty HashMap. - /// - /// Example: - /// - /// ```noir - /// let hashmap: HashMap> = HashMap::default(); - /// assert(hashmap.is_empty()); - /// ``` - fn default() -> Self { -``` -> Source code: noir_stdlib/src/collections/map.nr#L681-L696 - - -Constructs an empty HashMap. - -Example: - -```rust title="default_example" showLineNumbers -let hashmap: HashMap> = HashMap::default(); - assert(hashmap.is_empty()); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L207-L210 - - -### eq - -```rust title="eq" showLineNumbers -impl Eq for HashMap -where - K: Eq + Hash, - V: Eq, - B: BuildHasher, - H: Hasher, -{ - /// Checks if two HashMaps are equal. - /// - /// Example: - /// - /// ```noir - /// let mut map1: HashMap> = HashMap::default(); - /// let mut map2: HashMap> = HashMap::default(); - /// - /// map1.insert(1, 2); - /// map1.insert(3, 4); - /// - /// map2.insert(3, 4); - /// map2.insert(1, 2); - /// - /// assert(map1 == map2); - /// ``` - fn eq(self, other: HashMap) -> bool { -``` -> Source code: noir_stdlib/src/collections/map.nr#L629-L654 - - -Checks if two HashMaps are equal. - -Example: - -```rust title="eq_example" showLineNumbers -let mut map1: HashMap> = HashMap::default(); - let mut map2: HashMap> = HashMap::default(); - - map1.insert(1, 2); - map1.insert(3, 4); - - map2.insert(3, 4); - map2.insert(1, 2); - - assert(map1 == map2); -``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L282-L293 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/index.md deleted file mode 100644 index ea84c6d5c21..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/index.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -title: Containers -description: Container types provided by Noir's standard library for storing and retrieving data -keywords: [containers, data types, vec, hashmap] ---- diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/vec.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/vec.mdx deleted file mode 100644 index 475011922f8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/containers/vec.mdx +++ /dev/null @@ -1,170 +0,0 @@ ---- -title: Vectors -description: Delve into the Vec data type in Noir. Learn about its methods, practical examples, and best practices for using Vectors in your Noir code. -keywords: [noir, vector type, methods, examples, dynamic arrays] -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A vector is a collection type similar to Rust's `Vec` type. In Noir, it is a convenient way to use slices as mutable arrays. - -Example: - -```rust -let mut vector: Vec = Vec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -``` - -## Methods - -### new - -Creates a new, empty vector. - -```rust -pub fn new() -> Self -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### from_slice - -Creates a vector containing each element from a given slice. Mutations to the resulting vector will not affect the original slice. - -```rust -pub fn from_slice(slice: [T]) -> Self -``` - -Example: - -```rust -let slice: [Field] = &[1, 2, 3]; -let vector_from_slice = Vec::from_slice(slice); -assert(vector_from_slice.len() == 3); -``` - -### len - -Returns the number of elements in the vector. - -```rust -pub fn len(self) -> Field -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### get - -Retrieves an element from the vector at a given index. Panics if the index points beyond the vector's end. - -```rust -pub fn get(self, index: Field) -> T -``` - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -``` - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -Panics if the index points beyond the vector's end. - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -vector.set(1, 42); -assert(vector.get(1) == 42); -``` - -### push - -Adds a new element to the vector's end, returning a new vector with a length one greater than the original unmodified vector. - -```rust -pub fn push(&mut self, elem: T) -``` - -Example: - -```rust -let mut vector: Vec = Vec::new(); -vector.push(10); -assert(vector.len() == 1); -``` - -### pop - -Removes an element from the vector's end, returning a new vector with a length one less than the original vector, along with the removed element. Panics if the vector's length is zero. - -```rust -pub fn pop(&mut self) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20]); -let popped_elem = vector.pop(); -assert(popped_elem == 20); -assert(vector.len() == 1); -``` - -### insert - -Inserts an element at a specified index, shifting subsequent elements to the right. - -```rust -pub fn insert(&mut self, index: Field, elem: T) -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 30]); -vector.insert(1, 20); -assert(vector.get(1) == 20); -``` - -### remove - -Removes an element at a specified index, shifting subsequent elements to the left, and returns the removed element. - -```rust -pub fn remove(&mut self, index: Field) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20, 30]); -let removed_elem = vector.remove(1); -assert(removed_elem == 20); -assert(vector.len() == 2); -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ciphers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ciphers.mdx deleted file mode 100644 index d6a5e1a79eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ciphers.mdx +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Ciphers -description: - Learn about the implemented ciphers ready to use for any Noir project -keywords: - [ciphers, Noir project, aes128, encrypt] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## aes128 - -Given a plaintext as an array of bytes, returns the corresponding aes128 ciphertext (CBC mode). Input padding is automatically performed using PKCS#7, so that the output length is `input.len() + (16 - input.len() % 16)`. - -```rust title="aes128" showLineNumbers -pub fn aes128_encrypt(input: [u8; N], iv: [u8; 16], key: [u8; 16]) -> [u8] {} -``` -> Source code: noir_stdlib/src/aes128.nr#L2-L4 - - -```rust -fn main() { - let input: [u8; 4] = [0, 12, 3, 15] // Random bytes, will be padded to 16 bytes. - let iv: [u8; 16] = [0; 16]; // Initialisation vector - let key: [u8; 16] = [0; 16] // AES key - let ciphertext = std::aes128::aes128_encrypt(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()); // In this case, the output length will be 16 bytes. -} -``` - - - \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ec_primitives.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ec_primitives.md deleted file mode 100644 index f262d8160d6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ec_primitives.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Elliptic Curve Primitives -keywords: [cryptographic primitives, Noir project] -sidebar_position: 4 ---- - -Data structures and methods on them that allow you to carry out computations involving elliptic -curves over the (mathematical) field corresponding to `Field`. For the field currently at our -disposal, applications would involve a curve embedded in BN254, e.g. the -[Baby Jubjub curve](https://eips.ethereum.org/EIPS/eip-2494). - -## Data structures - -### Elliptic curve configurations - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Curve`), i.e. the specific elliptic -curve you want to use, which would be specified using any one of the methods -`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::new` which take the coefficients in the -defining equation together with a generator point as parameters. You can find more detail in the -comments in -[`noir_stdlib/src/ec/mod.nr`](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr), but -the gist of it is that the elliptic curves of interest are usually expressed in one of the standard -forms implemented here (Twisted Edwards, Montgomery and Short Weierstraß), and in addition to that, -you could choose to use `affine` coordinates (Cartesian coordinates - the usual (x,y) - possibly -together with a point at infinity) or `curvegroup` coordinates (some form of projective coordinates -requiring more coordinates but allowing for more efficient implementations of elliptic curve -operations). Conversions between all of these forms are provided, and under the hood these -conversions are done whenever an operation is more efficient in a different representation (or a -mixed coordinate representation is employed). - -### Points - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Point`), i.e. points lying on the -elliptic curve. For a curve configuration `c` and a point `p`, it may be checked that `p` -does indeed lie on `c` by calling `c.contains(p1)`. - -## Methods - -(given a choice of curve representation, e.g. use `std::ec::tecurve::affine::Curve` and use -`std::ec::tecurve::affine::Point`) - -- The **zero element** is given by `Point::zero()`, and we can verify whether a point `p: Point` is - zero by calling `p.is_zero()`. -- **Equality**: Points `p1: Point` and `p2: Point` may be checked for equality by calling - `p1.eq(p2)`. -- **Addition**: For `c: Curve` and points `p1: Point` and `p2: Point` on the curve, adding these two - points is accomplished by calling `c.add(p1,p2)`. -- **Negation**: For a point `p: Point`, `p.negate()` is its negation. -- **Subtraction**: For `c` and `p1`, `p2` as above, subtracting `p2` from `p1` is accomplished by - calling `c.subtract(p1,p2)`. -- **Scalar multiplication**: For `c` as above, `p: Point` a point on the curve and `n: Field`, - scalar multiplication is given by `c.mul(n,p)`. If instead `n :: [u1; N]`, i.e. `n` is a bit - array, the `bit_mul` method may be used instead: `c.bit_mul(n,p)` -- **Multi-scalar multiplication**: For `c` as above and arrays `n: [Field; N]` and `p: [Point; N]`, - multi-scalar multiplication is given by `c.msm(n,p)`. -- **Coordinate representation conversions**: The `into_group` method converts a point or curve - configuration in the affine representation to one in the CurveGroup representation, and - `into_affine` goes in the other direction. -- **Curve representation conversions**: `tecurve` and `montcurve` curves and points are equivalent - and may be converted between one another by calling `into_montcurve` or `into_tecurve` on their - configurations or points. `swcurve` is more general and a curve c of one of the other two types - may be converted to this representation by calling `c.into_swcurve()`, whereas a point `p` lying - on the curve given by `c` may be mapped to its corresponding `swcurve` point by calling - `c.map_into_swcurve(p)`. -- **Map-to-curve methods**: The Elligator 2 method of mapping a field element `n: Field` into a - `tecurve` or `montcurve` with configuration `c` may be called as `c.elligator2_map(n)`. For all of - the curve configurations, the SWU map-to-curve method may be called as `c.swu_map(z,n)`, where - `z: Field` depends on `Field` and `c` and must be chosen by the user (the conditions it needs to - satisfy are specified in the comments - [here](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr)). - -## Examples - -The -[ec_baby_jubjub test](https://github.com/noir-lang/noir/blob/master/test_programs/compile_success_empty/ec_baby_jubjub/src/main.nr) -illustrates all of the above primitives on various forms of the Baby Jubjub curve. A couple of more -interesting examples in Noir would be: - -Public-key cryptography: Given an elliptic curve and a 'base point' on it, determine the public key -from the private key. This is a matter of using scalar multiplication. In the case of Baby Jubjub, -for example, this code would do: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; - -fn bjj_pub_key(priv_key: Field) -> Point -{ - - let bjj = Curve::new(168700, 168696, G::new(995203441582195749578291179787384436505546430278305826713579947235728471134,5472060717959818805561601436314318772137091100104008585924551046643952123905)); - - let base_pt = Point::new(5299619240641551281634865583518297030282874472190772894086521144482721001553, 16950150798460657717958625567821834550301663161624707787222815936182638968203); - - bjj.mul(priv_key,base_pt) -} -``` - -This would come in handy in a Merkle proof. - -- EdDSA signature verification: This is a matter of combining these primitives with a suitable hash - function. See - [feat(stdlib): EdDSA sig verification noir#1136](https://github.com/noir-lang/noir/pull/1136) for - the case of Baby Jubjub and the Poseidon hash function. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx deleted file mode 100644 index 8d96027b42c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: ECDSA Signature Verification -description: Learn about the cryptographic primitives regarding ECDSA over the secp256k1 and secp256r1 curves -keywords: [cryptographic primitives, Noir project, ecdsa, secp256k1, secp256r1, signatures] -sidebar_position: 3 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports ECDSA signatures verification over the secp256k1 and secp256r1 curves. - -## ecdsa_secp256k1::verify_signature - -Verifier for ECDSA Secp256k1 signatures. -See ecdsa_secp256k1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256k1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N], -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256k1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256k1::verify_signature_slice - -Verifier for ECDSA Secp256k1 signatures where the message is a slice. - -```rust title="ecdsa_secp256k1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8], -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L13-L20 - - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures. -See ecdsa_secp256r1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256r1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N], -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256r1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures where the message is a slice. - -```rust title="ecdsa_secp256r1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8], -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/eddsa.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/eddsa.mdx deleted file mode 100644 index b283de693c8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/eddsa.mdx +++ /dev/null @@ -1,37 +0,0 @@ ---- -title: EdDSA Verification -description: Learn about the cryptographic primitives regarding EdDSA -keywords: [cryptographic primitives, Noir project, eddsa, signatures] -sidebar_position: 5 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## eddsa::eddsa_poseidon_verify - -Verifier for EdDSA signatures - -```rust -fn eddsa_poseidon_verify(public_key_x : Field, public_key_y : Field, signature_s: Field, signature_r8_x: Field, signature_r8_y: Field, message: Field) -> bool -``` - -It is also possible to specify the hash algorithm used for the signature by using the `eddsa_verify` function by passing a type implementing the Hasher trait with the turbofish operator. -For instance, if you want to use Poseidon2 instead, you can do the following: -```rust -use std::hash::poseidon2::Poseidon2Hasher; - -eddsa_verify::(pub_key_a.x, pub_key_a.y, s_a, r8_a.x, r8_a.y, msg); -``` - - - -## eddsa::eddsa_to_pub - -Private to public key conversion. - -Returns `(pub_key_x, pub_key_y)` - -```rust -fn eddsa_to_pub(secret : Field) -> (Field, Field) -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx deleted file mode 100644 index 482a36932b9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx +++ /dev/null @@ -1,95 +0,0 @@ ---- -title: Scalar multiplication -description: See how you can perform scalar multiplication in Noir -keywords: [cryptographic primitives, Noir project, scalar multiplication] -sidebar_position: 1 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -The following functions perform operations over the embedded curve whose coordinates are defined by the configured noir field. -For the BN254 scalar field, this is BabyJubJub or Grumpkin. - -:::note -Suffixes `_low` and `_high` denote low and high limbs of a scalar. -::: - -## embedded_curve_ops::multi_scalar_mul - -Performs multi scalar multiplication over the embedded curve. -The function accepts arbitrary amount of point-scalar pairs on the input, it multiplies the individual pairs over -the curve and returns a sum of the resulting points. - -Points represented as x and y coordinates [x1, y1, x2, y2, ...], scalars as low and high limbs [low1, high1, low2, high2, ...]. - -```rust title="multi_scalar_mul" showLineNumbers -pub fn multi_scalar_mul( - points: [EmbeddedCurvePoint; N], - scalars: [EmbeddedCurveScalar; N], -) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L103-L108 - - -example - -```rust -fn main(point_x: Field, point_y: Field, scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::multi_scalar_mul([point_x, point_y], [scalar_low, scalar_high]); - println(point); -} -``` - -## embedded_curve_ops::fixed_base_scalar_mul - -Performs fixed base scalar multiplication over the embedded curve (multiplies input scalar with a generator point). -The function accepts a single scalar on the input represented as 2 fields. - -```rust title="fixed_base_scalar_mul" showLineNumbers -pub fn fixed_base_scalar_mul(scalar: EmbeddedCurveScalar) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L120-L122 - - -example - -```rust -fn main(scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::fixed_base_scalar_mul(scalar_low, scalar_high); - println(point); -} -``` - -## embedded_curve_ops::embedded_curve_add - -Adds two points on the embedded curve. -This function takes two `EmbeddedCurvePoint` structures as parameters, representing points on the curve, and returns a new `EmbeddedCurvePoint` structure that represents their sum. - -### Parameters: -- `point1` (`EmbeddedCurvePoint`): The first point to add. -- `point2` (`EmbeddedCurvePoint`): The second point to add. - -### Returns: -- `EmbeddedCurvePoint`: The resulting point after the addition of `point1` and `point2`. - -```rust title="embedded_curve_add" showLineNumbers -pub fn embedded_curve_add( - point1: EmbeddedCurvePoint, - point2: EmbeddedCurvePoint, -) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L136-L141 - - -example - -```rust -fn main() { - let point1 = EmbeddedCurvePoint { x: 1, y: 2 }; - let point2 = EmbeddedCurvePoint { x: 3, y: 4 }; - let result = std::embedded_curve_ops::embedded_curve_add(point1, point2); - println!("Resulting Point: ({}, {})", result.x, result.y); -} -``` - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/hashes.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/hashes.mdx deleted file mode 100644 index 541a1971561..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/hashes.mdx +++ /dev/null @@ -1,227 +0,0 @@ ---- -title: Hash methods -description: - Learn about the cryptographic primitives ready to use for any Noir project, including sha256, - blake2s and pedersen -keywords: - [cryptographic primitives, Noir project, sha256, blake2s, pedersen, hash] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## sha256 - -Given an array of bytes, returns the resulting sha256 hash. -Specify a message_size to hash only the first `message_size` bytes of the input. - -```rust title="sha256" showLineNumbers -pub fn sha256(input: [u8; N]) -> HASH -``` -> Source code: noir_stdlib/src/hash/sha256.nr#L47-L49 - - -example: -```rust title="sha256_var" showLineNumbers -let digest = std::hash::sha256_var([x as u8], 1); -``` -> Source code: test_programs/execution_success/sha256/src/main.nr#L15-L17 - - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::sha256::sha256_var(x, 4); -} -``` - - - - -## blake2s - -Given an array of bytes, returns an array with the Blake2 hash - -```rust title="blake2s" showLineNumbers -pub fn blake2s(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L18-L20 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake2s(x); -} -``` - - - -## blake3 - -Given an array of bytes, returns an array with the Blake3 hash - -```rust title="blake3" showLineNumbers -pub fn blake3(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L24-L26 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake3(x); -} -``` - - - -## pedersen_hash - -Given an array of Fields, returns the Pedersen hash. - -```rust title="pedersen_hash" showLineNumbers -pub fn pedersen_hash(input: [Field; N]) -> Field -``` -> Source code: noir_stdlib/src/hash/mod.nr#L49-L51 - - -example: - -```rust title="pedersen-hash" showLineNumbers -fn main(x: Field, y: Field, expected_hash: Field) { - let hash = std::hash::pedersen_hash([x, y]); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/pedersen_hash/src/main.nr#L1-L6 - - - - -## pedersen_commitment - -Given an array of Fields, returns the Pedersen commitment. - -```rust title="pedersen_commitment" showLineNumbers -pub fn pedersen_commitment(input: [Field; N]) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/hash/mod.nr#L29-L31 - - -example: - -```rust title="pedersen-commitment" showLineNumbers -fn main(x: Field, y: Field, expected_commitment: std::embedded_curve_ops::EmbeddedCurvePoint) { - let commitment = std::hash::pedersen_commitment([x, y]); - assert_eq(commitment.x, expected_commitment.x); - assert_eq(commitment.y, expected_commitment.y); -} -``` -> Source code: test_programs/execution_success/pedersen_commitment/src/main.nr#L1-L7 - - - - -## keccak256 - -Given an array of bytes (`u8`), returns the resulting keccak hash as an array of -32 bytes (`[u8; 32]`). Specify a message_size to hash only the first -`message_size` bytes of the input. - -```rust title="keccak256" showLineNumbers -pub fn keccak256(input: [u8; N], message_size: u32) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L116-L118 - - -example: - -```rust title="keccak256" showLineNumbers -fn main(x: Field, result: [u8; 32]) { - // We use the `as` keyword here to denote the fact that we want to take just the first byte from the x Field - // The padding is taken care of by the program - let digest = std::hash::keccak256([x as u8], 1); - assert(digest == result); - - //#1399: variable message size - let message_size = 4; - let hash_a = std::hash::keccak256([1, 2, 3, 4], message_size); - let hash_b = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size); - - assert(hash_a == hash_b); - - let message_size_big = 8; - let hash_c = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size_big); - - assert(hash_a != hash_c); -} -``` -> Source code: test_programs/execution_success/keccak256/src/main.nr#L1-L20 - - - - -## poseidon - -Given an array of Fields, returns a new Field with the Poseidon Hash. Mind that you need to specify -how many inputs are there to your Poseidon function. - -```rust -// example for hash_1, hash_2 accepts an array of length 2, etc -fn hash_1(input: [Field; 1]) -> Field -``` - -example: - -```rust title="poseidon" showLineNumbers -use std::hash::poseidon; - -fn main(x1: [Field; 2], y1: pub Field, x2: [Field; 4], y2: pub Field) { - let hash1 = poseidon::bn254::hash_2(x1); - assert(hash1 == y1); - - let hash2 = poseidon::bn254::hash_4(x2); - assert(hash2 == y2); -} -``` -> Source code: test_programs/execution_success/poseidon_bn254_hash/src/main.nr#L1-L11 - - -## poseidon 2 - -Given an array of Fields, returns a new Field with the Poseidon2 Hash. Contrary to the Poseidon -function, there is only one hash and you can specify a message_size to hash only the first -`message_size` bytes of the input, - -```rust -// example for hashing the first three elements of the input -Poseidon2::hash(input, 3); -``` - -example: - -```rust title="poseidon2" showLineNumbers -use std::hash::poseidon2; - -fn main(inputs: [Field; 4], expected_hash: Field) { - let hash = poseidon2::Poseidon2::hash(inputs, inputs.len()); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/poseidon2/src/main.nr#L1-L8 - - -## hash_to_field - -```rust -fn hash_to_field(_input : [Field]) -> Field {} -``` - -Calculates the `blake2s` hash of the inputs and returns the hash modulo the field modulus to return -a value which can be represented as a `Field`. - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/index.md deleted file mode 100644 index 650f30165d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/index.md +++ /dev/null @@ -1,14 +0,0 @@ ---- -title: Cryptographic Primitives -description: - Learn about the cryptographic primitives ready to use for any Noir project -keywords: - [ - cryptographic primitives, - Noir project, - ] ---- - -The Noir team is progressively adding new cryptographic primitives to the standard library. Reach out for news or if you would be interested in adding more of these calculations in Noir. - -Some methods are available thanks to the Aztec backend, not being performed using Noir. When using other backends, these methods may or may not be supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/schnorr.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/schnorr.mdx deleted file mode 100644 index 030452645c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/cryptographic_primitives/schnorr.mdx +++ /dev/null @@ -1,64 +0,0 @@ ---- -title: Schnorr Signatures -description: Learn how you can verify Schnorr signatures using Noir -keywords: [cryptographic primitives, Noir project, schnorr, signatures] -sidebar_position: 2 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## schnorr::verify_signature - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin). -See schnorr::verify_signature_slice for a version that works directly on slices. - -```rust title="schnorr_verify" showLineNumbers -pub fn verify_signature( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8; N], -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L4-L11 - - -where `_signature` can be generated like so using the npm package -[@noir-lang/barretenberg](https://www.npmjs.com/package/@noir-lang/barretenberg) - -```js -const { BarretenbergWasm } = require('@noir-lang/barretenberg/dest/wasm'); -const { Schnorr } = require('@noir-lang/barretenberg/dest/crypto/schnorr'); - -... - -const barretenberg = await BarretenbergWasm.new(); -const schnorr = new Schnorr(barretenberg); -const pubKey = schnorr.computePublicKey(privateKey); -const message = ... -const signature = Array.from( - schnorr.constructSignature(hash, privateKey).toBuffer() -); - -... -``` - - - -## schnorr::verify_signature_slice - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin) -where the message is a slice. - -```rust title="schnorr_verify_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8], -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L15-L22 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/fmtstr.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/fmtstr.md deleted file mode 100644 index 19809d60261..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/fmtstr.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: fmtstr ---- - -`fmtstr` is the type resulting from using format string (`f"..."`). - -## Methods - -### quoted_contents - -```rust title="quoted_contents" showLineNumbers -pub comptime fn quoted_contents(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/format_string.nr#L3-L5 - - -Returns the format string contents (that is, without the leading and trailing double quotes) as a `Quoted` value. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/is_unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/is_unconstrained.md deleted file mode 100644 index 51bb1bda8f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/is_unconstrained.md +++ /dev/null @@ -1,69 +0,0 @@ ---- -title: Is Unconstrained Function -description: - The is_unconstrained function returns wether the context at that point of the program is unconstrained or not. -keywords: - [ - unconstrained - ] ---- - -It's very common for functions in circuits to take unconstrained hints of an expensive computation and then verify it. This is done by running the hint in an unconstrained context and then verifying the result in a constrained context. - -When a function is marked as unconstrained, any subsequent functions that it calls will also be run in an unconstrained context. However, if we are implementing a library function, other users might call it within an unconstrained context or a constrained one. Generally, in an unconstrained context we prefer just computing the result instead of taking a hint of it and verifying it, since that'd mean doing the same computation twice: - -```rust - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - my_expensive_computation_hint(); - // verify my_expensive_computation: If external_interface is called from unconstrained, this is redundant - ... -} - -``` - -In order to improve the performance in an unconstrained context you can use the function at `std::runtime::is_unconstrained() -> bool`: - - -```rust -use dep::std::runtime::is_unconstrained; - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - if is_unconstrained() { - my_expensive_computation(); - } else { - my_expensive_computation_hint(); - // verify my_expensive_computation - ... - } -} - -``` - -The is_unconstrained result is resolved at compile time, so in unconstrained contexts the compiler removes the else branch, and in constrained contexts the compiler removes the if branch, reducing the amount of compute necessary to run external_interface. - -Note that using `is_unconstrained` in a `comptime` context will also return `true`: - -``` -fn main() { - comptime { - assert(is_unconstrained()); - } -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/logging.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/logging.md deleted file mode 100644 index db75ef9f86f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/logging.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Logging -description: - Learn how to use the println statement for debugging in Noir with this tutorial. Understand the - basics of logging in Noir and how to implement it in your code. -keywords: - [ - noir logging, - println statement, - print statement, - debugging in noir, - noir std library, - logging tutorial, - basic logging in noir, - noir logging implementation, - noir debugging techniques, - rust, - ] ---- - -The standard library provides two familiar statements you can use: `println` and `print`. Despite being a limited implementation of rust's `println!` and `print!` macros, these constructs can be useful for debugging. - -You can print the output of both statements in your Noir code by using the `nargo execute` command or the `--show-output` flag when using `nargo test` (provided there are print statements in your tests). - -It is recommended to use `nargo execute` if you want to debug failing constraints with `println` or `print` statements. This is due to every input in a test being a constant rather than a witness, so we issue an error during compilation while we only print during execution (which comes after compilation). Neither `println`, nor `print` are callable for failed constraints caught at compile time. - -Both `print` and `println` are generic functions which can work on integers, fields, strings, and even structs or expressions. Note however, that slices are currently unsupported. For example: - -```rust -struct Person { - age: Field, - height: Field, -} - -fn main(age: Field, height: Field) { - let person = Person { - age: age, - height: height, - }; - println(person); - println(age + height); - println("Hello world!"); -} -``` - -You can print different types in the same statement (including strings) with a type called `fmtstr`. It can be specified in the same way as a normal string, just prepended with an "f" character: - -```rust - let fmt_str = f"i: {i}, j: {j}"; - println(fmt_str); - - let s = myStruct { y: x, x: y }; - println(s); - - println(f"i: {i}, s: {s}"); - - println(x); - println([x, y]); - - let foo = fooStruct { my_struct: s, foo: 15 }; - println(f"s: {s}, foo: {foo}"); - - println(15); // prints 0x0f, implicit Field - println(-1 as u8); // prints 255 - println(-1 as i8); // prints -1 -``` - -Examples shown above are interchangeable between the two `print` statements: - -```rust -let person = Person { age : age, height : height }; - -println(person); -print(person); - -println("Hello world!"); // Prints with a newline at the end of the input -print("Hello world!"); // Prints the input and keeps cursor on the same line -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/mem.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/mem.md deleted file mode 100644 index 95d36ac2a72..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/mem.md +++ /dev/null @@ -1,52 +0,0 @@ ---- -title: Memory Module -description: - This module contains functions which manipulate memory in a low-level way -keywords: - [ - mem, memory, zeroed, transmute, checked_transmute - ] ---- - -# `std::mem::zeroed` - -```rust -fn zeroed() -> T -``` - -Returns a zeroed value of any type. -This function is generally unsafe to use as the zeroed bit pattern is not guaranteed to be valid for all types. -It can however, be useful in cases when the value is guaranteed not to be used such as in a BoundedVec library implementing a growable vector, up to a certain length, backed by an array. -The array can be initialized with zeroed values which are guaranteed to be inaccessible until the vector is pushed to. -Similarly, enumerations in noir can be implemented using this method by providing zeroed values for the unused variants. - -This function currently supports the following types: - -- Field -- Bool -- Uint -- Array -- Slice -- String -- Tuple -- Functions - -Using it on other types could result in unexpected behavior. - -# `std::mem::checked_transmute` - -```rust -fn checked_transmute(value: T) -> U -``` - -Transmutes a value of one type into the same value but with a new type `U`. - -This function is safe to use since both types are asserted to be equal later during compilation after the concrete values for generic types become known. -This function is useful for cases where the compiler may fails a type check that is expected to pass where -a user knows the two types to be equal. For example, when using arithmetic generics there are cases the compiler -does not see as equal, such as `[Field; N*(A + B)]` and `[Field; N*A + N*B]`, which users may know to be equal. -In these cases, `checked_transmute` can be used to cast the value to the desired type while also preserving safety -by checking this equality once `N`, `A`, `B` are fully resolved. - -Note that since this safety check is performed after type checking rather than during, no error is issued if the function -containing `checked_transmute` is never called. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/merkle_trees.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/merkle_trees.md deleted file mode 100644 index 6a9ebf72ada..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/merkle_trees.md +++ /dev/null @@ -1,58 +0,0 @@ ---- -title: Merkle Trees -description: Learn about Merkle Trees in Noir with this tutorial. Explore the basics of computing a merkle root using a proof, with examples. -keywords: - [ - Merkle trees in Noir, - Noir programming language, - check membership, - computing root from leaf, - Noir Merkle tree implementation, - Merkle tree tutorial, - Merkle tree code examples, - Noir libraries, - pedersen hash., - ] ---- - -## compute_merkle_root - -Returns the root of the tree from the provided leaf and its hash path, using a [Pedersen hash](./cryptographic_primitives/hashes.mdx#pedersen_hash). - -```rust -fn compute_merkle_root(leaf : Field, index : Field, hash_path: [Field]) -> Field -``` - -example: - -```rust -/** - // these values are for this example only - index = "0" - priv_key = "0x000000000000000000000000000000000000000000000000000000616c696365" - secret = "0x1929ea3ab8d9106a899386883d9428f8256cfedb3c4f6b66bf4aa4d28a79988f" - note_hash_path = [ - "0x1e61bdae0f027b1b2159e1f9d3f8d00fa668a952dddd822fda80dc745d6f65cc", - "0x0e4223f3925f98934393c74975142bd73079ab0621f4ee133cee050a3c194f1a", - "0x2fd7bb412155bf8693a3bd2a3e7581a679c95c68a052f835dddca85fa1569a40" - ] - */ -fn main(index: Field, priv_key: Field, secret: Field, note_hash_path: [Field; 3]) { - - let pubkey = std::scalar_mul::fixed_base_embedded_curve(priv_key); - let pubkey_x = pubkey[0]; - let pubkey_y = pubkey[1]; - let note_commitment = std::hash::pedersen(&[pubkey_x, pubkey_y, secret]); - - let root = std::merkle::compute_merkle_root(note_commitment[0], index, note_hash_path.as_slice()); - println(root); -} -``` - -To check merkle tree membership: - -1. Include a merkle root as a program input. -2. Compute the merkle root of a given leaf, index and hash path. -3. Assert the merkle roots are equal. - -For more info about merkle trees, see the Wikipedia [page](https://en.wikipedia.org/wiki/Merkle_tree). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/ctstring.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/ctstring.md deleted file mode 100644 index b76f873ca03..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/ctstring.md +++ /dev/null @@ -1,100 +0,0 @@ ---- -title: CtString ---- - -`std::meta::ctstring` contains methods on the built-in `CtString` type which is -a compile-time, dynamically-sized string type. Compared to `str` and `fmtstr`, -`CtString` is useful because its size does not need to be specified in its type. This -can be used for formatting items at compile-time or general string handling in `comptime` -code. - -Since `fmtstr`s can be converted into `CtString`s, you can make use of their formatting -abilities in CtStrings by formatting in `fmtstr`s then converting the result to a CtString -afterward. - -## Traits - -### AsCtString - -```rust title="as-ctstring" showLineNumbers -pub trait AsCtString { - comptime fn as_ctstring(self) -> CtString; -} -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L43-L47 - - -Converts an object into a compile-time string. - -Implementations: - -```rust -impl AsCtString for str { ... } -impl AsCtString for fmtstr { ... } -``` - -## Methods - -### new - -```rust title="new" showLineNumbers -pub comptime fn new() -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L4-L6 - - -Creates an empty `CtString`. - -### append_str - -```rust title="append_str" showLineNumbers -pub comptime fn append_str(self, s: str) -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L11-L13 - - -Returns a new CtString with the given str appended onto the end. - -### append_fmtstr - -```rust title="append_fmtstr" showLineNumbers -pub comptime fn append_fmtstr(self, s: fmtstr) -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L17-L19 - - -Returns a new CtString with the given fmtstr appended onto the end. - -### as_quoted_str - -```rust title="as_quoted_str" showLineNumbers -pub comptime fn as_quoted_str(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L26-L28 - - -Returns a quoted string literal from this string's contents. - -There is no direct conversion from a `CtString` to a `str` since -the size would not be known. To get around this, this function can -be used in combination with macro insertion (`!`) to insert this string -literal at this function's call site. - -Example: - -```rust title="as_quoted_str_example" showLineNumbers -let my_ctstring = "foo bar".as_ctstring(); - let my_str = my_ctstring.as_quoted_str!(); - - assert_eq(crate::meta::type_of(my_str), quote { str<7> }.as_type()); -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L92-L97 - - -## Trait Implementations - -```rust -impl Eq for CtString -impl Hash for CtString -impl Append for CtString -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/expr.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/expr.md deleted file mode 100644 index b6d395c6700..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/expr.md +++ /dev/null @@ -1,380 +0,0 @@ ---- -title: Expr ---- - -`std::meta::expr` contains methods on the built-in `Expr` type for quoted, syntactically valid expressions. - -## Methods - -### as_array - -```rust title="as_array" showLineNumbers -pub comptime fn as_array(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L10-L12 - - -If this expression is an array, this returns a slice of each element in the array. - -### as_assert - -```rust title="as_assert" showLineNumbers -pub comptime fn as_assert(self) -> Option<(Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L16-L18 - - -If this expression is an assert, this returns the assert expression and the optional message. - -### as_assert_eq - -```rust title="as_assert_eq" showLineNumbers -pub comptime fn as_assert_eq(self) -> Option<(Expr, Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L23-L25 - - -If this expression is an assert_eq, this returns the left-hand-side and right-hand-side -expressions, together with the optional message. - -### as_assign - -```rust title="as_assign" showLineNumbers -pub comptime fn as_assign(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L30-L32 - - -If this expression is an assignment, this returns a tuple with the left hand side -and right hand side in order. - -### as_binary_op - -```rust title="as_binary_op" showLineNumbers -pub comptime fn as_binary_op(self) -> Option<(Expr, BinaryOp, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L37-L39 - - -If this expression is a binary operator operation ` `, -return the left-hand side, operator, and the right-hand side of the operation. - -### as_block - -```rust title="as_block" showLineNumbers -pub comptime fn as_block(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L44-L46 - - -If this expression is a block `{ stmt1; stmt2; ...; stmtN }`, return -a slice containing each statement. - -### as_bool - -```rust title="as_bool" showLineNumbers -pub comptime fn as_bool(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L50-L52 - - -If this expression is a boolean literal, return that literal. - -### as_cast - -```rust title="as_cast" showLineNumbers -#[builtin(expr_as_cast)] - pub comptime fn as_cast(self) -> Option<(Expr, UnresolvedType)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L56-L59 - - -If this expression is a cast expression (`expr as type`), returns the casted -expression and the type to cast to. - -### as_comptime - -```rust title="as_comptime" showLineNumbers -pub comptime fn as_comptime(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L64-L66 - - -If this expression is a `comptime { stmt1; stmt2; ...; stmtN }` block, -return each statement in the block. - -### as_constructor - -```rust title="as_constructor" showLineNumbers -pub comptime fn as_constructor(self) -> Option<(UnresolvedType, [(Quoted, Expr)])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L71-L73 - - -If this expression is a constructor `Type { field1: expr1, ..., fieldN: exprN }`, -return the type and the fields. - -### as_for - -```rust title="as_for" showLineNumbers -pub comptime fn as_for(self) -> Option<(Quoted, Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L78-L80 - - -If this expression is a for statement over a single expression, return the identifier, -the expression and the for loop body. - -### as_for_range - -```rust title="as_for" showLineNumbers -pub comptime fn as_for(self) -> Option<(Quoted, Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L78-L80 - - -If this expression is a for statement over a range, return the identifier, -the range start, the range end and the for loop body. - -### as_function_call - -```rust title="as_function_call" showLineNumbers -pub comptime fn as_function_call(self) -> Option<(Expr, [Expr])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L92-L94 - - -If this expression is a function call `foo(arg1, ..., argN)`, return -the function and a slice of each argument. - -### as_if - -```rust title="as_if" showLineNumbers -pub comptime fn as_if(self) -> Option<(Expr, Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L100-L102 - - -If this expression is an `if condition { then_branch } else { else_branch }`, -return the condition, then branch, and else branch. If there is no else branch, -`None` is returned for that branch instead. - -### as_index - -```rust title="as_index" showLineNumbers -pub comptime fn as_index(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L107-L109 - - -If this expression is an index into an array `array[index]`, return the -array and the index. - -### as_integer - -```rust title="as_integer" showLineNumbers -pub comptime fn as_integer(self) -> Option<(Field, bool)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L114-L116 - - -If this expression is an integer literal, return the integer as a field -as well as whether the integer is negative (true) or not (false). - -### as_lambda - -```rust title="as_lambda" showLineNumbers -pub comptime fn as_lambda( - self, - ) -> Option<([(Expr, Option)], Option, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L120-L124 - - -If this expression is a lambda, returns the parameters, return type and body. - -### as_let - -```rust title="as_let" showLineNumbers -pub comptime fn as_let(self) -> Option<(Expr, Option, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L129-L131 - - -If this expression is a let statement, returns the let pattern as an `Expr`, -the optional type annotation, and the assigned expression. - -### as_member_access - -```rust title="as_member_access" showLineNumbers -pub comptime fn as_member_access(self) -> Option<(Expr, Quoted)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L136-L138 - - -If this expression is a member access `foo.bar`, return the struct/tuple -expression and the field. The field will be represented as a quoted value. - -### as_method_call - -```rust title="as_method_call" showLineNumbers -pub comptime fn as_method_call(self) -> Option<(Expr, Quoted, [UnresolvedType], [Expr])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L143-L145 - - -If this expression is a method call `foo.bar::(arg1, ..., argN)`, return -the receiver, method name, a slice of each generic argument, and a slice of each argument. - -### as_repeated_element_array - -```rust title="as_repeated_element_array" showLineNumbers -pub comptime fn as_repeated_element_array(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L150-L152 - - -If this expression is a repeated element array `[elem; length]`, return -the repeated element and the length expressions. - -### as_repeated_element_slice - -```rust title="as_repeated_element_slice" showLineNumbers -pub comptime fn as_repeated_element_slice(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L157-L159 - - -If this expression is a repeated element slice `[elem; length]`, return -the repeated element and the length expressions. - -### as_slice - -```rust title="as_slice" showLineNumbers -pub comptime fn as_slice(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L164-L166 - - -If this expression is a slice literal `&[elem1, ..., elemN]`, -return each element of the slice. - -### as_tuple - -```rust title="as_tuple" showLineNumbers -pub comptime fn as_tuple(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L171-L173 - - -If this expression is a tuple `(field1, ..., fieldN)`, -return each element of the tuple. - -### as_unary_op - -```rust title="as_unary_op" showLineNumbers -pub comptime fn as_unary_op(self) -> Option<(UnaryOp, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L178-L180 - - -If this expression is a unary operation ` `, -return the unary operator as well as the right-hand side expression. - -### as_unsafe - -```rust title="as_unsafe" showLineNumbers -pub comptime fn as_unsafe(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L185-L187 - - -If this expression is an `unsafe { stmt1; ...; stmtN }` block, -return each statement inside in a slice. - -### has_semicolon - -```rust title="has_semicolon" showLineNumbers -pub comptime fn has_semicolon(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L206-L208 - - -`true` if this expression is trailed by a semicolon. E.g. - -``` -comptime { - let expr1 = quote { 1 + 2 }.as_expr().unwrap(); - let expr2 = quote { 1 + 2; }.as_expr().unwrap(); - - assert(expr1.as_binary_op().is_some()); - assert(expr2.as_binary_op().is_some()); - - assert(!expr1.has_semicolon()); - assert(expr2.has_semicolon()); -} -``` - -### is_break - -```rust title="is_break" showLineNumbers -pub comptime fn is_break(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L212-L214 - - -`true` if this expression is `break`. - -### is_continue - -```rust title="is_continue" showLineNumbers -pub comptime fn is_continue(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L218-L220 - - -`true` if this expression is `continue`. - -### modify - -```rust title="modify" showLineNumbers -pub comptime fn modify(self, f: fn[Env](Expr) -> Option) -> Expr { -``` -> Source code: noir_stdlib/src/meta/expr.nr#L229-L231 - - -Applies a mapping function to this expression and to all of its sub-expressions. -`f` will be applied to each sub-expression first, then applied to the expression itself. - -This happens recursively for every expression within `self`. - -For example, calling `modify` on `(&[1], &[2, 3])` with an `f` that returns `Option::some` -for expressions that are integers, doubling them, would return `(&[2], &[4, 6])`. - -### quoted - -```rust title="quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/expr.nr#L266-L268 - - -Returns this expression as a `Quoted` value. It's the same as `quote { $self }`. - -### resolve - -```rust title="resolve" showLineNumbers -pub comptime fn resolve(self, in_function: Option) -> TypedExpr {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L282-L284 - - -Resolves and type-checks this expression and returns the result as a `TypedExpr`. - -The `in_function` argument specifies where the expression is resolved: -- If it's `none`, the expression is resolved in the function where `resolve` was called -- If it's `some`, the expression is resolved in the given function - -If any names used by this expression are not in scope or if there are any type errors, -this will give compiler errors as if the expression was written directly into -the current `comptime` function. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/function_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/function_def.md deleted file mode 100644 index b7f2ebdb889..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/function_def.md +++ /dev/null @@ -1,166 +0,0 @@ ---- -title: FunctionDefinition ---- - -`std::meta::function_def` contains methods on the built-in `FunctionDefinition` type representing -a function definition in the source program. - -## Methods - -### add_attribute - -```rust title="add_attribute" showLineNumbers -pub comptime fn add_attribute(self, attribute: str) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L3-L5 - - -Adds an attribute to the function. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### body - -```rust title="body" showLineNumbers -pub comptime fn body(self) -> Expr {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L8-L10 - - -Returns the body of the function as an expression. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -pub comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L13-L15 - - -Returns true if this function has a custom attribute with the given name. - -### is_unconstrained - -```rust title="is_unconstrained" showLineNumbers -pub comptime fn is_unconstrained(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L18-L20 - - -Returns true if this function is unconstrained. - -### module - -```rust title="module" showLineNumbers -pub comptime fn module(self) -> Module {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L23-L25 - - -Returns the module where the function is defined. - -### name - -```rust title="name" showLineNumbers -pub comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L28-L30 - - -Returns the name of the function. - -### parameters - -```rust title="parameters" showLineNumbers -pub comptime fn parameters(self) -> [(Quoted, Type)] {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L33-L35 - - -Returns each parameter of the function as a tuple of (parameter pattern, parameter type). - -### return_type - -```rust title="return_type" showLineNumbers -pub comptime fn return_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L38-L40 - - -The return type of the function. - -### set_body - -```rust title="set_body" showLineNumbers -pub comptime fn set_body(self, body: Expr) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L43-L45 - - -Mutate the function body to a new expression. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_parameters - -```rust title="set_parameters" showLineNumbers -pub comptime fn set_parameters(self, parameters: [(Quoted, Type)]) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L48-L50 - - -Mutates the function's parameters to a new set of parameters. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -Expects a slice of (parameter pattern, parameter type) for each parameter. Requires -each parameter pattern to be a syntactically valid parameter. - -### set_return_type - -```rust title="set_return_type" showLineNumbers -pub comptime fn set_return_type(self, return_type: Type) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L53-L55 - - -Mutates the function's return type to a new type. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_return_public - -```rust title="set_return_public" showLineNumbers -pub comptime fn set_return_public(self, public: bool) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L58-L60 - - -Mutates the function's return visibility to public (if `true` is given) or private (if `false` is given). -This is only valid on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_unconstrained - -```rust title="set_unconstrained" showLineNumbers -pub comptime fn set_unconstrained(self, value: bool) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L66-L68 - - -Mutates the function to be unconstrained (if `true` is given) or not (if `false` is given). -This is only valid on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -## Trait Implementations - -```rust -impl Eq for FunctionDefinition -impl Hash for FunctionDefinition -``` - -Note that each function is assigned a unique ID internally and this is what is used for -equality and hashing. So even functions with identical signatures and bodies may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/index.md deleted file mode 100644 index 14544c07442..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/index.md +++ /dev/null @@ -1,224 +0,0 @@ ---- -title: Metaprogramming -description: Noir's Metaprogramming API -keywords: [metaprogramming, comptime, macros, macro, quote, unquote] ---- - -`std::meta` is the entry point for Noir's metaprogramming API. This consists of `comptime` functions -and types used for inspecting and modifying Noir programs. - -## Functions - -### type_of - -```rust title="type_of" showLineNumbers -pub comptime fn type_of(x: T) -> Type {} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L27-L29 - - -Returns the type of a variable at compile-time. - -Example: -```rust -comptime { - let x: i32 = 1; - let x_type: Type = std::meta::type_of(x); - - assert_eq(x_type, quote { i32 }.as_type()); -} -``` - -### unquote - -```rust title="unquote" showLineNumbers -pub comptime fn unquote(code: Quoted) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L19-L21 - - -Unquotes the passed-in token stream where this function was called. - -Example: -```rust -comptime { - let code = quote { 1 + 2 }; - - // let x = 1 + 2; - let x = unquote!(code); -} -``` - -### derive - -```rust title="derive" showLineNumbers -#[varargs] -pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L48-L51 - - -Attribute placed on struct definitions. - -Creates a trait impl for each trait passed in as an argument. -To do this, the trait must have a derive handler registered -with `derive_via` beforehand. The traits in the stdlib that -can be derived this way are `Eq`, `Ord`, `Default`, and `Hash`. - -Example: -```rust -#[derive(Eq, Default)] -struct Foo { - x: i32, - y: T, -} - -fn main() { - let foo1 = Foo::default(); - let foo2 = Foo { x: 0, y: &[0] }; - assert_eq(foo1, foo2); -} -``` - -### derive_via - -```rust title="derive_via_signature" showLineNumbers -pub comptime fn derive_via(t: TraitDefinition, f: DeriveFunction) { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L68-L70 - - -Attribute placed on trait definitions. - -Registers a function to create impls for the given trait -when the trait is used in a `derive` call. Users may use -this to register their own functions to enable their traits -to be derived by `derive`. - -Because this function requires a function as an argument which -should produce a trait impl for any given struct, users may find -it helpful to use a function like `std::meta::make_trait_impl` to -help creating these impls. - -Example: -```rust -#[derive_via(derive_do_nothing)] -trait DoNothing { - fn do_nothing(self); -} - -comptime fn derive_do_nothing(s: StructDefinition) -> Quoted { - let typ = s.as_type(); - quote { - impl DoNothing for $typ { - fn do_nothing(self) { - println("Nothing"); - } - } - } -} -``` - -As another example, `derive_eq` in the stdlib is used to derive the `Eq` -trait for any struct. It makes use of `make_trait_impl` to do this: - -```rust title="derive_eq" showLineNumbers -comptime fn derive_eq(s: StructDefinition) -> Quoted { - let signature = quote { fn eq(_self: Self, _other: Self) -> bool }; - let for_each_field = |name| quote { (_self.$name == _other.$name) }; - let body = |fields| { - if s.fields().len() == 0 { - quote { true } - } else { - fields - } - }; - crate::meta::make_trait_impl( - s, - quote { Eq }, - signature, - for_each_field, - quote { & }, - body, - ) -} -``` -> Source code: noir_stdlib/src/cmp.nr#L10-L30 - - -### make_trait_impl - -```rust title="make_trait_impl" showLineNumbers -pub comptime fn make_trait_impl( - s: StructDefinition, - trait_name: Quoted, - function_signature: Quoted, - for_each_field: fn[Env1](Quoted) -> Quoted, - join_fields_with: Quoted, - body: fn[Env2](Quoted) -> Quoted, -) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L87-L96 - - -A helper function to more easily create trait impls while deriving traits. - -Note that this function only works for traits which: -1. Have only one method -2. Have no generics on the trait itself. - - E.g. Using this on a trait such as `trait Foo { ... }` will result in the - generated impl incorrectly missing the `T` generic. - -If your trait fits these criteria then `make_trait_impl` is likely the easiest -way to write your derive handler. The arguments are as follows: - -- `s`: The struct to make the impl for -- `trait_name`: The name of the trait to derive. E.g. `quote { Eq }`. -- `function_signature`: The signature of the trait method to derive. E.g. `fn eq(self, other: Self) -> bool`. -- `for_each_field`: An operation to be performed on each field. E.g. `|name| quote { (self.$name == other.$name) }`. -- `join_fields_with`: A separator to join each result of `for_each_field` with. - E.g. `quote { & }`. You can also use an empty `quote {}` for no separator. -- `body`: The result of the field operations are passed into this function for any final processing. - This is the place to insert any setup/teardown code the trait requires. If the trait doesn't require - any such code, you can return the body as-is: `|body| body`. - -Example deriving `Hash`: - -```rust title="derive_hash" showLineNumbers -comptime fn derive_hash(s: StructDefinition) -> Quoted { - let name = quote { Hash }; - let signature = quote { fn hash(_self: Self, _state: &mut H) where H: std::hash::Hasher }; - let for_each_field = |name| quote { _self.$name.hash(_state); }; - crate::meta::make_trait_impl( - s, - name, - signature, - for_each_field, - quote {}, - |fields| fields, - ) -} -``` -> Source code: noir_stdlib/src/hash/mod.nr#L137-L151 - - -Example deriving `Ord`: - -```rust title="derive_ord" showLineNumbers -comptime fn derive_ord(s: StructDefinition) -> Quoted { - let signature = quote { fn cmp(_self: Self, _other: Self) -> std::cmp::Ordering }; - let for_each_field = |name| quote { - if result == std::cmp::Ordering::equal() { - result = _self.$name.cmp(_other.$name); - } - }; - let body = |fields| quote { - let mut result = std::cmp::Ordering::equal(); - $fields - result - }; - crate::meta::make_trait_impl(s, quote { Ord }, signature, for_each_field, quote {}, body) -} -``` -> Source code: noir_stdlib/src/cmp.nr#L216-L231 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/module.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/module.md deleted file mode 100644 index f47231972b7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/module.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: Module ---- - -`std::meta::module` contains methods on the built-in `Module` type which represents a module in the source program. -Note that this type represents a module generally, it isn't limited to only `mod my_submodule { ... }` -declarations in the source program. - -## Methods - -### add_item - -```rust title="add_item" showLineNumbers -pub comptime fn add_item(self, item: Quoted) {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L3-L5 - - -Adds a top-level item (a function, a struct, a global, etc.) to the module. -Adding multiple items in one go is also valid if the `Quoted` value has multiple items in it. -Note that the items are type-checked as if they are inside the module they are being added to. - -### functions - -```rust title="functions" showLineNumbers -pub comptime fn functions(self) -> [FunctionDefinition] {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L18-L20 - - -Returns each function defined in the module. - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -pub comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L8-L10 - - -Returns true if this module has a custom attribute with the given name. - -### is_contract - -```rust title="is_contract" showLineNumbers -pub comptime fn is_contract(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L13-L15 - - -`true` if this module is a contract module (was declared via `contract foo { ... }`). - -### name - -```rust title="name" showLineNumbers -pub comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L28-L30 - - -Returns the name of the module. - -### structs - -```rust title="structs" showLineNumbers -pub comptime fn structs(self) -> [StructDefinition] {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L23-L25 - - -Returns each struct defined in the module. - -## Trait Implementations - -```rust -impl Eq for Module -impl Hash for Module -``` - -Note that each module is assigned a unique ID internally and this is what is used for -equality and hashing. So even modules with identical names and contents may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/op.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/op.md deleted file mode 100644 index 03ea49ad8ec..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/op.md +++ /dev/null @@ -1,244 +0,0 @@ ---- -title: UnaryOp and BinaryOp ---- - -`std::meta::op` contains the `UnaryOp` and `BinaryOp` types as well as methods on them. -These types are used to represent a unary or binary operator respectively in Noir source code. - -## Types - -### UnaryOp - -Represents a unary operator. One of `-`, `!`, `&mut`, or `*`. - -### Methods - -#### is_minus - -```rust title="is_minus" showLineNumbers -pub fn is_minus(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L24-L26 - - -Returns `true` if this operator is `-`. - -#### is_not - -```rust title="is_not" showLineNumbers -pub fn is_not(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L30-L32 - - -`true` if this operator is `!` - -#### is_mutable_reference - -```rust title="is_mutable_reference" showLineNumbers -pub fn is_mutable_reference(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L36-L38 - - -`true` if this operator is `&mut` - -#### is_dereference - -```rust title="is_dereference" showLineNumbers -pub fn is_dereference(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L42-L44 - - -`true` if this operator is `*` - -#### quoted - -```rust title="unary_quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/op.nr#L48-L50 - - -Returns this operator as a `Quoted` value. - -### Trait Implementations - -```rust -impl Eq for UnaryOp -impl Hash for UnaryOp -``` - -### BinaryOp - -Represents a binary operator. One of `+`, `-`, `*`, `/`, `%`, `==`, `!=`, `<`, `<=`, `>`, `>=`, `&`, `|`, `^`, `>>`, or `<<`. - -### Methods - -#### is_add - -```rust title="is_add" showLineNumbers -pub fn is_add(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L86-L88 - - -`true` if this operator is `+` - -#### is_subtract - -```rust title="is_subtract" showLineNumbers -pub fn is_subtract(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L92-L94 - - -`true` if this operator is `-` - -#### is_multiply - -```rust title="is_multiply" showLineNumbers -pub fn is_multiply(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L98-L100 - - -`true` if this operator is `*` - -#### is_divide - -```rust title="is_divide" showLineNumbers -pub fn is_divide(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L104-L106 - - -`true` if this operator is `/` - -#### is_modulo - -```rust title="is_modulo" showLineNumbers -pub fn is_modulo(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L176-L178 - - -`true` if this operator is `%` - -#### is_equal - -```rust title="is_equal" showLineNumbers -pub fn is_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L110-L112 - - -`true` if this operator is `==` - -#### is_not_equal - -```rust title="is_not_equal" showLineNumbers -pub fn is_not_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L116-L118 - - -`true` if this operator is `!=` - -#### is_less_than - -```rust title="is_less_than" showLineNumbers -pub fn is_less_than(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L122-L124 - - -`true` if this operator is `<` - -#### is_less_than_or_equal - -```rust title="is_less_than_or_equal" showLineNumbers -pub fn is_less_than_or_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L128-L130 - - -`true` if this operator is `<=` - -#### is_greater_than - -```rust title="is_greater_than" showLineNumbers -pub fn is_greater_than(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L134-L136 - - -`true` if this operator is `>` - -#### is_greater_than_or_equal - -```rust title="is_greater_than_or_equal" showLineNumbers -pub fn is_greater_than_or_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L140-L142 - - -`true` if this operator is `>=` - -#### is_and - -```rust title="is_and" showLineNumbers -pub fn is_and(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L146-L148 - - -`true` if this operator is `&` - -#### is_or - -```rust title="is_or" showLineNumbers -pub fn is_or(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L152-L154 - - -`true` if this operator is `|` - -#### is_shift_right - -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L164-L166 - - -`true` if this operator is `>>` - -#### is_shift_left - -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L164-L166 - - -`true` if this operator is `<<` - -#### quoted - -```rust title="binary_quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/op.nr#L182-L184 - - -Returns this operator as a `Quoted` value. - -### Trait Implementations - -```rust -impl Eq for BinaryOp -impl Hash for BinaryOp -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/quoted.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/quoted.md deleted file mode 100644 index d7acf23bc07..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/quoted.md +++ /dev/null @@ -1,141 +0,0 @@ ---- -title: Quoted ---- - -`std::meta::quoted` contains methods on the built-in `Quoted` type which represents -quoted token streams and is the result of the `quote { ... }` expression. - -## Methods - -### as_expr - -```rust title="as_expr" showLineNumbers -pub comptime fn as_expr(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L6-L8 - - -Parses the quoted token stream as an expression. Returns `Option::none()` if -the expression failed to parse. - -Example: - -```rust title="as_expr_example" showLineNumbers -#[test] - fn test_expr_as_function_call() { - comptime - { - let expr = quote { foo(42) }.as_expr().unwrap(); - let (_function, args) = expr.as_function_call().unwrap(); - assert_eq(args.len(), 1); - assert_eq(args[0].as_integer().unwrap(), (42, false)); - } - } -``` -> Source code: test_programs/noir_test_success/comptime_expr/src/main.nr#L360-L371 - - -### as_module - -```rust title="as_module" showLineNumbers -pub comptime fn as_module(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L11-L13 - - -Interprets this token stream as a module path leading to the name of a module. -Returns `Option::none()` if the module isn't found or this token stream cannot be parsed as a path. - -Example: - -```rust title="as_module_example" showLineNumbers -mod baz { - pub mod qux {} -} - -#[test] -fn as_module_test() { - comptime { - let my_mod = quote { baz::qux }.as_module().unwrap(); - assert_eq(my_mod.name(), quote { qux }); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_module/src/main.nr#L115-L127 - - -### as_trait_constraint - -```rust title="as_trait_constraint" showLineNumbers -pub comptime fn as_trait_constraint(self) -> TraitConstraint {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L16-L18 - - -Interprets this token stream as a trait constraint (without an object type). -Note that this function panics instead of returning `Option::none()` if the token -stream does not parse and resolve to a valid trait constraint. - -Example: - -```rust title="implements_example" showLineNumbers -pub fn function_with_where(_x: T) -where - T: SomeTrait, -{ - comptime { - let t = quote { T }.as_type(); - let some_trait_i32 = quote { SomeTrait }.as_trait_constraint(); - assert(t.implements(some_trait_i32)); - - assert(t.get_trait_impl(some_trait_i32).is_none()); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L160-L173 - - -### as_type - -```rust title="as_type" showLineNumbers -pub comptime fn as_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L21-L23 - - -Interprets this token stream as a resolved type. Panics if the token -stream doesn't parse to a type or if the type isn't a valid type in scope. - -```rust title="implements_example" showLineNumbers -pub fn function_with_where(_x: T) -where - T: SomeTrait, -{ - comptime { - let t = quote { T }.as_type(); - let some_trait_i32 = quote { SomeTrait }.as_trait_constraint(); - assert(t.implements(some_trait_i32)); - - assert(t.get_trait_impl(some_trait_i32).is_none()); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L160-L173 - - -### tokens - -```rust title="tokens" showLineNumbers -pub comptime fn tokens(self) -> [Quoted] {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L26-L28 - - -Returns a slice of the individual tokens that form this token stream. - -## Trait Implementations - -```rust -impl Eq for Quoted -impl Hash for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/struct_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/struct_def.md deleted file mode 100644 index fd609942f4e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/struct_def.md +++ /dev/null @@ -1,177 +0,0 @@ ---- -title: StructDefinition ---- - -`std::meta::struct_def` contains methods on the built-in `StructDefinition` type. -This type corresponds to `struct Name { field1: Type1, ... }` items in the source program. - -## Methods - -### add_attribute - -```rust title="add_attribute" showLineNumbers -pub comptime fn add_attribute(self, attribute: str) {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L3-L5 - - -Adds an attribute to the struct. - -### add_generic - -```rust title="add_generic" showLineNumbers -pub comptime fn add_generic(self, generic_name: str) -> Type {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L8-L10 - - -Adds an generic to the struct. Returns the new generic type. -Errors if the given generic name isn't a single identifier or if -the struct already has a generic with the same name. - -This method should be used carefully, if there is existing code referring -to the struct type it may be checked before this function is called and -see the struct with the original number of generics. This method should -thus be preferred to use on code generated from other macros and structs -that are not used in function signatures. - -Example: - -```rust title="add-generic-example" showLineNumbers -comptime fn add_generic(s: StructDefinition) { - assert_eq(s.generics().len(), 0); - let new_generic = s.add_generic("T"); - - let generics = s.generics(); - assert_eq(generics.len(), 1); - assert_eq(generics[0], new_generic); - } -``` -> Source code: test_programs/compile_success_empty/comptime_struct_definition/src/main.nr#L35-L44 - - -### as_type - -```rust title="as_type" showLineNumbers -pub comptime fn as_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L15-L17 - - -Returns this struct as a type in the source program. If this struct has -any generics, the generics are also included as-is. - -### generics - -```rust title="generics" showLineNumbers -pub comptime fn generics(self) -> [Type] {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L26-L28 - - -Returns each generic on this struct. - -Example: - -``` -#[example] -struct Foo { - bar: [T; 2], - baz: Baz, -} - -comptime fn example(foo: StructDefinition) { - assert_eq(foo.generics().len(), 2); - - // Fails because `T` isn't in scope - // let t = quote { T }.as_type(); - // assert_eq(foo.generics()[0], t); -} -``` - -### fields - -```rust title="fields" showLineNumbers -pub comptime fn fields(self) -> [(Quoted, Type)] {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L33-L35 - - -Returns each field of this struct as a pair of (field name, field type). - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -pub comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L20-L22 - - -Returns true if this struct has a custom attribute with the given name. - -### module - -```rust title="module" showLineNumbers -pub comptime fn module(self) -> Module {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L38-L40 - - -Returns the module where the struct is defined. - -### name - -```rust title="name" showLineNumbers -pub comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L43-L45 - - -Returns the name of this struct - -Note that the returned quoted value will be just the struct name, it will -not be the full path to the struct, nor will it include any generics. - -### set_fields - -```rust title="set_fields" showLineNumbers -pub comptime fn set_fields(self, new_fields: [(Quoted, Type)]) {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L52-L54 - - -Sets the fields of this struct to the given fields list where each element -is a pair of the field's name and the field's type. Expects each field name -to be a single identifier. Note that this will override any previous fields -on this struct. If those should be preserved, use `.fields()` to retrieve the -current fields on the struct type and append the new fields from there. - -Example: - -```rust -// Change this struct to: -// struct Foo { -// a: u32, -// b: i8, -// } -#[mangle_fields] -struct Foo { x: Field } - -comptime fn mangle_fields(s: StructDefinition) { - s.set_fields(&[ - (quote { a }, quote { u32 }.as_type()), - (quote { b }, quote { i8 }.as_type()), - ]); -} -``` - -## Trait Implementations - -```rust -impl Eq for StructDefinition -impl Hash for StructDefinition -``` - -Note that each struct is assigned a unique ID internally and this is what is used for -equality and hashing. So even structs with identical generics and fields may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_constraint.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_constraint.md deleted file mode 100644 index 3106f732b5a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_constraint.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: TraitConstraint ---- - -`std::meta::trait_constraint` contains methods on the built-in `TraitConstraint` type which represents -a trait constraint that can be used to search for a trait implementation. This is similar -syntactically to just the trait itself, but can also contain generic arguments. E.g. `Eq`, `Default`, -`BuildHasher`. - -This type currently has no public methods but it can be used alongside `Type` in `implements` or `get_trait_impl`. - -## Trait Implementations - -```rust -impl Eq for TraitConstraint -impl Hash for TraitConstraint -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_def.md deleted file mode 100644 index e661d3af7f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_def.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: TraitDefinition ---- - -`std::meta::trait_def` contains methods on the built-in `TraitDefinition` type. This type -represents trait definitions such as `trait Foo { .. }` at the top-level of a program. - -## Methods - -### as_trait_constraint - -```rust title="as_trait_constraint" showLineNumbers -pub comptime fn as_trait_constraint(_self: Self) -> TraitConstraint {} -``` -> Source code: noir_stdlib/src/meta/trait_def.nr#L6-L8 - - -Converts this trait into a trait constraint. If there are any generics on this -trait, they will be kept as-is without instantiating or replacing them. - -## Trait Implementations - -```rust -impl Eq for TraitDefinition -impl Hash for TraitDefinition -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_impl.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_impl.md deleted file mode 100644 index a527617c1e6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/trait_impl.md +++ /dev/null @@ -1,60 +0,0 @@ ---- -title: TraitImpl ---- - -`std::meta::trait_impl` contains methods on the built-in `TraitImpl` type which represents a trait -implementation such as `impl Foo for Bar { ... }`. - -## Methods - -### trait_generic_args - -```rust title="trait_generic_args" showLineNumbers -pub comptime fn trait_generic_args(self) -> [Type] {} -``` -> Source code: noir_stdlib/src/meta/trait_impl.nr#L3-L5 - - -Returns any generic arguments on the trait of this trait implementation, if any. - -```rs -impl Foo for Bar { ... } - -comptime { - let bar_type = quote { Bar }.as_type(); - let foo = quote { Foo }.as_trait_constraint(); - - let my_impl: TraitImpl = bar_type.get_trait_impl(foo).unwrap(); - - let generics = my_impl.trait_generic_args(); - assert_eq(generics.len(), 2); - - assert_eq(generics[0], quote { i32 }.as_type()); - assert_eq(generics[1], quote { Field }.as_type()); -} -``` - -### methods - -```rust title="methods" showLineNumbers -pub comptime fn methods(self) -> [FunctionDefinition] {} -``` -> Source code: noir_stdlib/src/meta/trait_impl.nr#L8-L10 - - -Returns each method in this trait impl. - -Example: - -```rs -comptime { - let i32_type = quote { i32 }.as_type(); - let eq = quote { Eq }.as_trait_constraint(); - - let impl_eq_for_i32: TraitImpl = i32_type.get_trait_impl(eq).unwrap(); - let methods = impl_eq_for_i32.methods(); - - assert_eq(methods.len(), 1); - assert_eq(methods[0].name(), quote { eq }); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/typ.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/typ.md deleted file mode 100644 index 90222c222f5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/typ.md +++ /dev/null @@ -1,264 +0,0 @@ ---- -title: Type ---- - -`std::meta::typ` contains methods on the built-in `Type` type used for representing -a type in the source program. - -## Functions - -```rust title="fresh_type_variable" showLineNumbers -pub comptime fn fresh_type_variable() -> Type {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L57-L59 - - -Creates and returns an unbound type variable. This is a special kind of type internal -to type checking which will type check with any other type. When it is type checked -against another type it will also be set to that type. For example, if `a` is a type -variable and we have the type equality `(a, i32) = (u8, i32)`, the compiler will set -`a` equal to `u8`. - -Unbound type variables will often be rendered as `_` while printing them. Bound type -variables will appear as the type they are bound to. - -This can be used in conjunction with functions which internally perform type checks -such as `Type::implements` or `Type::get_trait_impl` to potentially grab some of the types used. - -Note that calling `Type::implements` or `Type::get_trait_impl` on a type variable will always -fail. - -Example: - -```rust title="serialize-setup" showLineNumbers -trait Serialize {} - -impl Serialize<1> for Field {} - -impl Serialize for [T; N] -where - T: Serialize, -{} - -impl Serialize for (T, U) -where - T: Serialize, - U: Serialize, -{} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L14-L29 - -```rust title="fresh-type-variable-example" showLineNumbers -let typevar1 = std::meta::typ::fresh_type_variable(); - let constraint = quote { Serialize<$typevar1> }.as_trait_constraint(); - let field_type = quote { Field }.as_type(); - - // Search for a trait impl (binding typevar1 to 1 when the impl is found): - assert(field_type.implements(constraint)); - - // typevar1 should be bound to the "1" generic now: - assert_eq(typevar1.as_constant().unwrap(), 1); - - // If we want to do the same with a different type, we need to - // create a new type variable now that `typevar1` is bound - let typevar2 = std::meta::typ::fresh_type_variable(); - let constraint = quote { Serialize<$typevar2> }.as_trait_constraint(); - let array_type = quote { [(Field, Field); 5] }.as_type(); - assert(array_type.implements(constraint)); - - // Now typevar2 should be bound to the serialized pair size 2 times the array length 5 - assert_eq(typevar2.as_constant().unwrap(), 10); -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L129-L149 - - -## Methods - -### as_array - -```rust title="as_array" showLineNumbers -pub comptime fn as_array(self) -> Option<(Type, Type)> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L76-L78 - - -If this type is an array, return a pair of (element type, size type). - -Example: - -```rust -comptime { - let array_type = quote { [Field; 3] }.as_type(); - let (field_type, three_type) = array_type.as_array().unwrap(); - - assert(field_type.is_field()); - assert_eq(three_type.as_constant().unwrap(), 3); -} -``` - -### as_constant - -```rust title="as_constant" showLineNumbers -pub comptime fn as_constant(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L83-L85 - - -If this type is a constant integer (such as the `3` in the array type `[Field; 3]`), -return the numeric constant. - -### as_integer - -```rust title="as_integer" showLineNumbers -pub comptime fn as_integer(self) -> Option<(bool, u8)> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L90-L92 - - -If this is an integer type, return a boolean which is `true` -if the type is signed, as well as the number of bits of this integer type. - -### as_mutable_reference - -```rust title="as_mutable_reference" showLineNumbers -comptime fn as_mutable_reference(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L96-L98 - - -If this is a mutable reference type `&mut T`, returns the mutable type `T`. - -### as_slice - -```rust title="as_slice" showLineNumbers -pub comptime fn as_slice(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L102-L104 - - -If this is a slice type, return the element type of the slice. - -### as_str - -```rust title="as_str" showLineNumbers -pub comptime fn as_str(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L108-L110 - - -If this is a `str` type, returns the length `N` as a type. - -### as_struct - -```rust title="as_struct" showLineNumbers -pub comptime fn as_struct(self) -> Option<(StructDefinition, [Type])> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L114-L116 - - -If this is a struct type, returns the struct in addition to -any generic arguments on this type. - -### as_tuple - -```rust title="as_tuple" showLineNumbers -pub comptime fn as_tuple(self) -> Option<[Type]> {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L120-L122 - - -If this is a tuple type, returns each element type of the tuple. - -### get_trait_impl - -```rust title="get_trait_impl" showLineNumbers -pub comptime fn get_trait_impl(self, constraint: TraitConstraint) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L143-L145 - - -Retrieves the trait implementation that implements the given -trait constraint for this type. If the trait constraint is not -found, `None` is returned. Note that since the concrete trait implementation -for a trait constraint specified from a `where` clause is unknown, -this function will return `None` in these cases. If you only want to know -whether a type implements a trait, use `implements` instead. - -Example: - -```rust -comptime { - let field_type = quote { Field }.as_type(); - let default = quote { Default }.as_trait_constraint(); - - let the_impl: TraitImpl = field_type.get_trait_impl(default).unwrap(); - assert(the_impl.methods().len(), 1); -} -``` - -### implements - -```rust title="implements" showLineNumbers -pub comptime fn implements(self, constraint: TraitConstraint) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L166-L168 - - -`true` if this type implements the given trait. Note that unlike -`get_trait_impl` this will also return true for any `where` constraints -in scope. - -Example: - -```rust -fn foo() where T: Default { - comptime { - let field_type = quote { Field }.as_type(); - let default = quote { Default }.as_trait_constraint(); - assert(field_type.implements(default)); - - let t = quote { T }.as_type(); - assert(t.implements(default)); - } -} -``` - -### is_bool - -```rust title="is_bool" showLineNumbers -pub comptime fn is_bool(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L172-L174 - - -`true` if this type is `bool`. - -### is_field - -```rust title="is_field" showLineNumbers -pub comptime fn is_field(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L178-L180 - - -`true` if this type is `Field`. - -### is_unit - -```rust title="is_unit" showLineNumbers -comptime fn is_unit(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/typ.nr#L184-L186 - - -`true` if this type is the unit `()` type. - -## Trait Implementations - -```rust -impl Eq for Type -impl Hash for Type -``` -Note that this is syntactic equality, this is not the same as whether two types will type check -to be the same type. Unless type inference or generics are being used however, users should not -typically have to worry about this distinction unless `std::meta::typ::fresh_type_variable` is used. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/typed_expr.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/typed_expr.md deleted file mode 100644 index 0db7dbfef61..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/typed_expr.md +++ /dev/null @@ -1,27 +0,0 @@ ---- -title: TypedExpr ---- - -`std::meta::typed_expr` contains methods on the built-in `TypedExpr` type for resolved and type-checked expressions. - -## Methods - -### get_type - -```rust title="as_function_definition" showLineNumbers -pub comptime fn as_function_definition(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typed_expr.nr#L7-L9 - - -If this expression refers to a function definitions, returns it. Otherwise returns `Option::none()`. - -### get_type - -```rust title="get_type" showLineNumbers -pub comptime fn get_type(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typed_expr.nr#L13-L15 - - -Returns the type of the expression, or `Option::none()` if there were errors when the expression was previously resolved. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/unresolved_type.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/unresolved_type.md deleted file mode 100644 index 2826ec5ec0f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/meta/unresolved_type.md +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: UnresolvedType ---- - -`std::meta::unresolved_type` contains methods on the built-in `UnresolvedType` type for the syntax of types. - -## Methods - -### as_mutable_reference - -```rust title="as_mutable_reference" showLineNumbers -comptime fn as_mutable_reference(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L8-L10 - - -If this is a mutable reference type `&mut T`, returns the mutable type `T`. - -### as_slice - -```rust title="as_slice" showLineNumbers -comptime fn as_slice(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L14-L16 - - -If this is a slice `&[T]`, returns the element type `T`. - -### is_bool - -```rust title="is_bool" showLineNumbers -comptime fn is_bool(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L20-L22 - - -Returns `true` if this type is `bool`. - -### is_field - -```rust title="is_field" showLineNumbers -pub comptime fn is_field(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L26-L28 - - -Returns true if this type refers to the Field type. - -### is_unit - -```rust title="is_unit" showLineNumbers -comptime fn is_unit(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L32-L34 - - -Returns true if this type is the unit `()` type. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/options.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/options.md deleted file mode 100644 index a1bd4e1de5f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/options.md +++ /dev/null @@ -1,101 +0,0 @@ ---- -title: Option Type ---- - -The `Option` type is a way to express that a value might be present (`Some(T))` or absent (`None`). It's a safer way to handle potential absence of values, compared to using nulls in many other languages. - -```rust -struct Option { - None, - Some(T), -} -``` - -The `Option` type, already imported into your Noir program, can be used directly: - -```rust -fn main() { - let none = Option::none(); - let some = Option::some(3); -} -``` - -See [this test](https://github.com/noir-lang/noir/blob/5cbfb9c4a06c8865c98ff2b594464b037d821a5c/crates/nargo_cli/tests/test_data/option/src/main.nr) for a more comprehensive set of examples of each of the methods described below. - -## Methods - -### none - -Constructs a none value. - -### some - -Constructs a some wrapper around a given value. - -### is_none - -Returns true if the Option is None. - -### is_some - -Returns true of the Option is Some. - -### unwrap - -Asserts `self.is_some()` and returns the wrapped value. - -### unwrap_unchecked - -Returns the inner value without asserting `self.is_some()`. This method can be useful within an if condition when we already know that `option.is_some()`. If the option is None, there is no guarantee what value will be returned, only that it will be of type T for an `Option`. - -### unwrap_or - -Returns the wrapped value if `self.is_some()`. Otherwise, returns the given default value. - -### unwrap_or_else - -Returns the wrapped value if `self.is_some()`. Otherwise, calls the given function to return a default value. - -### expect - -Asserts `self.is_some()` with a provided custom message and returns the contained `Some` value. The custom message is expected to be a format string. - -### map - -If self is `Some(x)`, this returns `Some(f(x))`. Otherwise, this returns `None`. - -### map_or - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns the given default value. - -### map_or_else - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns `default()`. - -### and - -Returns None if self is None. Otherwise, this returns `other`. - -### and_then - -If self is None, this returns None. Otherwise, this calls the given function with the Some value contained within self, and returns the result of that call. In some languages this function is called `flat_map` or `bind`. - -### or - -If self is Some, return self. Otherwise, return `other`. - -### or_else - -If self is Some, return self. Otherwise, return `default()`. - -### xor - -If only one of the two Options is Some, return that option. Otherwise, if both options are Some or both are None, None is returned. - -### filter - -Returns `Some(x)` if self is `Some(x)` and `predicate(x)` is true. Otherwise, this returns `None`. - -### flatten - -Flattens an `Option>` into a `Option`. This returns `None` if the outer Option is None. Otherwise, this returns the inner Option. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/recursion.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/recursion.mdx deleted file mode 100644 index 60414a2fa51..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/recursion.mdx +++ /dev/null @@ -1,85 +0,0 @@ ---- -title: Recursive Proofs -description: Learn about how to write recursive proofs in Noir. -keywords: [recursion, recursive proofs, verification_key, verify_proof] ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports recursively verifying proofs, meaning you verify the proof of a Noir program in another Noir program. This enables creating proofs of arbitrary size by doing step-wise verification of smaller components of a large proof. - -Read [the explainer on recursion](../../explainers/explainer-recursion.md) to know more about this function and the [guide on how to use it.](../../how_to/how-to-recursion.md) - -## The `#[recursive]` Attribute - -In Noir, the `#[recursive]` attribute is used to indicate that a circuit is designed for recursive proof generation. When applied, it informs the compiler and the tooling that the circuit should be compiled in a way that makes its proofs suitable for recursive verification. This attribute eliminates the need for manual flagging of recursion at the tooling level, streamlining the proof generation process for recursive circuits. - -### Example usage with `#[recursive]` - -```rust -#[recursive] -fn main(x: Field, y: pub Field) { - assert(x == y, "x and y are not equal"); -} - -// This marks the circuit as recursion-friendly and indicates that proofs generated from this circuit -// are intended for recursive verification. -``` - -By incorporating this attribute directly in the circuit's definition, tooling like Nargo and NoirJS can automatically execute recursive-specific duties for Noir programs (e.g. recursive-friendly proof artifact generation) without additional flags or configurations. - -## Verifying Recursive Proofs - -```rust -#[foreign(recursive_aggregation)] -pub fn verify_proof(verification_key: [Field], proof: [Field], public_inputs: [Field], key_hash: Field) {} -``` - - - -## Example usage - -```rust - -fn main( - verification_key : [Field; 114], - proof : [Field; 93], - public_inputs : [Field; 1], - key_hash : Field, - proof_b : [Field; 93], -) { - std::verify_proof( - verification_key, - proof, - public_inputs, - key_hash - ); - - std::verify_proof( - verification_key, - proof_b, - public_inputs, - key_hash - ); -} -``` - -You can see a full example of recursive proofs in [this example recursion demo repo](https://github.com/noir-lang/noir-examples/tree/master/recursion). - -## Parameters - -### `verification_key` - -The verification key for the zk program that is being verified. - -### `proof` - -The proof for the zk program that is being verified. - -### `public_inputs` - -These represent the public inputs of the proof we are verifying. - -### `key_hash` - -A key hash is used to check the validity of the verification key. The circuit implementing this opcode can use this hash to ensure that the key provided to the circuit matches the key produced by the circuit creator. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/traits.md deleted file mode 100644 index ee20f9cd949..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/standard_library/traits.md +++ /dev/null @@ -1,628 +0,0 @@ ---- -title: Traits -description: Noir's stdlib provides a few commonly used traits. -keywords: [traits, trait, interface, protocol, default, add, eq] ---- - -## `std::default` - -### `std::default::Default` - -```rust title="default-trait" showLineNumbers -pub trait Default { - fn default() -> Self; -} -``` -> Source code: noir_stdlib/src/default.nr#L4-L8 - - -Constructs a default value of a type. - -Implementations: -```rust -impl Default for Field { .. } - -impl Default for i8 { .. } -impl Default for i16 { .. } -impl Default for i32 { .. } -impl Default for i64 { .. } - -impl Default for u8 { .. } -impl Default for u16 { .. } -impl Default for u32 { .. } -impl Default for u64 { .. } - -impl Default for () { .. } -impl Default for bool { .. } - -impl Default for [T; N] - where T: Default { .. } - -impl Default for [T] { .. } - -impl Default for (A, B) - where A: Default, B: Default { .. } - -impl Default for (A, B, C) - where A: Default, B: Default, C: Default { .. } - -impl Default for (A, B, C, D) - where A: Default, B: Default, C: Default, D: Default { .. } - -impl Default for (A, B, C, D, E) - where A: Default, B: Default, C: Default, D: Default, E: Default { .. } -``` - -For primitive integer types, the return value of `default` is `0`. Container -types such as arrays are filled with default values of their element type, -except slices whose length is unknown and thus defaulted to zero. - ---- - -## `std::convert` - -### `std::convert::From` - -```rust title="from-trait" showLineNumbers -pub trait From { - fn from(input: T) -> Self; -} -``` -> Source code: noir_stdlib/src/convert.nr#L1-L5 - - -The `From` trait defines how to convert from a given type `T` to the type on which the trait is implemented. - -The Noir standard library provides a number of implementations of `From` between primitive types. -```rust title="from-impls" showLineNumbers -// Unsigned integers - -impl From for u32 { - fn from(value: u8) -> u32 { - value as u32 - } -} - -impl From for u64 { - fn from(value: u8) -> u64 { - value as u64 - } -} -impl From for u64 { - fn from(value: u32) -> u64 { - value as u64 - } -} - -impl From for Field { - fn from(value: u8) -> Field { - value as Field - } -} -impl From for Field { - fn from(value: u32) -> Field { - value as Field - } -} -impl From for Field { - fn from(value: u64) -> Field { - value as Field - } -} - -// Signed integers - -impl From for i32 { - fn from(value: i8) -> i32 { - value as i32 - } -} - -impl From for i64 { - fn from(value: i8) -> i64 { - value as i64 - } -} -impl From for i64 { - fn from(value: i32) -> i64 { - value as i64 - } -} - -// Booleans -impl From for u8 { - fn from(value: bool) -> u8 { - value as u8 - } -} -impl From for u32 { - fn from(value: bool) -> u32 { - value as u32 - } -} -impl From for u64 { - fn from(value: bool) -> u64 { - value as u64 - } -} -impl From for i8 { - fn from(value: bool) -> i8 { - value as i8 - } -} -impl From for i32 { - fn from(value: bool) -> i32 { - value as i32 - } -} -impl From for i64 { - fn from(value: bool) -> i64 { - value as i64 - } -} -impl From for Field { - fn from(value: bool) -> Field { - value as Field - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L28-L119 - - -#### When to implement `From` - -As a general rule of thumb, `From` may be implemented in the [situations where it would be suitable in Rust](https://doc.rust-lang.org/std/convert/trait.From.html#when-to-implement-from): - -- The conversion is *infallible*: Noir does not provide an equivalent to Rust's `TryFrom`, if the conversion can fail then provide a named method instead. -- The conversion is *lossless*: semantically, it should not lose or discard information. For example, `u32: From` can losslessly convert any `u16` into a valid `u32` such that the original `u16` can be recovered. On the other hand, `u16: From` should not be implemented as `2**16` is a `u32` which cannot be losslessly converted into a `u16`. -- The conversion is *value-preserving*: the conceptual kind and meaning of the resulting value is the same, even though the Noir type and technical representation might be different. While it's possible to infallibly and losslessly convert a `u8` into a `str<2>` hex representation, `4u8` and `"04"` are too different for `str<2>: From` to be implemented. -- The conversion is *obvious*: it's the only reasonable conversion between the two types. If there's ambiguity on how to convert between them such that the same input could potentially map to two different values then a named method should be used. For instance rather than implementing `U128: From<[u8; 16]>`, the methods `U128::from_le_bytes` and `U128::from_be_bytes` are used as otherwise the endianness of the array would be ambiguous, resulting in two potential values of `U128` from the same byte array. - -One additional recommendation specific to Noir is: -- The conversion is *efficient*: it's relatively cheap to convert between the two types. Due to being a ZK DSL, it's more important to avoid unnecessary computation compared to Rust. If the implementation of `From` would encourage users to perform unnecessary conversion, resulting in additional proving time, then it may be preferable to expose functionality such that this conversion may be avoided. - -### `std::convert::Into` - -The `Into` trait is defined as the reciprocal of `From`. It should be easy to convince yourself that if we can convert to type `A` from type `B`, then it's possible to convert type `B` into type `A`. - -For this reason, implementing `From` on a type will automatically generate a matching `Into` implementation. One should always prefer implementing `From` over `Into` as implementing `Into` will not generate a matching `From` implementation. - -```rust title="into-trait" showLineNumbers -pub trait Into { - fn into(self) -> T; -} - -impl Into for U -where - T: From, -{ - fn into(self) -> T { - T::from(self) - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L13-L26 - - -`Into` is most useful when passing function arguments where the types don't quite match up with what the function expects. In this case, the compiler has enough type information to perform the necessary conversion by just appending `.into()` onto the arguments in question. - ---- - -## `std::cmp` - -### `std::cmp::Eq` - -```rust title="eq-trait" showLineNumbers -pub trait Eq { - fn eq(self, other: Self) -> bool; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L4-L8 - - -Returns `true` if `self` is equal to `other`. Implementing this trait on a type -allows the type to be used with `==` and `!=`. - -Implementations: -```rust -impl Eq for Field { .. } - -impl Eq for i8 { .. } -impl Eq for i16 { .. } -impl Eq for i32 { .. } -impl Eq for i64 { .. } - -impl Eq for u8 { .. } -impl Eq for u16 { .. } -impl Eq for u32 { .. } -impl Eq for u64 { .. } - -impl Eq for () { .. } -impl Eq for bool { .. } - -impl Eq for [T; N] - where T: Eq { .. } - -impl Eq for [T] - where T: Eq { .. } - -impl Eq for (A, B) - where A: Eq, B: Eq { .. } - -impl Eq for (A, B, C) - where A: Eq, B: Eq, C: Eq { .. } - -impl Eq for (A, B, C, D) - where A: Eq, B: Eq, C: Eq, D: Eq { .. } - -impl Eq for (A, B, C, D, E) - where A: Eq, B: Eq, C: Eq, D: Eq, E: Eq { .. } -``` - -### `std::cmp::Ord` - -```rust title="ord-trait" showLineNumbers -pub trait Ord { - fn cmp(self, other: Self) -> Ordering; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L210-L214 - - -`a.cmp(b)` compares two values returning `Ordering::less()` if `a < b`, -`Ordering::equal()` if `a == b`, or `Ordering::greater()` if `a > b`. -Implementing this trait on a type allows `<`, `<=`, `>`, and `>=` to be -used on values of the type. - -`std::cmp` also provides `max` and `min` functions for any type which implements the `Ord` trait. - -Implementations: - -```rust -impl Ord for u8 { .. } -impl Ord for u16 { .. } -impl Ord for u32 { .. } -impl Ord for u64 { .. } - -impl Ord for i8 { .. } -impl Ord for i16 { .. } -impl Ord for i32 { .. } - -impl Ord for i64 { .. } - -impl Ord for () { .. } -impl Ord for bool { .. } - -impl Ord for [T; N] - where T: Ord { .. } - -impl Ord for [T] - where T: Ord { .. } - -impl Ord for (A, B) - where A: Ord, B: Ord { .. } - -impl Ord for (A, B, C) - where A: Ord, B: Ord, C: Ord { .. } - -impl Ord for (A, B, C, D) - where A: Ord, B: Ord, C: Ord, D: Ord { .. } - -impl Ord for (A, B, C, D, E) - where A: Ord, B: Ord, C: Ord, D: Ord, E: Ord { .. } -``` - ---- - -## `std::ops` - -### `std::ops::Add`, `std::ops::Sub`, `std::ops::Mul`, and `std::ops::Div` - -These traits abstract over addition, subtraction, multiplication, and division respectively. -Implementing these traits for a given type will also allow that type to be used with the corresponding operator -for that trait (`+` for Add, etc) in addition to the normal method names. - -```rust title="add-trait" showLineNumbers -pub trait Add { - fn add(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L1-L5 - -```rust title="sub-trait" showLineNumbers -pub trait Sub { - fn sub(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L60-L64 - -```rust title="mul-trait" showLineNumbers -pub trait Mul { - fn mul(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L119-L123 - -```rust title="div-trait" showLineNumbers -pub trait Div { - fn div(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L178-L182 - - -The implementations block below is given for the `Add` trait, but the same types that implement -`Add` also implement `Sub`, `Mul`, and `Div`. - -Implementations: -```rust -impl Add for Field { .. } - -impl Add for i8 { .. } -impl Add for i16 { .. } -impl Add for i32 { .. } -impl Add for i64 { .. } - -impl Add for u8 { .. } -impl Add for u16 { .. } -impl Add for u32 { .. } -impl Add for u64 { .. } -``` - -### `std::ops::Rem` - -```rust title="rem-trait" showLineNumbers -pub trait Rem { - fn rem(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L237-L241 - - -`Rem::rem(a, b)` is the remainder function returning the result of what is -left after dividing `a` and `b`. Implementing `Rem` allows the `%` operator -to be used with the implementation type. - -Unlike other numeric traits, `Rem` is not implemented for `Field`. - -Implementations: -```rust -impl Rem for u8 { fn rem(self, other: u8) -> u8 { self % other } } -impl Rem for u16 { fn rem(self, other: u16) -> u16 { self % other } } -impl Rem for u32 { fn rem(self, other: u32) -> u32 { self % other } } -impl Rem for u64 { fn rem(self, other: u64) -> u64 { self % other } } - -impl Rem for i8 { fn rem(self, other: i8) -> i8 { self % other } } -impl Rem for i16 { fn rem(self, other: i16) -> i16 { self % other } } -impl Rem for i32 { fn rem(self, other: i32) -> i32 { self % other } } -impl Rem for i64 { fn rem(self, other: i64) -> i64 { self % other } } -``` - -### `std::ops::Neg` - -```rust title="neg-trait" showLineNumbers -pub trait Neg { - fn neg(self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L290-L294 - - -`Neg::neg` is equivalent to the unary negation operator `-`. - -Implementations: -```rust title="neg-trait-impls" showLineNumbers -impl Neg for Field { - fn neg(self) -> Field { - -self - } -} - -impl Neg for i8 { - fn neg(self) -> i8 { - -self - } -} -impl Neg for i16 { - fn neg(self) -> i16 { - -self - } -} -impl Neg for i32 { - fn neg(self) -> i32 { - -self - } -} -impl Neg for i64 { - fn neg(self) -> i64 { - -self - } -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L296-L323 - - -### `std::ops::Not` - -```rust title="not-trait" showLineNumbers -pub trait Not { - fn not(self: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L1-L5 - - -`Not::not` is equivalent to the unary bitwise NOT operator `!`. - -Implementations: -```rust title="not-trait-impls" showLineNumbers -impl Not for bool { - fn not(self) -> bool { - !self - } -} - -impl Not for u64 { - fn not(self) -> u64 { - !self - } -} -impl Not for u32 { - fn not(self) -> u32 { - !self - } -} -impl Not for u16 { - fn not(self) -> u16 { - !self - } -} -impl Not for u8 { - fn not(self) -> u8 { - !self - } -} -impl Not for u1 { - fn not(self) -> u1 { - !self - } -} - -impl Not for i8 { - fn not(self) -> i8 { - !self - } -} -impl Not for i16 { - fn not(self) -> i16 { - !self - } -} -impl Not for i32 { - fn not(self) -> i32 { - !self - } -} -impl Not for i64 { - fn not(self) -> i64 { - !self - } -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L7-L60 - - -### `std::ops::{ BitOr, BitAnd, BitXor }` - -```rust title="bitor-trait" showLineNumbers -pub trait BitOr { - fn bitor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L62-L66 - -```rust title="bitand-trait" showLineNumbers -pub trait BitAnd { - fn bitand(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L121-L125 - -```rust title="bitxor-trait" showLineNumbers -pub trait BitXor { - fn bitxor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L180-L184 - - -Traits for the bitwise operations `|`, `&`, and `^`. - -Implementing `BitOr`, `BitAnd` or `BitXor` for a type allows the `|`, `&`, or `^` operator respectively -to be used with the type. - -The implementations block below is given for the `BitOr` trait, but the same types that implement -`BitOr` also implement `BitAnd` and `BitXor`. - -Implementations: -```rust -impl BitOr for bool { fn bitor(self, other: bool) -> bool { self | other } } - -impl BitOr for u8 { fn bitor(self, other: u8) -> u8 { self | other } } -impl BitOr for u16 { fn bitor(self, other: u16) -> u16 { self | other } } -impl BitOr for u32 { fn bitor(self, other: u32) -> u32 { self | other } } -impl BitOr for u64 { fn bitor(self, other: u64) -> u64 { self | other } } - -impl BitOr for i8 { fn bitor(self, other: i8) -> i8 { self | other } } -impl BitOr for i16 { fn bitor(self, other: i16) -> i16 { self | other } } -impl BitOr for i32 { fn bitor(self, other: i32) -> i32 { self | other } } -impl BitOr for i64 { fn bitor(self, other: i64) -> i64 { self | other } } -``` - -### `std::ops::{ Shl, Shr }` - -```rust title="shl-trait" showLineNumbers -pub trait Shl { - fn shl(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L239-L243 - -```rust title="shr-trait" showLineNumbers -pub trait Shr { - fn shr(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L292-L296 - - -Traits for a bit shift left and bit shift right. - -Implementing `Shl` for a type allows the left shift operator (`<<`) to be used with the implementation type. -Similarly, implementing `Shr` allows the right shift operator (`>>`) to be used with the type. - -Note that bit shifting is not currently implemented for signed types. - -The implementations block below is given for the `Shl` trait, but the same types that implement -`Shl` also implement `Shr`. - -Implementations: -```rust -impl Shl for u8 { fn shl(self, other: u8) -> u8 { self << other } } -impl Shl for u16 { fn shl(self, other: u16) -> u16 { self << other } } -impl Shl for u32 { fn shl(self, other: u32) -> u32 { self << other } } -impl Shl for u64 { fn shl(self, other: u64) -> u64 { self << other } } -``` - ---- - -## `std::append` - -### `std::append::Append` - -`Append` can abstract over types that can be appended to - usually container types: - -```rust title="append-trait" showLineNumbers -pub trait Append { - fn empty() -> Self; - fn append(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/append.nr#L9-L14 - - -`Append` requires two methods: - -- `empty`: Constructs an empty value of `Self`. -- `append`: Append two values together, returning the result. - -Additionally, it is expected that for any implementation: - -- `T::empty().append(x) == x` -- `x.append(T::empty()) == x` - -Implementations: -```rust -impl Append for [T] -impl Append for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/classes/Noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/classes/Noir.md deleted file mode 100644 index ead255bc504..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/classes/Noir.md +++ /dev/null @@ -1,52 +0,0 @@ -# Noir - -## Constructors - -### new Noir(circuit) - -```ts -new Noir(circuit): Noir -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `circuit` | `CompiledCircuit` | - -#### Returns - -[`Noir`](Noir.md) - -## Methods - -### execute() - -```ts -execute(inputs, foreignCallHandler?): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `inputs` | `InputMap` | -| `foreignCallHandler`? | [`ForeignCallHandler`](../type-aliases/ForeignCallHandler.md) | - -#### Returns - -`Promise`\<`object`\> - -#### Description - -Allows to execute a circuit to get its witness and return value. - -#### Example - -```typescript -async execute(inputs) -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/and.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/and.md deleted file mode 100644 index c783283e396..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/and.md +++ /dev/null @@ -1,22 +0,0 @@ -# and() - -```ts -and(lhs, rhs): string -``` - -Performs a bitwise AND operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/blake2s256.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/blake2s256.md deleted file mode 100644 index 7882d0da8d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/blake2s256.md +++ /dev/null @@ -1,21 +0,0 @@ -# blake2s256() - -```ts -blake2s256(inputs): Uint8Array -``` - -Calculates the Blake2s256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md deleted file mode 100644 index 5e3cd53e9d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256k1\_verify() - -```ts -ecdsa_secp256k1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256k1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md deleted file mode 100644 index 0b20ff68957..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256r1\_verify() - -```ts -ecdsa_secp256r1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256r1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/xor.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/xor.md deleted file mode 100644 index 8d762b895d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/functions/xor.md +++ /dev/null @@ -1,22 +0,0 @@ -# xor() - -```ts -xor(lhs, rhs): string -``` - -Performs a bitwise XOR operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/index.md deleted file mode 100644 index 4de7a696991..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/index.md +++ /dev/null @@ -1,47 +0,0 @@ -# noir_js - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [Noir](classes/Noir.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [ErrorWithPayload](type-aliases/ErrorWithPayload.md) | - | -| [ForeignCallHandler](type-aliases/ForeignCallHandler.md) | A callback which performs an foreign call and returns the response. | -| [ForeignCallInput](type-aliases/ForeignCallInput.md) | - | -| [ForeignCallOutput](type-aliases/ForeignCallOutput.md) | - | -| [WitnessMap](type-aliases/WitnessMap.md) | - | - -### Functions - -| Function | Description | -| :------ | :------ | -| [and](functions/and.md) | Performs a bitwise AND operation between `lhs` and `rhs` | -| [blake2s256](functions/blake2s256.md) | Calculates the Blake2s256 hash of the input bytes | -| [ecdsa\_secp256k1\_verify](functions/ecdsa_secp256k1_verify.md) | Verifies a ECDSA signature over the secp256k1 curve. | -| [ecdsa\_secp256r1\_verify](functions/ecdsa_secp256r1_verify.md) | Verifies a ECDSA signature over the secp256r1 curve. | -| [xor](functions/xor.md) | Performs a bitwise XOR operation between `lhs` and `rhs` | - -## References - -### CompiledCircuit - -Renames and re-exports [InputMap](index.md#inputmap) - -## Variables - -### InputMap - -```ts -InputMap: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md deleted file mode 100644 index e8c2f4aef3d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md +++ /dev/null @@ -1,15 +0,0 @@ -# ErrorWithPayload - -```ts -type ErrorWithPayload: ExecutionError & object; -``` - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `decodedAssertionPayload` | `any` | - | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md deleted file mode 100644 index 812b8b16481..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md +++ /dev/null @@ -1,24 +0,0 @@ -# ForeignCallHandler - -```ts -type ForeignCallHandler: (name, inputs) => Promise; -``` - -A callback which performs an foreign call and returns the response. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | The identifier for the type of foreign call being performed. | -| `inputs` | [`ForeignCallInput`](ForeignCallInput.md)[] | An array of hex encoded inputs to the foreign call. | - -## Returns - -`Promise`\<[`ForeignCallOutput`](ForeignCallOutput.md)[]\> - -outputs - An array of hex encoded outputs containing the results of the foreign call. - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md deleted file mode 100644 index dd95809186a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallInput - -```ts -type ForeignCallInput: string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md deleted file mode 100644 index b71fb78a946..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallOutput - -```ts -type ForeignCallOutput: string | string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md deleted file mode 100644 index 258c46f9d0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md +++ /dev/null @@ -1,9 +0,0 @@ -# WitnessMap - -```ts -type WitnessMap: Map; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs deleted file mode 100644 index 4796b5abaa8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/noir_js/classes/Noir","label":"Noir"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ErrorWithPayload","label":"ErrorWithPayload"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallHandler","label":"ForeignCallHandler"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallInput","label":"ForeignCallInput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallOutput","label":"ForeignCallOutput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/WitnessMap","label":"WitnessMap"}]},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_js/functions/and","label":"and"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/blake2s256","label":"blake2s256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify","label":"ecdsa_secp256k1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify","label":"ecdsa_secp256r1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/xor","label":"xor"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/compile.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/compile.md deleted file mode 100644 index 6faf763b37f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/compile.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile() - -```ts -compile( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ProgramCompilationArtifacts`](../index.md#programcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_program(fm); -``` - -```typescript -// Browser - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_program(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/compile_contract.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/compile_contract.md deleted file mode 100644 index 7d0b39a43ef..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/compile_contract.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile\_contract() - -```ts -compile_contract( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ContractCompilationArtifacts`](../index.md#contractcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_contract(fm); -``` - -```typescript -// Browser - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_contract(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/createFileManager.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/createFileManager.md deleted file mode 100644 index 7e65c1d69c7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/createFileManager.md +++ /dev/null @@ -1,21 +0,0 @@ -# createFileManager() - -```ts -createFileManager(dataDir): FileManager -``` - -Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `dataDir` | `string` | root of the file system | - -## Returns - -`FileManager` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md deleted file mode 100644 index fcea9275341..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md +++ /dev/null @@ -1,21 +0,0 @@ -# inflateDebugSymbols() - -```ts -inflateDebugSymbols(debugSymbols): any -``` - -Decompresses and decodes the debug symbols - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `debugSymbols` | `string` | The base64 encoded debug symbols | - -## Returns - -`any` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/index.md deleted file mode 100644 index b6e0f9d1bc0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_wasm - -## Exports - -### Functions - -| Function | Description | -| :------ | :------ | -| [compile](functions/compile.md) | Compiles a Noir project | -| [compile\_contract](functions/compile_contract.md) | Compiles a Noir project | -| [createFileManager](functions/createFileManager.md) | Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) | -| [inflateDebugSymbols](functions/inflateDebugSymbols.md) | Decompresses and decodes the debug symbols | - -## References - -### compile\_program - -Renames and re-exports [compile](functions/compile.md) - -## Interfaces - -### ContractCompilationArtifacts - -The compilation artifacts of a given contract. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `contract` | `ContractArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -### ProgramCompilationArtifacts - -The compilation artifacts of a given program. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | not part of the compilation output, injected later | -| `program` | `ProgramArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs deleted file mode 100644 index e0870710349..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"doc","id":"reference/NoirJS/noir_wasm/index","label":"API"},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile","label":"compile"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile_contract","label":"compile_contract"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/createFileManager","label":"createFileManager"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/inflateDebugSymbols","label":"inflateDebugSymbols"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/_category_.json deleted file mode 100644 index 5b6a20a609a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 4, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/_category_.json deleted file mode 100644 index 27869205ad3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugger", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_known_limitations.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_known_limitations.md deleted file mode 100644 index 936d416ac4b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_known_limitations.md +++ /dev/null @@ -1,59 +0,0 @@ ---- -title: Known limitations -description: - An overview of known limitations of the current version of the Noir debugger -keywords: - [ - Nargo, - Noir Debugger, - VS Code, - ] -sidebar_position: 2 ---- - -# Debugger Known Limitations - -There are currently some limits to what the debugger can observe. - -## Mutable references - -The debugger is currently blind to any state mutated via a mutable reference. For example, in: - -``` -let mut x = 1; -let y = &mut x; -*y = 2; -``` - -The update on `x` will not be observed by the debugger. That means, when running `vars` from the debugger REPL, or inspecting the _local variables_ pane in the VS Code debugger, `x` will appear with value 1 despite having executed `*y = 2;`. - -## Variables of type function or mutable references are opaque - -When inspecting variables, any variable of type `Function` or `MutableReference` will render its value as `<>` or `<>`. - -## Debugger instrumentation affects resulting ACIR - -In order to make the state of local variables observable, the debugger compiles Noir circuits interleaving foreign calls that track any mutations to them. While this works (except in the cases described above) and doesn't introduce any behavior changes, it does as a side effect produce bigger bytecode. In particular, when running the command `opcodes` on the REPL debugger, you will notice Unconstrained VM blocks that look like this: - -``` -... -5 BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [], q_c: 2 }), Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(2))], q_c: 0 })] - | outputs=[] - 5.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 5.1 | Mov { destination: RegisterIndex(3), source: RegisterIndex(1) } - 5.2 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 5.3 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 5.4 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 5.5 | Mov { destination: RegisterIndex(3), source: RegisterIndex(3) } - 5.6 | Call { location: 8 } - 5.7 | Stop - 5.8 | ForeignCall { function: "__debug_var_assign", destinations: [], inputs: [RegisterIndex(RegisterIndex(2)), RegisterIndex(RegisterIndex(3))] } -... -``` - -If you are interested in debugging/inspecting compiled ACIR without these synthetic changes, you can invoke the REPL debugger with the `--skip-instrumentation` flag or launch the VS Code debugger with the `skipConfiguration` property set to true in its launch configuration. You can find more details about those in the [Debugger REPL reference](debugger_repl.md) and the [VS Code Debugger reference](debugger_vscode.md). - -:::note -Skipping debugger instrumentation means you won't be able to inspect values of local variables. -::: - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_repl.md deleted file mode 100644 index 46e2011304e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_repl.md +++ /dev/null @@ -1,360 +0,0 @@ ---- -title: REPL Debugger -description: - Noir Debugger REPL options and commands. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -## Running the REPL debugger - -`nargo debug [OPTIONS] [WITNESS_NAME]` - -Runs the Noir REPL debugger. If a `WITNESS_NAME` is provided the debugger writes the resulting execution witness to a `WITNESS_NAME` file. - -### Options - -| Option | Description | -| --------------------- | ------------------------------------------------------------ | -| `-p, --prover-name ` | The name of the toml file which contains the inputs for the prover [default: Prover]| -| `--package ` | The name of the package to debug | -| `--print-acir` | Display the ACIR for compiled circuit | -| `--deny-warnings` | Treat all warnings as errors | -| `--silence-warnings` | Suppress warnings | -| `-h, --help` | Print help | - -None of these options are required. - -:::note -Since the debugger starts by compiling the target package, all Noir compiler options are also available. Check out the [compiler reference](../nargo_commands.md#nargo-compile) to learn more about the compiler options. -::: - -## REPL commands - -Once the debugger is running, it accepts the following commands. - -#### `help` (h) - -Displays the menu of available commands. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) value - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -### Stepping through programs - -#### `next` (n) - -Step until the next Noir source code location. While other commands, such as [`into`](#into-i) and [`step`](#step-s), allow for finer grained control of the program's execution at the opcode level, `next` is source code centric. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `next` here would cause the debugger to jump to the definition of `deep_entry_point` (if available). - -If you want to step over `deep_entry_point` and go straight to line 8, use [the `over` command](#over) instead. - -#### `over` - -Step until the next source code location, without diving into function calls. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `over` here would cause the debugger to execute until line 8 (`multiple_values_entry_point(x);`). - -If you want to step into `deep_entry_point` instead, use [the `next` command](#next-n). - -#### `out` - -Step until the end of the current function call. For example: - -``` - 3 ... - 4 fn main(x: u32) { - 5 assert(entry_point(x) == 2); - 6 swap_entry_point(x, x + 1); - 7 -> assert(deep_entry_point(x) == 4); - 8 multiple_values_entry_point(x); - 9 } - 10 - 11 unconstrained fn returns_multiple_values(x: u32) -> (u32, u32, u32, u32) { - 12 ... - ... - 55 - 56 unconstrained fn deep_entry_point(x: u32) -> u32 { - 57 -> level_1(x + 1) - 58 } - -``` - -Running `out` here will resume execution until line 8. - -#### `step` (s) - -Skips to the next ACIR code. A compiled Noir program is a sequence of ACIR opcodes. However, an unconstrained VM opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `step` command at this point would result in the debugger stopping at ACIR opcode 2, `EXPR`, skipping unconstrained computation steps. - -Use [the `into` command](#into-i) instead if you want to follow unconstrained computation step by step. - -#### `into` (i) - -Steps into the next opcode. A compiled Noir program is a sequence of ACIR opcodes. However, a BRILLIG opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `into` command at this point would result in the debugger stopping at opcode 1.0, `Mov ...`, allowing the debugger user to follow unconstrained computation step by step. - -Use [the `step` command](#step-s) instead if you want to skip to the next ACIR code directly. - -#### `continue` (c) - -Continues execution until the next breakpoint, or the end of the program. - -#### `restart` (res) - -Interrupts execution, and restarts a new debugging session from scratch. - -#### `opcodes` (o) - -Display the program's ACIR opcode sequence. For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -### Breakpoints - -#### `break [Opcode]` (or shorthand `b [Opcode]`) - -Sets a breakpoint on the specified opcode index. To get a list of the program opcode numbers, see [the `opcode` command](#opcodes-o). For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -In this example, issuing a `break 1.2` command adds break on opcode 1.2, as denoted by the `*` character: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | * Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -Running [the `continue` command](#continue-c) at this point would cause the debugger to execute the program until opcode 1.2. - -#### `delete [Opcode]` (or shorthand `d [Opcode]`) - -Deletes a breakpoint at an opcode location. Usage is analogous to [the `break` command](#). - -### Variable inspection - -#### vars - -Show variable values available at this point in execution. - -:::note -The ability to inspect variable values from the debugger depends on compilation to be run in a special debug instrumentation mode. This instrumentation weaves variable tracing code with the original source code. - -So variable value inspection comes at the expense of making the resulting ACIR bytecode bigger and harder to understand and optimize. - -If you find this compromise unacceptable, you can run the debugger with the flag `--skip-debug-instrumentation`. This will compile your circuit without any additional debug information, so the resulting ACIR bytecode will be identical to the one produced by standard Noir compilation. However, if you opt for this, the `vars` command will not be available while debugging. -::: - - -### Stacktrace - -#### `stacktrace` - -Displays the current stack trace. - - -### Witness map - -#### `witness` (w) - -Show witness map. For example: - -``` -_0 = 0 -_1 = 2 -_2 = 1 -``` - -#### `witness [Witness Index]` - -Display a single witness from the witness map. For example: - -``` -> witness 1 -_1 = 2 -``` - -#### `witness [Witness Index] [New value]` - -Overwrite the given index with a new value. For example: - -``` -> witness 1 3 -_1 = 3 -``` - - -### Unconstrained VM memory - -#### `memory` - -Show unconstrained VM memory state. For example: - -``` -> memory -At opcode 1.13: Store { destination_pointer: RegisterIndex(0), source: RegisterIndex(3) } -... -> registers -0 = 0 -1 = 10 -2 = 0 -3 = 1 -4 = 1 -5 = 2³² -6 = 1 -> into -At opcode 1.14: Const { destination: RegisterIndex(5), value: Value { inner: 1 } } -... -> memory -0 = 1 -> -``` - -In the example above: we start with clean memory, then step through a `Store` opcode which stores the value of register 3 (1) into the memory address stored in register 0 (0). Thus now `memory` shows memory address 0 contains value 1. - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: - -#### `memset [Memory address] [New value]` - -Update a memory cell with the given value. For example: - -``` -> memory -0 = 1 -> memset 0 2 -> memory -0 = 2 -> memset 1 4 -> memory -0 = 2 -1 = 4 -> -``` - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_vscode.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_vscode.md deleted file mode 100644 index c027332b3b0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/debugger/debugger_vscode.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: VS Code Debugger -description: - VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -# VS Code Noir Debugger Reference - -The Noir debugger enabled by the vscode-noir extension ships with default settings such that the most common scenario should run without any additional configuration steps. - -These defaults can nevertheless be overridden by defining a launch configuration file. This page provides a reference for the properties you can override via a launch configuration file, as well as documenting the Nargo `dap` command, which is a dependency of the VS Code Noir debugger. - - -## Creating and editing launch configuration files - -To create a launch configuration file from VS Code, open the _debug pane_, and click on _create a launch.json file_. - -![Creating a launch configuration file](@site/static/img/debugger/ref1-create-launch.png) - -A `launch.json` file will be created, populated with basic defaults. - -### Noir Debugger launch.json properties - -#### projectFolder - -_String, optional._ - -Absolute path to the Nargo project to debug. By default, it is dynamically determined by looking for the nearest `Nargo.toml` file to the active file at the moment of launching the debugger. - -#### proverName - -_String, optional._ - -Name of the prover input to use. Defaults to `Prover`, which looks for a file named `Prover.toml` at the `projectFolder`. - -#### generateAcir - -_Boolean, optional._ - -If true, generate ACIR opcodes instead of unconstrained opcodes which will be closer to release binaries but less convenient for debugging. Defaults to `false`. - -#### skipInstrumentation - -_Boolean, optional._ - -Skips variables debugging instrumentation of code, making debugging less convenient but the resulting binary smaller and closer to production. Defaults to `false`. - -:::note -Skipping instrumentation causes the debugger to be unable to inspect local variables. -::: - -## `nargo dap [OPTIONS]` - -When run without any option flags, it starts the Nargo Debug Adapter Protocol server, which acts as the debugging backend for the VS Code Noir Debugger. - -All option flags are related to preflight checks. The Debug Adapter Protocol specifies how errors are to be informed from a running DAP server, but it doesn't specify mechanisms to communicate server initialization errors between the DAP server and its client IDE. - -Thus `nargo dap` ships with a _preflight check_ mode. If flag `--preflight-check` and the rest of the `--preflight-*` flags are provided, Nargo will run the same initialization routine except it will not start the DAP server. - -`vscode-noir` will then run `nargo dap` in preflight check mode first before a debugging session starts. If the preflight check ends in error, vscode-noir will present stderr and stdout output from this process through its own Output pane in VS Code. This makes it possible for users to diagnose what pieces of configuration might be wrong or missing in case of initialization errors. - -If the preflight check succeeds, `vscode-noir` proceeds to start the DAP server normally but running `nargo dap` without any additional flags. - -### Options - -| Option | Description | -| --------------------------------------------------------- | --------------------------------------------------------------------------------------------------------- | -| `--preflight-check` | If present, dap runs in preflight check mode. | -| `--preflight-project-folder ` | Absolute path to the project to debug for preflight check. | -| `--preflight-prover-name ` | Name of prover file to use for preflight check | -| `--preflight-generate-acir` | Optional. If present, compile in ACIR mode while running preflight check. | -| `--preflight-skip-instrumentation` | Optional. If present, compile without introducing debug instrumentation while running preflight check. | -| `-h, --help` | Print help. | diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/nargo_commands.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/nargo_commands.md deleted file mode 100644 index 8842fad6647..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/nargo_commands.md +++ /dev/null @@ -1,474 +0,0 @@ ---- -title: Nargo -description: - Noir CLI Commands for Noir Prover and Verifier to create, execute, prove and verify programs, - generate Solidity verifier smart contract and compile into JSON file containing ACIR - representation and ABI of circuit. -keywords: - [ - Nargo, - Noir CLI, - Noir Prover, - Noir Verifier, - generate Solidity verifier, - compile JSON file, - ACIR representation, - ABI of circuit, - TypeScript, - ] -sidebar_position: 0 ---- - -# Command-Line Help for `nargo` - -This document contains the help content for the `nargo` command-line program. - -**Command Overview:** - -* [`nargo`↴](#nargo) -* [`nargo check`↴](#nargo-check) -* [`nargo fmt`↴](#nargo-fmt) -* [`nargo compile`↴](#nargo-compile) -* [`nargo new`↴](#nargo-new) -* [`nargo init`↴](#nargo-init) -* [`nargo execute`↴](#nargo-execute) -* [`nargo debug`↴](#nargo-debug) -* [`nargo test`↴](#nargo-test) -* [`nargo info`↴](#nargo-info) -* [`nargo lsp`↴](#nargo-lsp) -* [`nargo generate-completion-script`↴](#nargo-generate-completion-script) - -## `nargo` - -Noir's package manager - -**Usage:** `nargo ` - -###### **Subcommands:** - -* `check` — Checks the constraint system for errors -* `fmt` — Format the Noir files in a workspace -* `compile` — Compile the program and its secret execution trace into ACIR format -* `new` — Create a Noir project in a new directory -* `init` — Create a Noir project in the current directory -* `execute` — Executes a circuit to calculate its return value -* `debug` — Executes a circuit in debug mode -* `test` — Run the tests for this program -* `info` — Provides detailed information on each of a program's function (represented by a single circuit) -* `lsp` — Starts the Noir LSP server -* `generate-completion-script` — Generates a shell completion script for your favorite shell - -###### **Options:** - - - - -## `nargo check` - -Checks the constraint system for errors - -**Usage:** `nargo check [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to check -* `--workspace` — Check all packages in the workspace - - Possible values: `true`, `false` - -* `--overwrite` — Force overwrite of existing files - - Possible values: `true`, `false` - -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` - - Possible values: `true`, `false` - -* `--force` — Force a full recompilation - - Possible values: `true`, `false` - -* `--print-acir` — Display the ACIR for compiled circuit - - Possible values: `true`, `false` - -* `--deny-warnings` — Treat all warnings as errors - - Possible values: `true`, `false` - -* `--silence-warnings` — Suppress warnings - - Possible values: `true`, `false` - -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - Possible values: `true`, `false` - - - - -## `nargo fmt` - -Format the Noir files in a workspace - -**Usage:** `nargo fmt [OPTIONS]` - -###### **Options:** - -* `--check` — Run noirfmt in check mode - - Possible values: `true`, `false` - - - - -## `nargo compile` - -Compile the program and its secret execution trace into ACIR format - -**Usage:** `nargo compile [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to compile -* `--workspace` — Compile all packages in the workspace - - Possible values: `true`, `false` - -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` - - Possible values: `true`, `false` - -* `--force` — Force a full recompilation - - Possible values: `true`, `false` - -* `--print-acir` — Display the ACIR for compiled circuit - - Possible values: `true`, `false` - -* `--deny-warnings` — Treat all warnings as errors - - Possible values: `true`, `false` - -* `--silence-warnings` — Suppress warnings - - Possible values: `true`, `false` - -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - Possible values: `true`, `false` - - - - -## `nargo new` - -Create a Noir project in a new directory - -**Usage:** `nargo new [OPTIONS] ` - -###### **Arguments:** - -* `` — The path to save the new project - -###### **Options:** - -* `--name ` — Name of the package [default: package directory name] -* `--lib` — Use a library template - - Possible values: `true`, `false` - -* `--bin` — Use a binary template [default] - - Possible values: `true`, `false` - -* `--contract` — Use a contract template - - Possible values: `true`, `false` - - - - -## `nargo init` - -Create a Noir project in the current directory - -**Usage:** `nargo init [OPTIONS]` - -###### **Options:** - -* `--name ` — Name of the package [default: current directory name] -* `--lib` — Use a library template - - Possible values: `true`, `false` - -* `--bin` — Use a binary template [default] - - Possible values: `true`, `false` - -* `--contract` — Use a contract template - - Possible values: `true`, `false` - - - - -## `nargo execute` - -Executes a circuit to calculate its return value - -**Usage:** `nargo execute [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -Defaults to the name of the package being executed. - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--workspace` — Execute all packages in the workspace - - Possible values: `true`, `false` - -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` - - Possible values: `true`, `false` - -* `--force` — Force a full recompilation - - Possible values: `true`, `false` - -* `--print-acir` — Display the ACIR for compiled circuit - - Possible values: `true`, `false` - -* `--deny-warnings` — Treat all warnings as errors - - Possible values: `true`, `false` - -* `--silence-warnings` — Suppress warnings - - Possible values: `true`, `false` - -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - Possible values: `true`, `false` - -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo debug` - -Executes a circuit in debug mode - -**Usage:** `nargo debug [OPTIONS] [WITNESS_NAME]` - -###### **Arguments:** - -* `` — Write the execution witness to named file - -###### **Options:** - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--package ` — The name of the package to execute -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` - - Possible values: `true`, `false` - -* `--force` — Force a full recompilation - - Possible values: `true`, `false` - -* `--print-acir` — Display the ACIR for compiled circuit - - Possible values: `true`, `false` - -* `--deny-warnings` — Treat all warnings as errors - - Possible values: `true`, `false` - -* `--silence-warnings` — Suppress warnings - - Possible values: `true`, `false` - -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - Possible values: `true`, `false` - -* `--acir-mode` — Force ACIR output (disabling instrumentation) - - Possible values: `true`, `false` - -* `--skip-instrumentation ` — Disable vars debug instrumentation (enabled by default) - - Possible values: `true`, `false` - - - - -## `nargo test` - -Run the tests for this program - -**Usage:** `nargo test [OPTIONS] [TEST_NAME]` - -###### **Arguments:** - -* `` — If given, only tests with names containing this string will be run - -###### **Options:** - -* `--show-output` — Display output of `println` statements - - Possible values: `true`, `false` - -* `--exact` — Only run tests that match exactly - - Possible values: `true`, `false` - -* `--package ` — The name of the package to test -* `--workspace` — Test all packages in the workspace - - Possible values: `true`, `false` - -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` - - Possible values: `true`, `false` - -* `--force` — Force a full recompilation - - Possible values: `true`, `false` - -* `--print-acir` — Display the ACIR for compiled circuit - - Possible values: `true`, `false` - -* `--deny-warnings` — Treat all warnings as errors - - Possible values: `true`, `false` - -* `--silence-warnings` — Suppress warnings - - Possible values: `true`, `false` - -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - Possible values: `true`, `false` - -* `--oracle-resolver ` — JSON RPC url to solve oracle calls - - - -## `nargo info` - -Provides detailed information on each of a program's function (represented by a single circuit) - -Current information provided per circuit: 1. The number of ACIR opcodes 2. Counts the final number gates in the circuit used by a backend - -**Usage:** `nargo info [OPTIONS]` - -###### **Options:** - -* `--package ` — The name of the package to detail -* `--workspace` — Detail all packages in the workspace - - Possible values: `true`, `false` - -* `--profile-execution` - - Possible values: `true`, `false` - -* `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover - - Default value: `Prover` -* `--expression-width ` — Specify the backend expression width that should be targeted -* `--bounded-codegen` — Generate ACIR with the target backend expression width. The default is to generate ACIR without a bound and split expressions after code generation. Activating this flag can sometimes provide optimizations for certain programs - - Default value: `false` - - Possible values: `true`, `false` - -* `--force` — Force a full recompilation - - Possible values: `true`, `false` - -* `--print-acir` — Display the ACIR for compiled circuit - - Possible values: `true`, `false` - -* `--deny-warnings` — Treat all warnings as errors - - Possible values: `true`, `false` - -* `--silence-warnings` — Suppress warnings - - Possible values: `true`, `false` - -* `--debug-comptime-in-file ` — Enable printing results of comptime evaluation: provide a path suffix for the module to debug, e.g. "package_name/src/main.nr" -* `--skip-underconstrained-check` — Flag to turn off the compiler check for under constrained values. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code - - Possible values: `true`, `false` - - - - -## `nargo lsp` - -Starts the Noir LSP server - -Starts an LSP server which allows IDEs such as VS Code to display diagnostics in Noir source. - -VS Code Noir Language Support: https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir - -**Usage:** `nargo lsp` - - - -## `nargo generate-completion-script` - -Generates a shell completion script for your favorite shell - -**Usage:** `nargo generate-completion-script ` - -###### **Arguments:** - -* `` — The shell to generate completions for. One of: bash, elvish, fish, powershell, zsh - - - -
- - - This document was generated automatically by - clap-markdown. - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/noir_codegen.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/noir_codegen.md deleted file mode 100644 index e4c362f9610..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/reference/noir_codegen.md +++ /dev/null @@ -1,116 +0,0 @@ ---- -title: Noir Codegen for TypeScript -description: Learn how to use Noir codegen to generate TypeScript bindings -keywords: [Nargo, Noir, compile, TypeScript] -sidebar_position: 3 ---- - -When using TypeScript, it is extra work to interpret Noir program outputs in a type-safe way. Third party libraries may exist for popular Noir programs, but they are either hard to find or unmaintained. - -Now you can generate TypeScript bindings for your Noir programs in two steps: - -1. Exporting Noir functions using `nargo export` -2. Using the TypeScript module `noir_codegen` to generate TypeScript binding - -**Note:** you can only export functions from a Noir *library* (not binary or contract program types). - -## Installation - -### Your TypeScript project - -If you don't already have a TypeScript project you can add the module with `yarn` (or `npm`), then initialize it: - -```bash -yarn add typescript -D -npx tsc --init -``` - -### Add TypeScript module - `noir_codegen` - -The following command will add the module to your project's devDependencies: - -```bash -yarn add @noir-lang/noir_codegen -D -``` - -### Nargo library - -Make sure you have Nargo, v0.25.0 or greater, installed. If you don't, follow the [installation guide](../getting_started/noir_installation.md). - -If you're in a new project, make a `circuits` folder and create a new Noir library: - -```bash -mkdir circuits && cd circuits -nargo new --lib myNoirLib -``` - -## Usage - -### Export ABI of specified functions - -First go to the `.nr` files in your Noir library, and add the `#[export]` macro to each function that you want to use in TypeScript. - -```rust -#[export] -fn your_function(... -``` - -From your Noir library (where `Nargo.toml` is), run the following command: - -```bash -nargo export -``` - -You will now have an `export` directory with a .json file per exported function. - -You can also specify the directory of Noir programs using `--program-dir`, for example: - -```bash -nargo export --program-dir=./circuits/myNoirLib -``` - -### Generate TypeScript bindings from exported functions - -To use the `noir-codegen` package we added to the TypeScript project: - -```bash -yarn noir-codegen ./export/your_function.json -``` - -This creates an `exports` directory with an `index.ts` file containing all exported functions. - -**Note:** adding `--out-dir` allows you to specify an output dir for your TypeScript bindings to go. Eg: - -```bash -yarn noir-codegen ./export/*.json --out-dir ./path/to/output/dir -``` - -## Example .nr function to .ts output - -Consider a Noir library with this function: - -```rust -#[export] -fn not_equal(x: Field, y: Field) -> bool { - x != y -} -``` - -After the export and codegen steps, you should have an `index.ts` like: - -```typescript -export type Field = string; - - -export const is_equal_circuit: CompiledCircuit = -{"abi":{"parameters":[{"name":"x","type":{"kind":"field"},"visibility":"private"},{"name":"y","type":{"kind":"field"},"visibility":"private"}],"return_type":{"abi_type":{"kind":"boolean"},"visibility":"private"}},"bytecode":"H4sIAAAAAAAA/7WUMQ7DIAxFQ0Krrr2JjSGYLVcpKrn/CaqqDQN12WK+hPBgmWd/wEyHbF1SS923uhOs3pfoChI+wKXMAXzIKyNj4PB0TFTYc0w5RUjoqeAeEu1wqK0F54RGkWvW44LPzExnlkbMEs4JNZmN8PxS42uHv82T8a3Jeyn2Ks+VLPcO558HmyLMCDOXAXXtpPt4R/Rt9T36ss6dS9HGPx/eG17nGegKBQAA"}; - -export async function is_equal(x: Field, y: Field, foreignCallHandler?: ForeignCallHandler): Promise { - const program = new Noir(is_equal_circuit); - const args: InputMap = { x, y }; - const { returnValue } = await program.execute(args, foreignCallHandler); - return returnValue as boolean; -} -``` - -Now the `is_equal()` function and relevant types are readily available for use in TypeScript. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/debugger.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/debugger.md deleted file mode 100644 index 200b5fc423a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/debugger.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Debugger -description: Learn about the Noir Debugger, in its REPL or VS Code versions. -keywords: [Nargo, VSCode, Visual Studio Code, REPL, Debugger] -sidebar_position: 2 ---- - -# Noir Debugger - -There are currently two ways of debugging Noir programs: - -1. From VS Code, via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). -2. Via the REPL debugger, which ships with Nargo. - -In order to use either version of the debugger, you will need to install recent enough versions of Noir, [Nargo](../getting_started/noir_installation.md) and vscode-noir: - -- Noir & Nargo ≥0.28.0 -- Noir's VS Code extension ≥0.0.11 - -:::info -At the moment, the debugger supports debugging binary projects, but not contracts. -::: - -We cover the VS Code Noir debugger more in depth in [its VS Code debugger how-to guide](../how_to/debugger/debugging_with_vs_code.md) and [the reference](../reference/debugger/debugger_vscode.md). - -The REPL debugger is discussed at length in [the REPL debugger how-to guide](../how_to/debugger/debugging_with_the_repl.md) and [the reference](../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/language_server.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/language_server.md deleted file mode 100644 index 81e0356ef8a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/language_server.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Language Server -description: Learn about the Noir Language Server, how to install the components, and configuration that may be required. -keywords: [Nargo, Language Server, LSP, VSCode, Visual Studio Code] -sidebar_position: 0 ---- - -This section helps you install and configure the Noir Language Server. - -The Language Server Protocol (LSP) has two components, the [Server](#language-server) and the [Client](#language-client). Below we describe each in the context of Noir. - -## Language Server - -The Server component is provided by the Nargo command line tool that you installed at the beginning of this guide. -As long as Nargo is installed and you've used it to run other commands in this guide, it should be good to go! - -If you'd like to verify that the `nargo lsp` command is available, you can run `nargo --help` and look for `lsp` in the list of commands. If you see it, you're using a version of Noir with LSP support. - -## Language Client - -The Client component is usually an editor plugin that launches the Server. It communicates LSP messages between the editor and the Server. For example, when you save a file, the Client will alert the Server, so it can try to compile the project and report any errors. - -Currently, Noir provides a Language Client for Visual Studio Code via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). - -> **Note:** Noir's Language Server Protocol support currently assumes users' VSCode workspace root to be the same as users' Noir project root (i.e. where Nargo.toml lies). -> -> If LSP features seem to be missing / malfunctioning, make sure you are opening your Noir project directly (instead of as a sub-folder) in your VSCode instance. - -When your language server is running correctly and the VSCode plugin is installed, you should see handy codelens buttons for compilation, measuring circuit size, execution, and tests: - -![Compile and Execute](@site/static/img/codelens_compile_execute.png) -![Run test](@site/static/img/codelens_run_test.png) - -You should also see your tests in the `testing` panel: - -![Testing panel](@site/static/img/codelens_testing_panel.png) - -### Configuration - -- **Noir: Enable LSP** - If checked, the extension will launch the Language Server via `nargo lsp` and communicate with it. -- **Noir: Nargo Flags** - Additional flags may be specified if you require them to be added when the extension calls `nargo lsp`. -- **Noir: Nargo Path** - An absolute path to a Nargo binary with the `lsp` command. This may be useful if Nargo is not within the `PATH` of your editor. -- **Noir > Trace: Server** - Setting this to `"messages"` or `"verbose"` will log LSP messages between the Client and Server. Useful for debugging. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/testing.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/testing.md deleted file mode 100644 index 866677da567..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tooling/testing.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Testing in Noir -description: Learn how to use Nargo to test your Noir program in a quick and easy way -keywords: [Nargo, testing, Noir, compile, test] -sidebar_position: 1 ---- - -You can test your Noir programs using Noir circuits. - -Nargo will automatically compile and run any functions which have the decorator `#[test]` on them if -you run `nargo test`. - -For example if you have a program like: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test] -fn test_add() { - assert(add(2,2) == 4); - assert(add(0,1) == 1); - assert(add(1,0) == 1); -} -``` - -Running `nargo test` will test that the `test_add` function can be executed while satisfying all -the constraints which allows you to test that add returns the expected values. Test functions can't -have any arguments currently. - -### Test fail - -You can write tests that are expected to fail by using the decorator `#[test(should_fail)]`. For example: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test(should_fail)] -fn test_add() { - assert(add(2,2) == 5); -} -``` - -You can be more specific and make it fail with a specific reason by using `should_fail_with = ""`: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "What is the airspeed velocity of an unladen swallow")] -fn test_bridgekeeper() { - main(32); -} -``` - -The string given to `should_fail_with` doesn't need to exactly match the failure reason, it just needs to be a substring of it: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "airspeed velocity")] -fn test_bridgekeeper() { - main(32); -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tutorials/noirjs_app.md deleted file mode 100644 index 6e69ea0bbed..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/tutorials/noirjs_app.md +++ /dev/null @@ -1,366 +0,0 @@ ---- -title: Building a web app with NoirJS -description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. -keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] -sidebar_position: 0 -pagination_next: noir/concepts/data_types/index ---- - -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! - -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/quick_start.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! - -### Nargo - -Run: - -```bash -nargo new circuit -``` - -And... That's about it. Your program is ready to be compiled and run. - -To compile, let's `cd` into the `circuit` folder to enter our project, and call: - -```bash -nargo compile -``` - -This compiles our circuit into `json` format and add it to a new `target` folder. - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` - -::: - -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/quick_start.md) guide. However, we want our app to run on the browser, so we need Vite. - -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. - -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". - -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: - -```bash -cd vite-project -``` - -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; - -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); -``` - -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: - -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev -``` - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... -``` - -::: - -#### Some cleanup - -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. - -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) - -## HTML - -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: - -```html - - - - - - -

Noir app

-
- - -
-
-

Logs

-

Proof

-
- - -``` - -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). - -Start by pasting in this boilerplate code: - -```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} - -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } -}); -``` - -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html -``` - -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - -::: - -## Some NoirJS - -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: - -```ts -import circuit from '../circuit/target/circuit.json'; -``` - -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: - -```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -import { Noir } from '@noir-lang/noir_js'; -``` - -And instantiate them inside our try-catch block: - -```ts -// try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); -// } -``` - -:::note - -For the remainder of the tutorial, everything will be happening inside the `try` block - -::: - -## Our app - -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: - -```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; -``` - -Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: - -```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); -const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); -``` - -You're probably eager to see stuff happening, so go and run your app now! - -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. - -![Getting Started 0](@site/static/img/noir_getting_started_1.png) - -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! - -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. - -## Verifying - -Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: - -```js -display('logs', 'Verifying proof... ⌛'); -const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); -``` - -You have successfully generated a client-side Noir web app! - -![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) - -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. - -## UltraHonk Backend - -Barretenberg has recently exposed a new UltraHonk backend. We can use UltraHonk in NoirJS after version 0.33.0. Everything will be the same as the tutorial above, except that the class we need to import will change: - -```js -import { UltraHonkBackend, UltraHonkVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -``` - -The backend will then be instantiated as such: - -```js -const backend = new UltraHonkBackend(circuit); -``` - -Then all the commands to prove and verify your circuit will be same. - -The only feature currently unsupported with UltraHonk are [recursive proofs](../explainers/explainer-recursion.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/cspell.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/cspell.json deleted file mode 100644 index c60b0a597b1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/cspell.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "words": [ - "Cryptdoku" - ] -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-oracle.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-oracle.md deleted file mode 100644 index 821e1f95c04..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-oracle.md +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: Oracles -description: This guide provides an in-depth understanding of how Oracles work in Noir programming. Learn how to use outside calculations in your programs, constrain oracles, and understand their uses and limitations. -keywords: - - Noir Programming - - Oracles - - JSON-RPC - - Foreign Call Handlers - - Constrained Functions - - Blockchain Programming -sidebar_position: 1 ---- - -If you've seen "The Matrix" you may recall "The Oracle" as Gloria Foster smoking cigarettes and baking cookies. While she appears to "know things", she is actually providing a calculation of a pre-determined future. Noir Oracles are similar, in a way. They don't calculate the future (yet), but they allow you to use outside calculations in your programs. - -![matrix oracle prediction](@site/static/img/memes/matrix_oracle.jpeg) - -A Noir program is usually self-contained. You can pass certain inputs to it, and it will generate a deterministic output for those inputs. But what if you wanted to defer some calculation to an outside process or source? - -Oracles are functions that provide this feature. - -## Use cases - -An example usage for Oracles is proving something on-chain. For example, proving that the ETH-USDC quote was below a certain target at a certain block time. Or even making more complex proofs like proving the ownership of an NFT as an anonymous login method. - -Another interesting use case is to defer expensive calculations to be made outside of the Noir program, and then constraining the result; similar to the use of [unconstrained functions](../noir/concepts//unconstrained.md). - -In short, anything that can be constrained in a Noir program but needs to be fetched from an external source is a great candidate to be used in oracles. - -## Constraining oracles - -Just like in The Matrix, Oracles are powerful. But with great power, comes great responsibility. Just because you're using them in a Noir program doesn't mean they're true. Noir has no superpowers. If you want to prove that Portugal won the Euro Cup 2016, you're still relying on potentially untrusted information. - -To give a concrete example, Alice wants to login to the [NounsDAO](https://nouns.wtf/) forum with her username "noir_nouner" by proving she owns a noun without revealing her ethereum address. Her Noir program could have an oracle call like this: - -```rust -#[oracle(getNoun)] -unconstrained fn get_noun(address: Field) -> Field -``` - -This oracle could naively resolve with the number of Nouns she possesses. However, it is useless as a trusted source, as the oracle could resolve to anything Alice wants. In order to make this oracle call actually useful, Alice would need to constrain the response from the oracle, by proving her address and the noun count belongs to the state tree of the contract. - -In short, **Oracles don't prove anything. Your Noir program does.** - -:::danger - -If you don't constrain the return of your oracle, you could be clearly opening an attack vector on your Noir program. Make double-triple sure that the return of an oracle call is constrained! - -::: - -## How to use Oracles - -On CLI, Nargo resolves oracles by making JSON RPC calls, which means it would require an RPC node to be running. - -In JavaScript, NoirJS accepts and resolves arbitrary call handlers (that is, not limited to JSON) as long as they match the expected types the developer defines. Refer to [Foreign Call Handler](../reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md) to learn more about NoirJS's call handling. - -If you want to build using oracles, follow through to the [oracle guide](../how_to/how-to-oracles.md) for a simple example on how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-recursion.md deleted file mode 100644 index df8529ef4e0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-recursion.md +++ /dev/null @@ -1,176 +0,0 @@ ---- -title: Recursive proofs -description: Explore the concept of recursive proofs in Zero-Knowledge programming. Understand how recursion works in Noir, a language for writing smart contracts on the EVM blockchain. Learn through practical examples like Alice and Bob's guessing game, Charlie's recursive merkle tree, and Daniel's reusable components. Discover how to use recursive proofs to optimize computational resources and improve efficiency. - -keywords: - [ - "Recursive Proofs", - "Zero-Knowledge Programming", - "Noir", - "EVM Blockchain", - "Smart Contracts", - "Recursion in Noir", - "Alice and Bob Guessing Game", - "Recursive Merkle Tree", - "Reusable Components", - "Optimizing Computational Resources", - "Improving Efficiency", - "Verification Key", - "Aggregation", - "Recursive zkSNARK schemes", - "PLONK", - "Proving and Verification Keys" - ] -sidebar_position: 1 -pagination_next: how_to/how-to-recursion ---- - -In programming, we tend to think of recursion as something calling itself. A classic example would be the calculation of the factorial of a number: - -```js -function factorial(n) { - if (n === 0 || n === 1) { - return 1; - } else { - return n * factorial(n - 1); - } -} -``` - -In this case, while `n` is not `1`, this function will keep calling itself until it hits the base case, bubbling up the result on the call stack: - -```md - Is `n` 1? <--------- - /\ / - / \ n = n -1 - / \ / - Yes No -------- -``` - -In Zero-Knowledge, recursion has some similarities. - -It is not a Noir function calling itself, but a proof being used as an input to another circuit. In short, you verify one proof *inside* another proof, returning the proof that both proofs are valid. - -This means that, given enough computational resources, you can prove the correctness of any arbitrary number of proofs in a single proof. This could be useful to design state channels (for which a common example would be [Bitcoin's Lightning Network](https://en.wikipedia.org/wiki/Lightning_Network)), to save on gas costs by settling one proof on-chain, or simply to make business logic less dependent on a consensus mechanism. - -## Examples - -Let us look at some of these examples - -### Alice and Bob - Guessing game - -Alice and Bob are friends, and they like guessing games. They want to play a guessing game online, but for that, they need a trusted third-party that knows both of their secrets and finishes the game once someone wins. - -So, they use zero-knowledge proofs. Alice tries to guess Bob's number, and Bob will generate a ZK proof stating whether she succeeded or failed. - -This ZK proof can go on a smart contract, revealing the winner and even giving prizes. However, this means every turn needs to be verified on-chain. This incurs some cost and waiting time that may simply make the game too expensive or time-consuming to be worth it. - -As a solution, Alice proposes the following: "what if Bob generates his proof, and instead of sending it on-chain, I verify it *within* my own proof before playing my own turn?". - -She can then generate a proof that she verified his proof, and so on. - -```md - Did you fail? <-------------------------- - / \ / - / \ n = n -1 - / \ / - Yes No / - | | / - | | / - | You win / - | / - | / -Generate proof of that / - + / - my own guess ---------------- -``` - -### Charlie - Recursive merkle tree - -Charlie is a concerned citizen, and wants to be sure his vote in an election is accounted for. He votes with a ZK proof, but he has no way of knowing that his ZK proof was included in the total vote count! - -If the vote collector puts all of the votes into a [Merkle tree](https://en.wikipedia.org/wiki/Merkle_tree), everyone can prove the verification of two proofs within one proof, as such: - -```md - abcd - __________|______________ - | | - ab cd - _____|_____ ______|______ - | | | | - alice bob charlie daniel -``` - -Doing this recursively allows us to arrive on a final proof `abcd` which if true, verifies the correctness of all the votes. - -### Daniel - Reusable components - -Daniel has a big circuit and a big headache. A part of his circuit is a setup phase that finishes with some assertions that need to be made. But that section alone takes most of the proving time, and is largely independent of the rest of the circuit. - -He might find it more efficient to generate a proof for that setup phase separately, and verify that proof recursively in the actual business logic section of his circuit. This will allow for parallelization of both proofs, which results in a considerable speedup. - -## What params do I need - -As you can see in the [recursion reference](noir/standard_library/recursion.mdx), a simple recursive proof requires: - -- The proof to verify -- The Verification Key of the circuit that generated the proof -- A hash of this verification key, as it's needed for some backends -- The public inputs for the proof - -:::info - -Recursive zkSNARK schemes do not necessarily "verify a proof" in the sense that you expect a true or false to be spit out by the verifier. Rather an aggregation object is built over the public inputs. - -So, taking the example of Alice and Bob and their guessing game: - -- Alice makes her guess. Her proof is *not* recursive: it doesn't verify any proof within it! It's just a standard `assert(x != y)` circuit -- Bob verifies Alice's proof and makes his own guess. In this circuit, he doesn't exactly *prove* the verification of Alice's proof. Instead, he *aggregates* his proof to Alice's proof. The actual verification is done when the full proof is verified, for example when using `nargo verify` or through the verifier smart contract. - -We can imagine recursive proofs a [relay race](https://en.wikipedia.org/wiki/Relay_race). The first runner doesn't have to receive the baton from anyone else, as he/she already starts with it. But when his/her turn is over, the next runner needs to receive it, run a bit more, and pass it along. Even though every runner could theoretically verify the baton mid-run (why not? 🏃🔍), only at the end of the race does the referee verify that the whole race is valid. - -::: - -## Some architecture - -As with everything in computer science, there's no one-size-fits all. But there are some patterns that could help understanding and implementing them. To give three examples: - -### Adding some logic to a proof verification - -This would be an approach for something like our guessing game, where proofs are sent back and forth and are verified by each opponent. This circuit would be divided in two sections: - -- A `recursive verification` section, which would be just the call to `std::verify_proof`, and that would be skipped on the first move (since there's no proof to verify) -- A `guessing` section, which is basically the logic part where the actual guessing happens - -In such a situation, and assuming Alice is first, she would skip the first part and try to guess Bob's number. Bob would then verify her proof on the first section of his run, and try to guess Alice's number on the second part, and so on. - -### Aggregating proofs - -In some one-way interaction situations, recursion would allow for aggregation of simple proofs that don't need to be immediately verified on-chain or elsewhere. - -To give a practical example, a barman wouldn't need to verify a "proof-of-age" on-chain every time he serves alcohol to a customer. Instead, the architecture would comprise two circuits: - -- A `main`, non-recursive circuit with some logic -- A `recursive` circuit meant to verify two proofs in one proof - -The customer's proofs would be intermediate, and made on their phones, and the barman could just verify them locally. He would then aggregate them into a final proof sent on-chain (or elsewhere) at the end of the day. - -### Recursively verifying different circuits - -Nothing prevents you from verifying different circuits in a recursive proof, for example: - -- A `circuit1` circuit -- A `circuit2` circuit -- A `recursive` circuit - -In this example, a regulator could verify that taxes were paid for a specific purchase by aggregating both a `payer` circuit (proving that a purchase was made and taxes were paid), and a `receipt` circuit (proving that the payment was received) - -## How fast is it - -At the time of writing, verifying recursive proofs is surprisingly fast. This is because most of the time is spent on generating the verification key that will be used to generate the next proof. So you are able to cache the verification key and reuse it later. - -Currently, Noir JS packages don't expose the functionality of loading proving and verification keys, but that feature exists in the underlying `bb.js` package. - -## How can I try it - -Learn more about using recursion in Nargo and NoirJS in the [how-to guide](../how_to/how-to-recursion.md) and see a full example in [noir-examples](https://github.com/noir-lang/noir-examples). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-writing-noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-writing-noir.md deleted file mode 100644 index 3ce4245dc45..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/explainers/explainer-writing-noir.md +++ /dev/null @@ -1,177 +0,0 @@ ---- -title: Thinking in Circuits -description: Considerations when writing Noir programs -keywords: [Noir, programming, rust] -tags: [Optimization] -sidebar_position: 0 ---- - - -This article intends to set you up with key concepts essential for writing more viable applications that use zero knowledge proofs, namely around efficient circuits. - -## Context - 'Efficient' is subjective - -When writing a web application for a performant computer with high-speed internet connection, writing efficient code sometimes is seen as an afterthought only if needed. Large multiplications running at the innermost of nested loops may not even be on a dev's radar. -When writing firmware for a battery-powered microcontroller, you think of cpu cycles as rations to keep within a product's power budget. - -> Code is written to create applications that perform specific tasks within specific constraints - -And these constraints differ depending on where the compiled code is execute. - -### The Ethereum Virtual Machine (EVM) - -In scenarios where extremely low gas costs are required for an Ethereum application to be viable/competitive, Ethereum smart contract developers get into what is colloquially known as: "*gas golfing*". Finding the lowest execution cost of their compiled code (EVM bytecode) to achieve a specific task. - -The equivalent optimization task when writing zk circuits is affectionately referred to as "*gate golfing*", finding the lowest gate representation of the compiled Noir code. - -### Coding for circuits - a paradigm shift - -In zero knowledge cryptography, code is compiled to "circuits" consisting of arithmetic gates, and gate count is the significant cost. Depending on the proving system this is linearly proportionate to proving time, and so from a product point this should be kept as low as possible. - -Whilst writing efficient code for web apps and Solidity has a few key differences, writing efficient circuits have a different set of considerations. It is a bit of a paradigm shift, like writing code for GPUs for the first time... - -For example, drawing a circle at (0, 0) of radius `r`: -- For a single CPU thread, -``` -for theta in 0..2*pi { - let x = r * cos(theta); - let y = r * sin(theta); - draw(x, y); -} // note: would do 0 - pi/2 and draw +ve/-ve x and y. -``` - -- For GPUs (simultaneous parallel calls with x, y across image), -``` -if (x^2 + y^2 = r^2) { - draw(x, y); -} -``` - -([Related](https://www.youtube.com/watch?v=-P28LKWTzrI)) - -Whilst this CPU -> GPU does not translate to circuits exactly, it is intended to exemplify the difference in intuition when coding for different machine capabilities/constraints. - -### Context Takeaway - -For those coming from a primarily web app background, this article will explain what you need to consider when writing circuits. Furthermore, for those experienced writing efficient machine code, prepare to shift what you think is efficient 😬 - -## Translating from Rust - -For some applications using Noir, existing code might be a convenient starting point to then proceed to optimize the gate count of. - -:::note -Many valuable functions and algorithms have been written in more established languages (C/C++), and converted to modern ones (like Rust). -::: - -Fortunately for Noir developers, when needing a particular function a Rust implementation can be readily compiled into Noir with some key changes. While the compiler does a decent amount of optimizations, it won't be able to change code that has been optimized for clock-cycles into code optimized for arithmetic gates. - -A few things to do when converting Rust code to Noir: -- `println!` is not a macro, use `println` function (same for `assert_eq`) -- No early `return` in function. Use constrain via assertion instead -- No passing by reference. Remove `&` operator to pass by value (copy) -- No boolean operators (`&&`, `||`). Use bitwise operators (`&`, `|`) with boolean values -- No type `usize`. Use types `u8`, `u32`, `u64`, ... -- `main` return must be public, `pub` -- No `const`, use `global` -- Noir's LSP is your friend, so error message should be informative enough to resolve syntax issues. - -## Writing efficient Noir for performant products - -The following points help refine our understanding over time. - -:::note -A Noir program makes a statement that can be verified. -::: - -It compiles to a structure that represents the calculation, and can assert results within the calculation at any stage (via the `constrain` keyword). - -A Noir program compiles to an Abstract Circuit Intermediate Representation which is: - - Conceptually a tree structure - - Leaves (inputs) are the `Field` type - - Nodes contain arithmetic operations to combine them (gates) - - The root is the final result (return value) - -:::tip -The command `nargo info` shows the programs circuit size, and is useful to compare the value of changes made. -You can dig deeper and use the `--print-acir` param to take a closer look at individual ACIR opcodes, and the proving backend to see its gate count (eg for barretenberg, `bb gates -b ./target/program.json`). -::: - -### Use the `Field` type - -Since the native type of values in circuits are `Field`s, using them for variables in Noir means less gates converting them under the hood. -Some things to be mindful of when using a Field type for a regular integer value: -- A variable of type `Field` can be cast `as` an integer type (eg `u8`, `u64`) - - Note: this retains only the bits of the integer type. Eg a Field value of 260 as a `u8` becomes 4 -- For Field types arithmetic operations meaningfully overflow/underflow, yet for integer types they are checked according to their size -- Comparisons and bitwise operations do not exist for `Field`s, cast to an appropriately sized integer type when you need to - -:::tip -Where possible, use `Field` type for values. Using smaller value types, and bit-packing strategies, will result in MORE gates -::: - - -### Use Arithmetic over non-arithmetic operations - -Since circuits are made of arithmetic gates, the cost of arithmetic operations tends to be one gate. Whereas for procedural code, they represent several clock cycles. - -Inversely, non-arithmetic operators are achieved with multiple gates, vs 1 clock cycle for procedural code. - -| (cost\op) | arithmetic
(`*`, `+`) | bit-wise ops
(eg `<`, `\|`, `>>`) | -| - | - | - | -| **cycles** | 10+ | 1 | -| **gates** | 1 | 10+ | - -Bit-wise operations (e.g. bit shifts `<<` and `>>`), albeit commonly used in general programming and especially for clock cycle optimizations, are on the contrary expensive in gates when performed within circuits. - -Translate away from bit shifts when writing constrained functions for the best performance. - -On the flip side, feel free to use bit shifts in unconstrained functions and tests if necessary, as they are executed outside of circuits and does not induce performance hits. - -### Use static over dynamic values - -Another general theme that manifests in different ways is that static reads are represented with less gates than dynamic ones. - -Reading from read-only memory (ROM) adds less gates than random-access memory (RAM), 2 vs ~3.25 due to the additional bounds checks. Arrays of fixed length (albeit used at a lower capacity), will generate less gates than dynamic storage. - -Related to this, if an index used to access an array is not known at compile time (ie unknown until run time), then ROM will be converted to RAM, expanding the gate count. - -:::tip -Use arrays and indices that are known at compile time where possible. -Using `assert_constant(i);` before an index, `i`, is used in an array will give a compile error if `i` is NOT known at compile time. -::: - -### Leverage unconstrained execution - -Constrained verification can leverage unconstrained execution, this is especially useful for operations that are represented by many gates. -Use an [unconstrained function](../noir/concepts/unconstrained.md) to perform gate-heavy calculations, then verify and constrain the result. - -Eg division generates more gates than multiplication, so calculating the quotient in an unconstrained function then constraining the product for the quotient and divisor (+ any remainder) equals the dividend will be more efficient. - -Use ` if is_unconstrained() { /`, to conditionally execute code if being called in an unconstrained vs constrained way. - -## Advanced - -Unless you're well into the depth of gate optimization, this advanced section can be ignored. - -### Combine arithmetic operations - -A Noir program can be honed further by combining arithmetic operators in a way that makes the most of each constraint of the backend proving system. This is in scenarios where the backend might not be doing this perfectly. - -Eg Barretenberg backend (current default for Noir) is a width-4 PLONKish constraint system -$ w_1*w_2*q_m + w_1*q_1 + w_2*q_2 + w_3*q_3 + w_4*q_4 + q_c = 0 $ - -Here we see there is one occurrence of witness 1 and 2 ($w_1$, $w_2$) being multiplied together, with addition to witnesses 1-4 ($w_1$ .. $w_4$) multiplied by 4 corresponding circuit constants ($q_1$ .. $q_4$) (plus a final circuit constant, $q_c$). - -Use `nargo info --print-acir`, to inspect the ACIR opcodes (and the proving system for gates), and it may present opportunities to amend the order of operations and reduce the number of constraints. - -#### Variable as witness vs expression - -If you've come this far and really know what you're doing at the equation level, a temporary lever (that will become unnecessary/useless over time) is: `std::as_witness`. This informs the compiler to save a variable as a witness not an expression. - -The compiler will mostly be correct and optimal, but this may help some near term edge cases that are yet to optimize. -Note: When used incorrectly it will create **less** efficient circuits (higher gate count). - -## References -- Guillaume's ["`Cryptdoku`" talk](https://www.youtube.com/watch?v=MrQyzuogxgg) (Jun'23) -- Tips from Tom, Jake and Zac. -- [Idiomatic Noir](https://www.vlayer.xyz/blog/idiomatic-noir-part-1-collections) blog post diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/noir_installation.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/noir_installation.md deleted file mode 100644 index a5c7e649278..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/noir_installation.md +++ /dev/null @@ -1,106 +0,0 @@ ---- -title: Standalone Noir Installation -description: There are different ways to install Nargo, the one-stop shop and command-line tool for developing Noir programs. This guide explains how to specify which version to install when using noirup, and using WSL for windows. -keywords: [ - Installation - Nargo - Noirup - Binaries - Compiling from Source - WSL for Windows - macOS - Linux - Nix - Direnv - Uninstalling Nargo - ] -sidebar_position: 2 ---- - -Noirup is the endorsed method for installing Nargo, streamlining the process of fetching binaries or compiling from source. It supports a range of options to cater to your specific needs, from nightly builds and specific versions to compiling from various sources. - -### Installing Noirup - -First, ensure you have `noirup` installed: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -### Fetching Binaries - -With `noirup`, you can easily switch between different Nargo versions, including nightly builds: - -- **Nightly Version**: Install the latest nightly build. - - ```sh - noirup --version nightly - ``` - -- **Specific Version**: Install a specific version of Nargo. - - ```sh - noirup --version - ``` - -### Compiling from Source - -`noirup` also enables compiling Nargo from various sources: - -- **From a Specific Branch**: Install from the latest commit on a branch. - - ```sh - noirup --branch - ``` - -- **From a Fork**: Install from the main branch of a fork. - - ```sh - noirup --repo - ``` - -- **From a Specific Branch in a Fork**: Install from a specific branch in a fork. - - ```sh - noirup --repo --branch - ``` - -- **From a Specific Pull Request**: Install from a specific PR. - - ```sh - noirup --pr - ``` - -- **From a Specific Commit**: Install from a specific commit. - - ```sh - noirup -C - ``` - -- **From Local Source**: Compile and install from a local directory. - - ```sh - noirup --path ./path/to/local/source - ``` - -## Installation on Windows - -The default backend for Noir (Barretenberg) doesn't provide Windows binaries at this time. For that reason, Noir cannot be installed natively. However, it is available by using Windows Subsystem for Linux (WSL). - -Step 1: Follow the instructions [here](https://learn.microsoft.com/en-us/windows/wsl/install) to install and run WSL. - -step 2: Follow the [Noirup instructions](#installing-noirup). - -## Setting up shell completions - -Once `nargo` is installed, you can [set up shell completions for it](setting_up_shell_completions). - -## Uninstalling Nargo - -If you installed Nargo with `noirup`, you can uninstall Nargo by removing the files in `~/.nargo`, `~/nargo`, and `~/noir_cache`. This ensures that all installed binaries, configurations, and cache related to Nargo are fully removed from your system. - -```bash -rm -r ~/.nargo -rm -r ~/nargo -rm -r ~/noir_cache -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/project_breakdown.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/project_breakdown.md deleted file mode 100644 index e442e377040..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/project_breakdown.md +++ /dev/null @@ -1,159 +0,0 @@ ---- -title: Project Breakdown -description: - Learn about the anatomy of a Nargo project, including the purpose of the Prover TOML - file, and how to prove and verify your program. -keywords: - [Nargo, Nargo project, Prover.toml, proof verification, private asset transfer] -sidebar_position: 1 ---- - -This section breaks down our hello world program from the previous section. - -## Anatomy of a Nargo Project - -Upon creating a new project with `nargo new` and building the in/output files with `nargo check` -commands, you would get a minimal Nargo project of the following structure: - - - src - - Prover.toml - - Nargo.toml - -The source directory _src_ holds the source code for your Noir program. By default only a _main.nr_ -file will be generated within it. - -### Prover.toml - -_Prover.toml_ is used for specifying the input values for executing and proving the program. You can specify `toml` files with different names by using the `--prover-name` or `-p` flags, see the [Prover](#provertoml) section below. Optionally you may specify expected output values for prove-time checking as well. - -### Nargo.toml - -_Nargo.toml_ contains the environmental options of your project. It contains a "package" section and a "dependencies" section. - -Example Nargo.toml: - -```toml -[package] -name = "noir_starter" -type = "bin" -authors = ["Alice"] -compiler_version = "0.9.0" -description = "Getting started with Noir" -entry = "circuit/main.nr" -license = "MIT" - -[dependencies] -ecrecover = {tag = "v0.9.0", git = "https://github.com/colinnielsen/ecrecover-noir.git"} -``` - -Nargo.toml for a [workspace](../noir/modules_packages_crates/workspaces.md) will look a bit different. For example: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -#### Package section - -The package section defines a number of fields including: - -- `name` (**required**) - the name of the package -- `type` (**required**) - can be "bin", "lib", or "contract" to specify whether its a binary, library or Aztec contract -- `authors` (optional) - authors of the project -- `compiler_version` - specifies the version of the compiler to use. This is enforced by the compiler and follow's [Rust's versioning](https://doc.rust-lang.org/cargo/reference/manifest.html#the-version-field), so a `compiler_version = 0.18.0` will enforce Nargo version 0.18.0, `compiler_version = ^0.18.0` will enforce anything above 0.18.0 but below 0.19.0, etc. For more information, see how [Rust handles these operators](https://docs.rs/semver/latest/semver/enum.Op.html) -- `description` (optional) -- `entry` (optional) - a relative filepath to use as the entry point into your package (overrides the default of `src/lib.nr` or `src/main.nr`) -- `backend` (optional) -- `license` (optional) -- `expression_width` (optional) - Sets the default backend expression width. This field will override the default backend expression width specified by the Noir compiler (currently set to width 4). - -#### Dependencies section - -This is where you will specify any dependencies for your project. See the [Dependencies page](../noir/modules_packages_crates/dependencies.md) for more info. - -`./proofs/` and `./contract/` directories will not be immediately visible until you create a proof or -verifier contract respectively. - -### main.nr - -The _main.nr_ file contains a `main` method, this method is the entry point into your Noir program. - -In our sample program, _main.nr_ looks like this: - -```rust -fn main(x : Field, y : Field) { - assert(x != y); -} -``` - -The parameters `x` and `y` can be seen as the API for the program and must be supplied by the prover. Since neither `x` nor `y` is marked as public, the verifier does not supply any inputs, when verifying the proof. - -The prover supplies the values for `x` and `y` in the _Prover.toml_ file. - -As for the program body, `assert` ensures that the condition to be satisfied (e.g. `x != y`) is constrained by the proof of the execution of said program (i.e. if the condition was not met, the verifier would reject the proof as an invalid proof). - -### Prover.toml - -The _Prover.toml_ file is a file which the prover uses to supply the inputs to the Noir program (both private and public). - -In our hello world program the _Prover.toml_ file looks like this: - -```toml -x = "1" -y = "2" -``` - -When the command `nargo execute` is executed, nargo will execute the Noir program using the inputs specified in `Prover.toml`, aborting if it finds that these do not satisfy the constraints defined by `main`. In this example, `x` and `y` must satisfy the inequality constraint `assert(x != y)`. - -If an output name is specified such as `nargo execute foo`, the witness generated by this execution will be written to `./target/foo.gz`. This can then be used to generate a proof of the execution. - -#### Arrays of Structs - -The following code shows how to pass an array of structs to a Noir program to generate a proof. - -```rust -// main.nr -struct Foo { - bar: Field, - baz: Field, -} - -fn main(foos: [Foo; 3]) -> pub Field { - foos[2].bar + foos[2].baz -} -``` - -Prover.toml: - -```toml -[[foos]] # foos[0] -bar = 0 -baz = 0 - -[[foos]] # foos[1] -bar = 0 -baz = 0 - -[[foos]] # foos[2] -bar = 1 -baz = 2 -``` - -#### Custom toml files - -You can specify a `toml` file with a different name to use for execution by using the `--prover-name` or `-p` flags. - -This command looks for proof inputs in the default **Prover.toml** and generates the witness and saves it at `./target/foo.gz`: - -```bash -nargo execute foo -``` - -This command looks for proof inputs in the custom **OtherProver.toml** and generates the witness and saves it at `./target/bar.gz`: - -```bash -nargo execute -p OtherProver bar -``` - -Now that you understand the concepts, you'll probably want some editor feedback while you are writing more complex code. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/quick_start.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/quick_start.md deleted file mode 100644 index 7deeae12fd9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/quick_start.md +++ /dev/null @@ -1,126 +0,0 @@ ---- -title: Quick Start -tags: [] -sidebar_position: 0 ---- - -## Installation - -### Noir - -The easiest way to develop with Noir is using Nargo the CLI tool. It provides you the ability to start new projects, compile, execute and test Noir programs from the terminal. - -You can use `noirup` the installation script to quickly install and update Nargo: - -```bash -curl -L https://raw.githubusercontent.com/noir-lang/noirup/refs/heads/main/install | bash -noirup -``` - -Once installed, you can [set up shell completions for the `nargo` command](setting_up_shell_completions). - -### Proving backend - -After installing Noir, we install a proving backend to work with our Noir programs. - -Proving backends provide you the abilities to generate proofs, verify proofs, generate smart contracts and more for your Noir programs. - -Different proving backends provide different tools for working with Noir programs, here we will use the [Barretenberg proving backend](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg) developed by Aztec Labs as an example. - -You can use the `bbup` installation script to quickly install and update BB, Barretenberg's CLI tool: - -You can find the full list of proving backends compatible with Noir in Awesome Noir. - -```bash -curl -L https://raw.githubusercontent.com/AztecProtocol/aztec-packages/refs/heads/master/barretenberg/bbup/install | bash -bbup -``` - -For the full list of proving backends compatible with Noir, visit [Awesome Noir](https://github.com/noir-lang/awesome-noir/?tab=readme-ov-file#proving-backends). - -## Nargo - -Nargo provides the ability to initiate and execute Noir projects. Let's initialize the traditional `hello_world`: - -```sh -nargo new hello_world -``` - -Two files will be created. - -- `src/main.nr` contains a simple boilerplate circuit -- `Nargo.toml` contains environmental options, such as name, author, dependencies, and others. - -Glancing at _main.nr_ , we can see that inputs in Noir are private by default, but can be labeled public using the keyword `pub`. This means that we will _assert_ that we know a value `x` which is different from `y` without revealing `x`: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -To learn more about private and public values, check the [Data Types](../noir/concepts/data_types/index.md) section. - -### Compiling and executing - -We can now use `nargo` to generate a _Prover.toml_ file, where our input values will be specified: - -```sh -cd hello_world -nargo check - -Let's feed some valid values into this file: - -```toml -x = "1" -y = "2" -``` - -We're now ready to compile and execute our Noir program. By default the `nargo execute` command will do both, and generate the `witness` that we need to feed to our proving backend: - -```sh -nargo execute -``` - -The witness corresponding to this execution will then be written to the file _./target/witness-name.gz_. - -The command also automatically compiles your Noir program if it was not already / was edited, which you may notice the compiled artifacts being written to the file _./target/hello_world.json_. - -With circuit compiled and witness generated, we're ready to prove. - -## Proving backend - -Different proving backends may provide different tools and commands to work with Noir programs. Here Barretenberg's `bb` CLI tool is used as an example: - -```sh -bb prove -b ./target/hello_world.json -w ./target/hello_world.gz -o ./target/proof -``` - -:::tip - -Naming can be confusing, specially as you pass them to the `bb` commands. If unsure, it won't hurt to delete the target folder and start anew to make sure you're using the most recent versions of the compiled circuit and witness. - -::: - -The proof is now generated in the `target` folder. To verify it we first need to compute the verification key from the compiled circuit, and use it to verify: - -```sh -bb write_vk -b ./target/hello_world.json -o ./target/vk -bb verify -k ./target/vk -p ./target/proof -``` - -:::info - -Notice that in order to verify a proof, the verifier knows nothing but the circuit, which is compiled and used to generate the verification key. This is obviously quite important: private inputs remain private. - -As for the public inputs, you may have noticed they haven't been specified. This behavior varies with each particular backend, but barretenberg typically attaches them to the proof. You can see them by parsing and splitting it. For example for if your public inputs are 32 bytes: - -```bash -head -c 32 ./target/proof | od -An -v -t x1 | tr -d $' \n' -``` - -::: - -Congratulations, you have now created and verified a proof for your very first Noir program! - -In the [next section](./project_breakdown.md), we will go into more detail on each step performed. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/setting_up_shell_completions.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/setting_up_shell_completions.md deleted file mode 100644 index 0447321cbab..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/getting_started/setting_up_shell_completions.md +++ /dev/null @@ -1,87 +0,0 @@ ---- -title: Setting up shell completions -tags: [] -sidebar_position: 3 ---- - -The `nargo` binary provides a command to generate shell completions: - -```bash -nargo generate-completion-script [shell] -``` - -where `shell` must be one of `bash`, `elvish`, `fish`, `powershell`, and `zsh`. - -Below we explain how to install them in some popular shells. - -## Installing Zsh Completions - -If you have `oh-my-zsh` installed, you might already have a directory of automatically loading completion scripts — `.oh-my-zsh/completions`. -If not, first create it: - -```bash -mkdir -p ~/.oh-my-zsh/completions` -``` - -Then copy the completion script to that directory: - -```bash -nargo generate-completion-script zsh > ~/.oh-my-zsh/completions/_nargo -``` - -Without `oh-my-zsh`, you’ll need to add a path for completion scripts to your function path, and turn on completion script auto-loading. -First, add these lines to `~/.zshrc`: - -```bash -fpath=(~/.zsh/completions $fpath) -autoload -U compinit -compinit -``` - -Next, create a directory at `~/.zsh/completions`: - -```bash -mkdir -p ~/.zsh/completions -``` - -Then copy the completion script to that directory: - -```bash -nargo generate-completion-script zsh > ~/.zsh/completions/_nargo -``` - -## Installing Bash Completions - -If you have [bash-completion](https://github.com/scop/bash-completion) installed, you can just copy the completion script to the `/usr/local/etc/bash_completion.d` directory: - -```bash -nargo generate-completion-script bash > /usr/local/etc/bash_completion.d/nargo -``` - -Without `bash-completion`, you’ll need to source the completion script directly. -First create a directory such as `~/.bash_completions/`: - -```bash -mkdir ~/.bash_completions/ -``` - -Copy the completion script to that directory: - -```bash -nargo generate-completion-script bash > ~/.bash_completions/nargo.bash -``` - -Then add the following line to `~/.bash_profile` or `~/.bashrc`: - - -```bash -source ~/.bash_completions/nargo.bash -``` - -## Installing Fish Completions - -Copy the completion script to any path listed in the environment variable `$fish_completion_path`. For example, a typical location is `~/.config/fish/completions/nargo.fish`: - -```bash -nargo generate-completion-script fish > ~/.config/fish/completions/nargo.fish -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/_category_.json deleted file mode 100644 index 23b560f610b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/_category_.json deleted file mode 100644 index cc2cbb1c253..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugging", - "position": 5, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/debugging_with_the_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/debugging_with_the_repl.md deleted file mode 100644 index 1d64dae3f37..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/debugging_with_the_repl.md +++ /dev/null @@ -1,164 +0,0 @@ ---- -title: Using the REPL Debugger -description: - Step-by-step guide on how to debug your Noir circuits with the REPL Debugger. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -#### Pre-requisites - -In order to use the REPL debugger, first you need to install recent enough versions of Nargo and vscode-noir. - -## Debugging a simple circuit - -Let's debug a simple circuit: - -```rust -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` - -To start the REPL debugger, using a terminal, go to a Noir circuit's home directory. Then: - -`$ nargo debug` - -You should be seeing this in your terminal: - -``` -[main] Starting debugger -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:9 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> -``` - -The debugger displays the current Noir code location, and it is now waiting for us to drive it. - -Let's first take a look at the available commands. For that we'll use the `help` command. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -Some commands operate only for unconstrained functions, such as `memory` and `memset`. If you try to use them while execution is paused at an ACIR opcode, the debugger will simply inform you that you are not executing unconstrained code: - -``` -> memory -Unconstrained VM memory not available -> -``` - -Before continuing, we can take a look at the initial witness map: - -``` -> witness -_0 = 1 -_1 = 2 -> -``` - -Cool, since `x==1`, `y==2`, and we want to check that `x != y`, our circuit should succeed. At this point we could intervene and use the witness setter command to change one of the witnesses. Let's set `y=3`, then back to 2, so we don't affect the expected result: - -``` -> witness -_0 = 1 -_1 = 2 -> witness 1 3 -_1 = 3 -> witness -_0 = 1 -_1 = 3 -> witness 1 2 -_1 = 2 -> witness -_0 = 1 -_1 = 2 -> -``` - -Now we can inspect the current state of local variables. For that we use the `vars` command. - -``` -> vars -> -``` - -We currently have no vars in context, since we are at the entry point of the program. Let's use `next` to execute until the next point in the program. - -``` -> vars -> next -At ~/noir-examples/recursion/circuits/main/src/main.nr:1:20 - 1 -> fn main(x : Field, y : pub Field) { - 2 assert(x != y); - 3 } -> vars -x:Field = 0x01 -``` - -As a result of stepping, the variable `x`, whose initial value comes from the witness map, is now in context and returned by `vars`. - -``` -> next - 1 fn main(x : Field, y : pub Field) { - 2 -> assert(x != y); - 3 } -> vars -y:Field = 0x02 -x:Field = 0x01 -``` - -Stepping again we can finally see both variables and their values. And now we can see that the next assertion should succeed. - -Let's continue to the end: - -``` -> continue -(Continuing execution...) -Finished execution -> q -[main] Circuit witness successfully solved -``` - -Upon quitting the debugger after a solved circuit, the resulting circuit witness gets saved, equivalent to what would happen if we had run the same circuit with `nargo execute`. - -We just went through the basics of debugging using Noir REPL debugger. For a comprehensive reference, check out [the reference page](../../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/debugging_with_vs_code.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/debugging_with_vs_code.md deleted file mode 100644 index a5858c1a5eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/debugger/debugging_with_vs_code.md +++ /dev/null @@ -1,68 +0,0 @@ ---- -title: Using the VS Code Debugger -description: - Step by step guide on how to debug your Noir circuits with the VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -This guide will show you how to use VS Code with the vscode-noir extension to debug a Noir project. - -#### Pre-requisites - -- Nargo -- vscode-noir -- A Noir project with a `Nargo.toml`, `Prover.toml` and at least one Noir (`.nr`) containing an entry point function (typically `main`). - -## Running the debugger - -The easiest way to start debugging is to open the file you want to debug, and press `F5`. This will cause the debugger to launch, using your `Prover.toml` file as input. - -You should see something like this: - -![Debugger launched](@site/static/img/debugger/1-started.png) - -Let's inspect the state of the program. For that, we open VS Code's _Debug pane_. Look for this icon: - -![Debug pane icon](@site/static/img/debugger/2-icon.png) - -You will now see two categories of variables: Locals and Witness Map. - -![Debug pane expanded](@site/static/img/debugger/3-debug-pane.png) - -1. **Locals**: variables of your program. At this point in execution this section is empty, but as we step through the code it will get populated by `x`, `result`, `digest`, etc. - -2. **Witness map**: these are initially populated from your project's `Prover.toml` file. In this example, they will be used to populate `x` and `result` at the beginning of the `main` function. - -Most of the time you will probably be focusing mostly on locals, as they represent the high level state of your program. - -You might be interested in inspecting the witness map in case you are trying to solve a really low level issue in the compiler or runtime itself, so this concerns mostly advanced or niche users. - -Let's step through the program, by using the debugger buttons or their corresponding keyboard shortcuts. - -![Debugger buttons](@site/static/img/debugger/4-debugger-buttons.png) - -Now we can see in the variables pane that there's values for `digest`, `result` and `x`. - -![Inspecting locals](@site/static/img/debugger/5-assert.png) - -We can also inspect the values of variables by directly hovering on them on the code. - -![Hover locals](@site/static/img/debugger/6-hover.png) - -Let's set a break point at the `keccak256` function, so we can continue execution up to the point when it's first invoked without having to go one step at a time. - -We just need to click the to the right of the line number 18. Once the breakpoint appears, we can click the `continue` button or use its corresponding keyboard shortcut (`F5` by default). - -![Breakpoint](@site/static/img/debugger/7-break.png) - -Now we are debugging the `keccak256` function, notice the _Call Stack pane_ at the lower right. This lets us inspect the current call stack of our process. - -That covers most of the current debugger functionalities. Check out [the reference](../../reference/debugger/debugger_vscode.md) for more details on how to configure the debugger. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/how-to-solidity-verifier.md deleted file mode 100644 index 2cc0f8e57ce..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/how-to-solidity-verifier.md +++ /dev/null @@ -1,259 +0,0 @@ ---- -title: Generate a Solidity Verifier -description: - Learn how to run the verifier as a smart contract on the blockchain. Compile a Solidity verifier - contract for your Noir program and deploy it on any EVM blockchain acting as a verifier smart - contract. Read more to find out -keywords: - [ - solidity verifier, - smart contract, - blockchain, - compiler, - plonk_vk.sol, - EVM blockchain, - verifying Noir programs, - proving backend, - Barretenberg, - ] -sidebar_position: 0 -pagination_next: tutorials/noirjs_app ---- - -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. - -This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. - -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: - -- You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network -- You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/quick_start.md) with Nargo and the example Hello Noir circuit -- You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. - -## Rundown - -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: - -1. How to generate a solidity smart contract -2. How to compile the smart contract in the RemixIDE -3. How to deploy it to a testnet - -## Step 1 - Generate a contract - -This is by far the most straightforward step. Just run: - -```sh -nargo compile -``` - -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: - -```sh -# Here we pass the path to the newly generated Noir artifact. -bb write_vk -b ./target/.json -bb contract -``` - -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. - -You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/quick_start.md#proving-backend). - -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. - -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: - -## Step 2 - Compiling - -We will mostly skip the details of RemixIDE, as the UI can change from version to version. For now, we can just open -Remix and create a blank workspace. - -![Create Workspace](@site/static/img/how-tos/solidity_verifier_1.png) - -We will create a new file to contain the contract Nargo generated, and copy-paste its content. - -:::warning - -You'll likely see a warning advising you to not trust pasted code. While it is an important warning, it is irrelevant in the context of this guide and can be ignored. We will not be deploying anywhere near a mainnet. - -::: - -To compile our the verifier, we can navigate to the compilation tab: - -![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) - -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: - -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) - -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: - -![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) - -## Step 3 - Deploying - -At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. - -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": - -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) - -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. - -:::note - -Why "UltraVerifier"? - -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. - -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: - -## Step 4 - Verifying - -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: - -```solidity -function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) -``` - -When using the default example in the [Hello Noir](../getting_started/quick_start.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. - -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/quick_start.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the - -```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') - -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS - -echo "Proof:" -echo "0x$HEX_PROOF" -``` - -Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. - -A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): - -```solidity -function castVote(bytes calldata proof, uint proposalId, uint vote, bytes32 nullifierHash) public returns (bool) { - // ... - bytes32[] memory publicInputs = new bytes32[](4); - publicInputs[0] = merkleRoot; - publicInputs[1] = bytes32(proposalId); - publicInputs[2] = bytes32(vote); - publicInputs[3] = nullifierHash; - require(verifier.verify(proof, publicInputs), "Invalid proof"); -``` - -:::info[Return Values] - -A circuit doesn't have the concept of a return value. Return values are just syntactic sugar in Noir. - -Under the hood, the return value is passed as an input to the circuit and is checked at the end of the circuit program. - -For example, if you have Noir program like this: - -```rust -fn main( - // Public inputs - pubkey_x: pub Field, - pubkey_y: pub Field, - // Private inputs - priv_key: Field, -) -> pub Field -``` - -the `verify` function will expect the public inputs array (second function parameter) to be of length 3, the two inputs and the return value. - -Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. - -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. - -::: - -:::tip[Structs] - -You can pass structs to the verifier contract. They will be flattened so that the array of inputs is 1-dimensional array. - -For example, consider the following program: - -```rust -struct Type1 { - val1: Field, - val2: Field, -} - -struct Nested { - t1: Type1, - is_true: bool, -} - -fn main(x: pub Field, nested: pub Nested, y: pub Field) { - //... -} -``` - -The order of these inputs would be flattened to: `[x, nested.t1.val1, nested.t1.val2, nested.is_true, y]` - -::: - -The other function you can call is our entrypoint `verify` function, as defined above. - -:::tip - -It's worth noticing that the `verify` function is actually a `view` function. A `view` function does not alter the blockchain state, so it doesn't need to be distributed (i.e. it will run only on the executing node), and therefore doesn't cost any gas. - -This can be particularly useful in some situations. If Alice generated a proof and wants Bob to verify its correctness, Bob doesn't need to run Nargo, NoirJS, or any Noir specific infrastructure. He can simply make a call to the blockchain with the proof and verify it is correct without paying any gas. - -It would be incorrect to say that a Noir proof verification costs any gas at all. However, most of the time the result of `verify` is used to modify state (for example, to update a balance, a game state, etc). In that case the whole network needs to execute it, which does incur gas costs (calldata and execution, but not storage). - -::: - -## A Note on EVM chains - -Noir proof verification requires the ecMul, ecAdd and ecPairing precompiles. Not all EVM chains support EC Pairings, notably some of the ZK-EVMs. This means that you won't be able to use the verifier contract in all of them. You can find an incomplete list of which EVM chains support these precompiles [here](https://www.evmdiff.com/features?feature=precompiles). - -For example, chains like `zkSync ERA` and `Polygon zkEVM` do not currently support these precompiles, so proof verification via Solidity verifier contracts won't work. Here's a quick list of EVM chains that have been tested and are known to work: - -- Optimism -- Arbitrum -- Polygon PoS -- Scroll -- Celo -- BSC -- Blast L2 -- Avalanche C-Chain -- Mode -- Linea -- Moonbeam - -If you test any other chains, please open a PR on this page to update the list. See [this doc](https://github.com/noir-lang/noir-starter/tree/main/with-foundry#testing-on-chain) for more info about testing verifier contracts on different EVM chains. - -## What's next - -Now that you know how to call a Noir Solidity Verifier on a smart contract using Remix, you should be comfortable with using it with some programmatic frameworks, such as [hardhat](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) and [foundry](https://github.com/noir-lang/noir-starter/tree/main/with-foundry). - -You can find other tools, examples, boilerplates and libraries in the [awesome-noir](https://github.com/noir-lang/awesome-noir) repository. - -You should also be ready to write and deploy your first NoirJS app and start generating proofs on websites, phones, and NodeJS environments! Head on to the [NoirJS tutorial](../tutorials/noirjs_app.md) to learn how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/merkle-proof.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/merkle-proof.mdx deleted file mode 100644 index 0a128adb2de..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/merkle-proof.mdx +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Prove Merkle Tree Membership -description: - Learn how to use merkle membership proof in Noir to prove that a given leaf is a member of a - merkle tree with a specified root, at a given index. -keywords: - [merkle proof, merkle membership proof, Noir, rust, hash function, Pedersen, sha256, merkle tree] -sidebar_position: 4 ---- - -Let's walk through an example of a merkle membership proof in Noir that proves that a given leaf is -in a merkle tree. - -```rust - -fn main(message : [Field; 62], index : Field, hashpath : [Field; 40], root : Field) { - let leaf = std::hash::hash_to_field(message.as_slice()); - let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); - assert(merkle_root == root); -} - -``` - -The message is hashed using `hash_to_field`. The specific hash function that is being used is chosen -by the backend. The only requirement is that this hash function can heuristically be used as a -random oracle. If only collision resistance is needed, then one can call `std::hash::pedersen_hash` -instead. - -```rust -let leaf = std::hash::hash_to_field(message.as_slice()); -``` - -The leaf is then passed to a compute_merkle_root function with the root, index and hashpath. The returned root can then be asserted to be the same as the provided root. - -```rust -let merkle_root = std::merkle::compute_merkle_root(leaf, index, hashpath); -assert (merkle_root == root); -``` - -> **Note:** It is possible to re-implement the merkle tree implementation without standard library. -> However, for most usecases, it is enough. In general, the standard library will always opt to be -> as conservative as possible, while striking a balance with efficiency. - -An example, the merkle membership proof, only requires a hash function that has collision -resistance, hence a hash function like Pedersen is allowed, which in most cases is more efficient -than the even more conservative sha256. - -[View an example on the starter repo](https://github.com/noir-lang/noir-examples/blob/3ea09545cabfa464124ec2f3ea8e60c608abe6df/stealthdrop/circuits/src/main.nr#L20) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/using-devcontainers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/using-devcontainers.mdx deleted file mode 100644 index 727ec6ca667..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/using-devcontainers.mdx +++ /dev/null @@ -1,110 +0,0 @@ ---- -title: Developer Containers and Codespaces -description: "Learn how to set up a devcontainer in your GitHub repository for a seamless coding experience with Codespaces. Follow our easy 8-step guide to create your own Noir environment without installing Nargo locally." -keywords: ["Devcontainer", "Codespaces", "GitHub", "Noir Environment", "Docker Image", "Development Environment", "Remote Coding", "GitHub Codespaces", "Noir Programming", "Nargo", "VSCode Extensions", "Noirup"] -sidebar_position: 1 ---- - -Adding a developer container configuration file to your Noir project is one of the easiest way to unlock coding in browser. - -## What's a devcontainer after all? - -A [Developer Container](https://containers.dev/) (devcontainer for short) is a Docker image that comes preloaded with tools, extensions, and other tools you need to quickly get started or continue a project, without having to install Nargo locally. Think of it as a development environment in a box. - -There are many advantages to this: - -- It's platform and architecture agnostic -- You don't need to have an IDE installed, or Nargo, or use a terminal at all -- It's safer for using on a public machine or public network - -One of the best ways of using devcontainers is... not using your machine at all, for maximum control, performance, and ease of use. -Enter Codespaces. - -## Codespaces - -If a devcontainer is just a Docker image, then what stops you from provisioning a `p3dn.24xlarge` AWS EC2 instance with 92 vCPUs and 768 GiB RAM and using it to prove your 10-gate SNARK proof? - -Nothing! Except perhaps the 30-40$ per hour it will cost you. - -The problem is that provisioning takes time, and I bet you don't want to see the AWS console every time you want to code something real quick. - -Fortunately, there's an easy and free way to get a decent remote machine ready and loaded in less than 2 minutes: Codespaces. [Codespaces is a Github feature](https://github.com/features/codespaces) that allows you to code in a remote machine by using devcontainers, and it's pretty cool: - -- You can start coding Noir in less than a minute -- It uses the resources of a remote machine, so you can code on your grandma's phone if needed be -- It makes it easy to share work with your frens -- It's fully reusable, you can stop and restart whenever you need to - -:::info - -Don't take out your wallet just yet. Free GitHub accounts get about [15-60 hours of coding](https://github.com/features/codespaces) for free per month, depending on the size of your provisioned machine. - -::: - -## Tell me it's _actually_ easy - -It is! - -Github comes with a default codespace and you can use it to code your own devcontainer. That's exactly what we will be doing in this guide. - - - -8 simple steps: - -#### 1. Create a new repository on GitHub. - -#### 2. Click "Start coding with Codespaces". This will use the default image. - -#### 3. Create a folder called `.devcontainer` in the root of your repository. - -#### 4. Create a Dockerfile in that folder, and paste the following code: - -```docker -FROM --platform=linux/amd64 node:lts-bookworm-slim -SHELL ["/bin/bash", "-c"] -RUN apt update && apt install -y curl bash git tar gzip libc++-dev -RUN curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -ENV PATH="/root/.nargo/bin:$PATH" -RUN noirup -ENTRYPOINT ["nargo"] -``` -#### 5. Create a file called `devcontainer.json` in the same folder, and paste the following code: - -```json -{ - "name": "Noir on Codespaces", - "build": { - "context": ".", - "dockerfile": "Dockerfile" - }, - "customizations": { - "vscode": { - "extensions": ["noir-lang.vscode-noir"] - } - } -} -``` -#### 6. Commit and push your changes - -This will pull the new image and build it, so it could take a minute or so - -#### 8. Done! -Just wait for the build to finish, and there's your easy Noir environment. - - -Refer to [noir-starter](https://github.com/noir-lang/noir-starter/) as an example of how devcontainers can be used together with codespaces. - - - -## How do I use it? - -Using the codespace is obviously much easier than setting it up. -Just navigate to your repository and click "Code" -> "Open with Codespaces". It should take a few seconds to load, and you're ready to go. - -:::info - -If you really like the experience, you can add a badge to your readme, links to existing codespaces, and more. -Check out the [official docs](https://docs.github.com/en/codespaces/setting-up-your-project-for-codespaces/setting-up-your-repository/facilitating-quick-creation-and-resumption-of-codespaces) for more info. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/index.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/index.mdx deleted file mode 100644 index 5c116a73b3f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/index.mdx +++ /dev/null @@ -1,76 +0,0 @@ ---- -title: Noir Lang -hide_title: true -description: - Learn about the public alpha release of Noir, a domain specific language heavily influenced by Rust that compiles to - an intermediate language which can be compiled to an arithmetic circuit or a rank-1 constraint system. -keywords: - [Noir, - Domain Specific Language, - Rust, - Intermediate Language, - Arithmetic Circuit, - Rank-1 Constraint System, - Ethereum Developers, - Protocol Developers, - Blockchain Developers, - Proving System, - Smart Contract Language] -sidebar_position: 0 ---- - -import Tabs from '@theme/Tabs'; -import TabItem from '@theme/TabItem'; -import ThemedImage from '@theme/ThemedImage'; -import useBaseUrl from '@docusaurus/useBaseUrl'; - - - -Noir is an open-source Domain-Specific Language for safe and seamless construction of privacy-preserving Zero-Knowledge programs, requiring no previous knowledge on the underlying mathematics or cryptography. - -ZK programs are programs that can generate short proofs of statements without revealing all inputs to the statements. You can read more about Zero-Knowledge Proofs [here](https://dev.to/spalladino/a-beginners-intro-to-coding-zero-knowledge-proofs-c56). - -## What's new about Noir? - -Noir works differently from most ZK languages by taking a two-pronged path. First, it compiles the program to an adaptable intermediate language known as ACIR. From there, depending on a given project's needs, ACIR can be further compiled into an arithmetic circuit for integration with the proving backend. - -:::info - -Noir is backend agnostic, which means it makes no assumptions on which proving backend powers the ZK proof. Being the language that powers [Aztec Contracts](https://docs.aztec.network/developers/contracts/main), it defaults to Aztec's Barretenberg proving backend. - -However, the ACIR output can be transformed to be compatible with other PLONK-based backends, or into a [rank-1 constraint system](https://www.rareskills.io/post/rank-1-constraint-system) suitable for backends such as Arkwork's Marlin. - -::: - -## Who is Noir for? - -Noir can be used both in complex cloud-based backends and in user's smartphones, requiring no knowledge on the underlying math or cryptography. From authorization systems that keep a password in the user's device, to complex on-chain verification of recursive proofs, Noir is designed to abstract away complexity without any significant overhead. Here are some examples of situations where Noir can be used: - - - - Noir Logo - - Aztec Contracts leverage Noir to allow for the storage and execution of private information. Writing an Aztec Contract is as easy as writing Noir, and Aztec developers can easily interact with the network storage and execution through the [Aztec.nr](https://docs.aztec.network/developers/contracts/main) library. - - - Soliditry Verifier Example - Noir can auto-generate Solidity verifier contracts that verify Noir proofs. This allows for non-interactive verification of proofs containing private information in an immutable system. This feature powers a multitude of use-case scenarios, from P2P chess tournaments, to [Aztec Layer-2 Blockchain](https://docs.aztec.network/) - - - Aztec Labs developed NoirJS, an easy interface to generate and verify Noir proofs in a Javascript environment. This allows for Noir to be used in webpages, mobile apps, games, and any other environment supporting JS execution in a standalone manner. - - - - -## Libraries - -Noir is meant to be easy to extend by simply importing Noir libraries just like in Rust. -The [awesome-noir repo](https://github.com/noir-lang/awesome-noir#libraries) is a collection of libraries developed by the Noir community. -Writing a new library is easy and makes code be composable and easy to reuse. See the section on [dependencies](noir/modules_packages_crates/dependencies.md) for more information. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/migration_notes.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/migration_notes.md deleted file mode 100644 index 6bd740024e5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/migration_notes.md +++ /dev/null @@ -1,105 +0,0 @@ ---- -title: Migration notes -description: Read about migration notes from previous versions, which could solve problems while updating -keywords: [Noir, notes, migration, updating, upgrading] ---- - -Noir is in full-speed development. Things break fast, wild, and often. This page attempts to leave some notes on errors you might encounter when upgrading and how to resolve them until proper patches are built. - -### `backend encountered an error: libc++.so.1` - -Depending on your OS, you may encounter the following error when running `nargo prove` for the first time: - -```text -The backend encountered an error: "/home/codespace/.nargo/backends/acvm-backend-barretenberg/backend_binary: error while loading shared libraries: libc++.so.1: cannot open shared object file: No such file or directory\n" -``` - -Install the `libc++-dev` library with: - -```bash -sudo apt install libc++-dev -``` - -## ≥0.19 - -### Enforcing `compiler_version` - -From this version on, the compiler will check for the `compiler_version` field in `Nargo.toml`, and will error if it doesn't match the current Nargo version in use. - -To update, please make sure this field in `Nargo.toml` matches the output of `nargo --version`. - -## ≥0.14 - -The index of the [for loops](noir/concepts/control_flow.md#loops) is now of type `u64` instead of `Field`. An example refactor would be: - -```rust -for i in 0..10 { - let i = i as Field; -} -``` - -## ≥v0.11.0 and Nargo backend - -From this version onwards, Nargo starts managing backends through the `nargo backend` command. Upgrading to the versions per usual steps might lead to: - -### `backend encountered an error` - -This is likely due to the existing locally installed version of proving backend (e.g. barretenberg) is incompatible with the version of Nargo in use. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo prove -``` - -with your Noir program. - -This will trigger the download and installation of the latest version of barretenberg compatible with your Nargo in use. - -### `backend encountered an error: illegal instruction` - -On certain Intel-based systems, an `illegal instruction` error may arise due to incompatibility of barretenberg with certain CPU instructions. - -To fix the issue: - -1. Uninstall the existing backend - -```bash -nargo backend uninstall acvm-backend-barretenberg -``` - -You may replace _acvm-backend-barretenberg_ with the name of your backend listed in `nargo backend ls` or in ~/.nargo/backends. - -2. Reinstall a compatible version of the proving backend. - -If you are using the default barretenberg backend, simply run: - -``` -nargo backend install acvm-backend-barretenberg https://github.com/noir-lang/barretenberg-js-binary/raw/master/run-bb.tar.gz -``` - -This downloads and installs a specific bb.js based version of barretenberg binary from GitHub. - -The gzipped file is running [this bash script](https://github.com/noir-lang/barretenberg-js-binary/blob/master/run-bb-js.sh), where we need to gzip it as the Nargo currently expect the backend to be zipped up. - -Then run: - -``` -DESIRED_BINARY_VERSION=0.8.1 nargo info -``` - -This overrides the bb native binary with a bb.js node application instead, which should be compatible with most if not all hardware. This does come with the drawback of being generally slower than native binary. - -0.8.1 indicates bb.js version 0.8.1, so if you change that it will update to a different version or the default version in the script if none was supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/_category_.json deleted file mode 100644 index 7da08f8a8c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Concepts", - "position": 0, - "collapsible": true, - "collapsed": true -} \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/assert.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/assert.md deleted file mode 100644 index 2132de42072..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/assert.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Assert Function -description: - Learn about the `assert` and `static_assert` functions in Noir, which can be used to explicitly - constrain the predicate or comparison expression that follows to be true, and what happens if - the expression is false at runtime or compile-time, respectively. -keywords: [Noir programming language, assert statement, predicate expression, comparison expression] -sidebar_position: 4 ---- - -Noir includes a special `assert` function which will explicitly constrain the predicate/comparison -expression that follows to be true. If this expression is false at runtime, the program will fail to -be proven. Example: - -```rust -fn main(x : Field, y : Field) { - assert(x == y); -} -``` - -> Assertions only work for predicate operations, such as `==`. If there's any ambiguity on the operation, the program will fail to compile. For example, it is unclear if `assert(x + y)` would check for `x + y == 0` or simply would return `true`. - -You can optionally provide a message to be logged when the assertion fails: - -```rust -assert(x == y, "x and y are not equal"); -``` - -Aside string literals, the optional message can be a format string or any other type supported as input for Noir's [print](../standard_library/logging.md) functions. This feature lets you incorporate runtime variables into your failed assertion logs: - -```rust -assert(x == y, f"Expected x == y, but got {x} == {y}"); -``` - -Using a variable as an assertion message directly: - -```rust -struct myStruct { - myField: Field -} - -let s = myStruct { myField: y }; -assert(s.myField == x, s); -``` - -There is also a special `static_assert` function that behaves like `assert`, -but that runs at compile-time. - -```rust -fn main(xs: [Field; 3]) { - let x = 2 + 2; - let y = 4; - static_assert(x == y, "expected 2 + 2 to equal 4"); - - // This passes since the length of `xs` is known at compile-time - static_assert(xs.len() == 3, "expected the input to have 3 elements"); -} -``` - -This function fails when passed a dynamic (run-time) argument: - -```rust -fn main(x : Field, y : Field) { - // this fails because `x` is not known at compile-time - static_assert(x == 2, "expected x to be known at compile-time and equal to 2"); - - let mut example_slice = &[]; - if y == 4 { - example_slice = example_slice.push_back(0); - } - - // This fails because the length of `example_slice` is not known at - // compile-time - let error_message = "expected an empty slice, known at compile-time"; - static_assert(example_slice.len() == 0, error_message); -} -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/comments.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/comments.md deleted file mode 100644 index b51a85f5c94..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/comments.md +++ /dev/null @@ -1,33 +0,0 @@ ---- -title: Comments -description: - Learn how to write comments in Noir programming language. A comment is a line of code that is - ignored by the compiler, but it can be read by programmers. Single-line and multi-line comments - are supported in Noir. -keywords: [Noir programming language, comments, single-line comments, multi-line comments] -sidebar_position: 10 ---- - -A comment is a line in your codebase which the compiler ignores, however it can be read by -programmers. - -Here is a single line comment: - -```rust -// This is a comment and is ignored -``` - -`//` is used to tell the compiler to ignore the rest of the line. - -Noir also supports multi-line block comments. Start a block comment with `/*` and end the block with `*/`. - -Noir does not natively support doc comments. You may be able to use [Rust doc comments](https://doc.rust-lang.org/reference/comments.html) in your code to leverage some Rust documentation build tools with Noir code. - -```rust -/* - This is a block comment describing a complex function. -*/ -fn main(x : Field, y : pub Field) { - assert(x != y); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/comptime.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/comptime.md deleted file mode 100644 index 2ceb030c7e1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/comptime.md +++ /dev/null @@ -1,445 +0,0 @@ ---- -title: Compile-time Code & Metaprogramming -description: Learn how to use metaprogramming in Noir to create macros or derive your own traits -keywords: [Noir, comptime, compile-time, metaprogramming, macros, quote, unquote] -sidebar_position: 15 ---- - -## Overview - -Metaprogramming in Noir is comprised of three parts: -1. `comptime` code -2. Quoting and unquoting -3. The metaprogramming API in `std::meta` - -Each of these are explained in more detail in the next sections but the wide picture is that -`comptime` allows us to write code which runs at compile-time. In this `comptime` code we -can quote and unquote snippets of the program, manipulate them, and insert them in other -parts of the program. Comptime functions which do this are said to be macros. Additionally, -there's a compile-time API of built-in types and functions provided by the compiler which allows -for greater analysis and modification of programs. - ---- - -## Comptime - -`comptime` is a new keyword in Noir which marks an item as executing or existing at compile-time. It can be used in several ways: - -- `comptime fn` to define functions which execute exclusively during compile-time. -- `comptime global` to define a global variable which is evaluated at compile-time. - - Unlike runtime globals, `comptime global`s can be mutable. -- `comptime { ... }` to execute a block of statements during compile-time. -- `comptime let` to define a variable whose value is evaluated at compile-time. -- `comptime for` to run a for loop at compile-time. Syntax sugar for `comptime { for .. }`. - -### Scoping - -Note that while in a `comptime` context, any runtime variables _local to the current function_ are never visible. - -### Evaluating - -Evaluation rules of `comptime` follows the normal unconstrained evaluation rules for other Noir code. There are a few things to note though: - -- Certain built-in functions may not be available, although more may be added over time. -- Evaluation order of global items is currently unspecified. For example, given the following two functions we can't guarantee -which `println` will execute first. The ordering of the two printouts will be arbitrary, but should be stable across multiple compilations with the same `nargo` version as long as the program is also unchanged. - -```rust -fn one() { - comptime { println("one"); } -} - -fn two() { - comptime { println("two"); } -} -``` - -- Since evaluation order is unspecified, care should be taken when using mutable globals so that they do not rely on a particular ordering. -For example, using globals to generate unique ids should be fine but relying on certain ids always being produced (especially after edits to the program) should be avoided. -- Although most ordering of globals is unspecified, two are: - - Dependencies of a crate will always be evaluated before the dependent crate. - - Any annotations on a function will be run before the function itself is resolved. This is to allow the annotation to modify the function if necessary. Note that if the - function itself was called at compile-time previously, it will already be resolved and cannot be modified. To prevent accidentally calling functions you wish to modify - at compile-time, it may be helpful to sort your `comptime` annotation functions into a different crate along with any dependencies they require. - -### Lowering - -When a `comptime` value is used in runtime code it must be lowered into a runtime value. This means replacing the expression with the literal that it evaluated to. For example, the code: - -```rust -struct Foo { array: [Field; 2], len: u32 } - -fn main() { - println(comptime { - let mut foo = std::mem::zeroed::(); - foo.array[0] = 4; - foo.len = 1; - foo - }); -} -``` - -will be converted to the following after `comptime` expressions are evaluated: - -```rust -struct Foo { array: [Field; 2], len: u32 } - -fn main() { - println(Foo { array: [4, 0], len: 1 }); -} -``` - -Not all types of values can be lowered. For example, `Type`s and `TypeDefinition`s (among other types) cannot be lowered at all. - -```rust -fn main() { - // There's nothing we could inline here to create a Type value at runtime - // let _ = get_type!(); -} - -comptime fn get_type() -> Type { ... } -``` - ---- - -## (Quasi) Quote - -Macros in Noir are `comptime` functions which return code as a value which is inserted into the call site when it is lowered there. -A code value in this case is of type `Quoted` and can be created by a `quote { ... }` expression. -More specifically, the code value `quote` creates is a token stream - a representation of source code as a series of words, numbers, string literals, or operators. -For example, the expression `quote { Hi "there reader"! }` would quote three tokens: the word "hi", the string "there reader", and an exclamation mark. -You'll note that snippets that would otherwise be invalid syntax can still be quoted. - -When a `Quoted` value is used in runtime code, it is lowered into a `quote { ... }` expression. Since this expression is only valid -in compile-time code however, we'd get an error if we tried this. Instead, we can use macro insertion to insert each token into the -program at that point, and parse it as an expression. To do this, we have to add a `!` after the function name returning the `Quoted` value. -If the value was created locally and there is no function returning it, `std::meta::unquote!(_)` can be used instead. -Calling such a function at compile-time without `!` will just return the `Quoted` value to be further manipulated. For example: - -```rust title="quote-example" showLineNumbers -comptime fn quote_one() -> Quoted { - quote { 1 } - } - - #[test] - fn returning_versus_macro_insertion() { - comptime { - // let _a: Quoted = quote { 1 }; - let _a: Quoted = quote_one(); - - // let _b: Field = 1; - let _b: Field = quote_one!(); - - // Since integers default to fields, if we - // want a different type we have to explicitly cast - // let _c: i32 = 1 as i32; - let _c: i32 = quote_one!() as i32; - } - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L120-L140 - - -For those familiar with quoting from other languages (primarily lisps), Noir's `quote` is actually a _quasiquote_. -This means we can escape the quoting by using the unquote operator to splice values in the middle of quoted code. - -## Unquote - -The unquote operator `$` is usable within a `quote` expression. -It takes a variable as an argument, evaluates the variable, and splices the resulting value into the quoted token stream at that point. For example, - -```rust -comptime { - let x = 1 + 2; - let y = quote { $x + 4 }; -} -``` - -The value of `y` above will be the token stream containing `3`, `+`, and `4`. We can also use this to combine `Quoted` values into larger token streams: - -```rust -comptime { - let x = quote { 1 + 2 }; - let y = quote { $x + 4 }; -} -``` - -The value of `y` above is now the token stream containing five tokens: `1 + 2 + 4`. - -Note that to unquote something, a variable name _must_ follow the `$` operator in a token stream. -If it is an expression (even a parenthesized one), it will do nothing. Most likely a parse error will be given when the macro is later unquoted. - -Unquoting can also be avoided by escaping the `$` with a backslash: - -``` -comptime { - let x = quote { 1 + 2 }; - - // y contains the four tokens: `$x + 4` - let y = quote { \$x + 4 }; -} -``` - ---- - -## Annotations - -Annotations provide a way to run a `comptime` function on an item in the program. -When you use an annotation, the function with the same name will be called with that item as an argument: - -```rust -#[my_struct_annotation] -struct Foo {} - -comptime fn my_struct_annotation(s: StructDefinition) { - println("Called my_struct_annotation!"); -} - -#[my_function_annotation] -fn foo() {} - -comptime fn my_function_annotation(f: FunctionDefinition) { - println("Called my_function_annotation!"); -} -``` - -Anything returned from one of these functions will be inserted at top-level along with the original item. -Note that expressions are not valid at top-level so you'll get an error trying to return `3` or similar just as if you tried to write a program containing `3; struct Foo {}`. -You can insert other top-level items such as trait impls, structs, or functions this way though. -For example, this is the mechanism used to insert additional trait implementations into the program when deriving a trait impl from a struct: - -```rust title="derive-field-count-example" showLineNumbers -trait FieldCount { - fn field_count() -> u32; - } - - #[derive_field_count] - struct Bar { - x: Field, - y: [Field; 2], - } - - comptime fn derive_field_count(s: StructDefinition) -> Quoted { - let typ = s.as_type(); - let field_count = s.fields().len(); - quote { - impl FieldCount for $typ { - fn field_count() -> u32 { - $field_count - } - } - } - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L142-L164 - - -### Calling annotations with additional arguments - -Arguments may optionally be given to annotations. -When this is done, these additional arguments are passed to the annotation function after the item argument. - -```rust title="annotation-arguments-example" showLineNumbers -#[assert_field_is_type(quote { i32 }.as_type())] - struct MyStruct { - my_field: i32, - } - - comptime fn assert_field_is_type(s: StructDefinition, typ: Type) { - // Assert the first field in `s` has type `typ` - let fields = s.fields(); - assert_eq(fields[0].1, typ); - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L166-L177 - - -We can also take any number of arguments by adding the `varargs` annotation: - -```rust title="annotation-varargs-example" showLineNumbers -#[assert_three_args(1, 2, 3)] - struct MyOtherStruct { - my_other_field: u32, - } - - #[varargs] - comptime fn assert_three_args(_s: StructDefinition, args: [Field]) { - assert_eq(args.len(), 3); - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L179-L189 - - ---- - -## Comptime API - -Although `comptime`, `quote`, and unquoting provide a flexible base for writing macros, -Noir's true metaprogramming ability comes from being able to interact with the compiler through a compile-time API. -This API can be accessed through built-in functions in `std::meta` as well as on methods of several `comptime` types. - -The following is an incomplete list of some `comptime` types along with some useful methods on them. You can see more in the standard library [Metaprogramming section](../standard_library/meta). - -- `Quoted`: A token stream -- `Type`: The type of a Noir type - - `fn implements(self, constraint: TraitConstraint) -> bool` - - Returns true if `self` implements the given trait constraint -- `Expr`: A syntactically valid expression. Can be used to recur on a program's parse tree to inspect how it is structured. - - Methods: - - `fn as_function_call(self) -> Option<(Expr, [Expr])>` - - If this is a function call expression, return `(function, arguments)` - - `fn as_block(self) -> Option<[Expr]>` - - If this is a block, return each statement in the block -- `FunctionDefinition`: A function definition - - Methods: - - `fn parameters(self) -> [(Quoted, Type)]` - - Returns a slice of `(name, type)` pairs for each parameter -- `StructDefinition`: A struct definition - - Methods: - - `fn as_type(self) -> Type` - - Returns this `StructDefinition` as a `Type`. Any generics are kept as-is - - `fn generics(self) -> [Quoted]` - - Return the name of each generic on this struct - - `fn fields(self) -> [(Quoted, Type)]` - - Return the name and type of each field -- `TraitConstraint`: A trait constraint such as `From` -- `TypedExpr`: A type-checked expression. -- `UnresolvedType`: A syntactic notation that refers to a Noir type that hasn't been resolved yet - -There are many more functions available by exploring the `std::meta` module and its submodules. -Using these methods is the key to writing powerful metaprogramming libraries. - -### `#[use_callers_scope]` - -Since certain functions such as `Quoted::as_type`, `Expression::as_type`, or `Quoted::as_trait_constraint` will attempt -to resolve their contents in a particular scope - it can be useful to change the scope they resolve in. By default -these functions will resolve in the current function's scope which is usually the attribute function they are called in. -If you're working on a library however, this may be a completely different module or crate to the item you're trying to -use the attribute on. If you want to be able to use `Quoted::as_type` to refer to types local to the caller's scope for -example, you can annotate your attribute function with `#[use_callers_scope]`. This will ensure your attribute, and any -closures it uses, can refer to anything in the caller's scope. `#[use_callers_scope]` also works recursively. So if both -your attribute function and a helper function it calls use it, then they can both refer to the same original caller. - ---- - -## Example: Derive - -Using all of the above, we can write a `derive` macro that behaves similarly to Rust's but is not built into the language. -From the user's perspective it will look like this: - -```rust -// Example usage -#[derive(Default, Eq, Ord)] -struct MyStruct { my_field: u32 } -``` - -To implement `derive` we'll have to create a `comptime` function that accepts -a variable amount of traits. - -```rust title="derive_example" showLineNumbers -// These are needed for the unconstrained hashmap we're using to store derive functions -use crate::collections::umap::UHashMap; -use crate::hash::BuildHasherDefault; -use crate::hash::poseidon2::Poseidon2Hasher; - -// A derive function is one that given a struct definition can -// create us a quoted trait impl from it. -pub type DeriveFunction = fn(StructDefinition) -> Quoted; - -// We'll keep a global HANDLERS map to keep track of the derive handler for each trait -comptime mut global HANDLERS: UHashMap> = - UHashMap::default(); - -// Given a struct and a slice of traits to derive, create trait impls for each. -// This function is as simple as iterating over the slice, checking if we have a trait -// handler registered for the given trait, calling it, and appending the result. -#[varargs] -pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted { - let mut result = quote {}; - - for trait_to_derive in traits { - let handler = unsafe { HANDLERS.get(trait_to_derive) }; - assert(handler.is_some(), f"No derive function registered for `{trait_to_derive}`"); - - let trait_impl = handler.unwrap()(s); - result = quote { $result $trait_impl }; - } - - result -} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L31-L64 - - -Registering a derive function could be done as follows: - -```rust title="derive_via" showLineNumbers -// To register a handler for a trait, just add it to our handlers map -pub comptime fn derive_via(t: TraitDefinition, f: DeriveFunction) { - HANDLERS.insert(t, f); -} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L66-L73 - - -```rust title="big-derive-usage-example" showLineNumbers -// Finally, to register a handler we call the above function as an annotation - // with our handler function. - #[derive_via(derive_do_nothing)] - trait DoNothing { - fn do_nothing(self); - } - - comptime fn derive_do_nothing(s: StructDefinition) -> Quoted { - // This is simplified since we don't handle generics or where clauses! - // In a real example we'd likely also need to introduce each of - // `s.generics()` as well as a trait constraint for each generic - // to ensure they also implement the trait. - let typ = s.as_type(); - quote { - impl DoNothing for $typ { - fn do_nothing(self) { - // Traits can't tell us what to do - println("something"); - } - } - } - } - - // Since `DoNothing` is a simple trait which: - // 1. Only has one method - // 2. Does not have any generics on the trait itself - // We can use `std::meta::make_trait_impl` to help us out. - // This helper function will generate our impl for us along with any - // necessary where clauses and still provides a flexible interface - // for us to work on each field on the struct. - comptime fn derive_do_nothing_alt(s: StructDefinition) -> Quoted { - let trait_name = quote { DoNothing }; - let method_signature = quote { fn do_nothing(self) }; - - // Call `do_nothing` recursively on each field in the struct - let for_each_field = |field_name| quote { self.$field_name.do_nothing(); }; - - // Some traits like Eq want to join each field expression with something like `&`. - // We don't need that here - let join_fields_with = quote {}; - - // The body function is a spot to insert any extra setup/teardown needed. - // We'll insert our println here. Since we recur on each field, we should see - // one println for the struct itself, followed by a println for every field (recursively). - let body = |body| quote { - println("something"); - $body - }; - crate::meta::make_trait_impl( - s, - trait_name, - method_signature, - for_each_field, - join_fields_with, - body, - ) - } -``` -> Source code: noir_stdlib/src/meta/mod.nr#L191-L249 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/control_flow.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/control_flow.md deleted file mode 100644 index b365bb22728..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/control_flow.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Control Flow -description: - Learn how to use loops and if expressions in the Noir programming language. Discover the syntax - and examples for for loops and if-else statements. -keywords: [Noir programming language, loops, for loop, if-else statements, Rust syntax] -sidebar_position: 2 ---- - -## If Expressions - -Noir supports `if-else` statements. The syntax is most similar to Rust's where it is not required -for the statement's conditional to be surrounded by parentheses. - -```rust -let a = 0; -let mut x: u32 = 0; - -if a == 0 { - if a != 0 { - x = 6; - } else { - x = 2; - } -} else { - x = 5; - assert(x == 5); -} -assert(x == 2); -``` - -## Loops - -Noir has one kind of loop: the `for` loop. `for` loops allow you to repeat a block of code multiple -times. - -The following block of code between the braces is run 10 times. - -```rust -for i in 0..10 { - // do something -} -``` - -Alternatively, `start..=end` can be used for a range that is inclusive on both ends. - -The index for loops is of type `u64`. - -### Break and Continue - -In unconstrained code, `break` and `continue` are also allowed in `for` loops. These are only allowed -in unconstrained code since normal constrained code requires that Noir knows exactly how many iterations -a loop may have. `break` and `continue` can be used like so: - -```rust -for i in 0 .. 10 { - println("Iteration start") - - if i == 2 { - continue; - } - - if i == 5 { - break; - } - - println(i); -} -println("Loop end") -``` - -When used, `break` will end the current loop early and jump to the statement after the for loop. In the example -above, the `break` will stop the loop and jump to the `println("Loop end")`. - -`continue` will stop the current iteration of the loop, and jump to the start of the next iteration. In the example -above, `continue` will jump to `println("Iteration start")` when used. Note that the loop continues as normal after this. -The iteration variable `i` is still increased by one as normal when `continue` is used. - -`break` and `continue` cannot currently be used to jump out of more than a single loop at a time. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_bus.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_bus.mdx deleted file mode 100644 index e55e58622ce..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_bus.mdx +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: Data Bus -sidebar_position: 13 ---- -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -The data bus is an optimization that the backend can use to make recursion more efficient. -In order to use it, you must define some inputs of the program entry points (usually the `main()` -function) with the `call_data` modifier, and the return values with the `return_data` modifier. -These modifiers are incompatible with `pub` and `mut` modifiers. - -## Example - -```rust -fn main(mut x: u32, y: call_data u32, z: call_data [u32;4] ) -> return_data u32 { - let a = z[x]; - a+y -} -``` - -As a result, both call_data and return_data will be treated as private inputs and encapsulated into a read-only array each, for the backend to process. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/arrays.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/arrays.md deleted file mode 100644 index 289145a8c4d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/arrays.md +++ /dev/null @@ -1,276 +0,0 @@ ---- -title: Arrays -description: - Dive into the Array data type in Noir. Grasp its methods, practical examples, and best practices for efficiently using Arrays in your Noir code. -keywords: - [ - noir, - array type, - methods, - examples, - indexing, - ] -sidebar_position: 4 ---- - -An array is one way of grouping together values into one compound type. Array types can be inferred -or explicitly specified via the syntax `[; ]`: - -```rust -fn main(x : Field, y : Field) { - let my_arr = [x, y]; - let your_arr: [Field; 2] = [x, y]; -} -``` - -Here, both `my_arr` and `your_arr` are instantiated as an array containing two `Field` elements. - -Array elements can be accessed using indexing: - -```rust -fn main() { - let a = [1, 2, 3, 4, 5]; - - let first = a[0]; - let second = a[1]; -} -``` - -All elements in an array must be of the same type (i.e. homogeneous). That is, an array cannot group -a `Field` value and a `u8` value together for example. - -You can write mutable arrays, like: - -```rust -fn main() { - let mut arr = [1, 2, 3, 4, 5]; - assert(arr[0] == 1); - - arr[0] = 42; - assert(arr[0] == 42); -} -``` - -You can instantiate a new array of a fixed size with the same value repeated for each element. The following example instantiates an array of length 32 where each element is of type Field and has the value 0. - -```rust -let array: [Field; 32] = [0; 32]; -``` - -Like in Rust, arrays in Noir are a fixed size. However, if you wish to convert an array to a [slice](./slices.mdx), you can just call `as_slice` on your array: - -```rust -let array: [Field; 32] = [0; 32]; -let sl = array.as_slice() -``` - -You can define multidimensional arrays: - -```rust -let array : [[Field; 2]; 2]; -let element = array[0][0]; -``` - -However, multidimensional slices are not supported. For example, the following code will error at compile time: - -```rust -let slice : [[Field]] = &[]; -``` - -## Types - -You can create arrays of primitive types or structs. There is not yet support for nested arrays -(arrays of arrays) or arrays of structs that contain arrays. - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for arrays. -Each of these functions are located within the generic impl `impl [T; N] {`. -So anywhere `self` appears, it refers to the variable `self: [T; N]`. - -### len - -Returns the length of an array - -```rust -fn len(self) -> Field -``` - -example - -```rust -fn main() { - let array = [42, 42]; - assert(array.len() == 2); -} -``` - -### sort - -Returns a new sorted array. The original array remains untouched. Notice that this function will -only work for arrays of fields or integers, not for any arbitrary type. This is because the sorting -logic it uses internally is optimized specifically for these values. If you need a sort function to -sort any type, you should use the function `sort_via` described below. - -```rust -fn sort(self) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32]; - let sorted = arr.sort(); - assert(sorted == [32, 42]); -} -``` - -### sort_via - -Sorts the array with a custom comparison function. The ordering function must return true if the first argument should be sorted to be before the second argument or is equal to the second argument. - -Using this method with an operator like `<` that does not return `true` for equal values will result in an assertion failure for arrays with equal elements. - -```rust -fn sort_via(self, ordering: fn(T, T) -> bool) -> [T; N] -``` - -example - -```rust -fn main() { - let arr = [42, 32] - let sorted_ascending = arr.sort_via(|a, b| a <= b); - assert(sorted_ascending == [32, 42]); // verifies - - let sorted_descending = arr.sort_via(|a, b| a >= b); - assert(sorted_descending == [32, 42]); // does not verify -} -``` - -### map - -Applies a function to each element of the array, returning a new array containing the mapped elements. - -```rust -fn map(self, f: fn(T) -> U) -> [U; N] -``` - -example - -```rust -let a = [1, 2, 3]; -let b = a.map(|a| a * 2); // b is now [2, 4, 6] -``` - -### fold - -Applies a function to each element of the array, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn(U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the array, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = [1]; -let a2 = [1, 2]; -let a3 = [1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let arr = [2, 2, 2, 2, 2]; - let folded = arr.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -Requires `self` to be non-empty. - -```rust -fn reduce(self, f: fn(T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let reduced = arr.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 2]; - let all = arr.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn(T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let arr = [2, 2, 2, 2, 5]; - let any = arr.any(|a| a == 5); - assert(any); -} -``` - -### as_str_unchecked - -Converts a byte array of type `[u8; N]` to a string. Note that this performs no UTF-8 validation - -the given array is interpreted as-is as a string. - -```rust -impl [u8; N] { - pub fn as_str_unchecked(self) -> str -} -``` - -example: - -```rust -fn main() { - let hi = [104, 105].as_str_unchecked(); - assert_eq(hi, "hi"); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/booleans.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/booleans.md deleted file mode 100644 index 2507af710e7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/booleans.md +++ /dev/null @@ -1,28 +0,0 @@ ---- -title: Booleans -description: - Delve into the Boolean data type in Noir. Understand its methods, practical examples, and best practices for using Booleans in your Noir programs. -keywords: - [ - noir, - boolean type, - methods, - examples, - logical operations, - ] -sidebar_position: 2 ---- - - -The `bool` type in Noir has two possible values: `true` and `false`: - -```rust -fn main() { - let t = true; - let f: bool = false; -} -``` - -The boolean type is most commonly used in conditionals like `if` expressions and `assert` -statements. More about conditionals is covered in the [Control Flow](../control_flow.md) and -[Assert Function](../assert.md) sections. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/fields.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/fields.md deleted file mode 100644 index b9b56f7ecc3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/fields.md +++ /dev/null @@ -1,246 +0,0 @@ ---- -title: Fields -description: - Dive deep into the Field data type in Noir. Understand its methods, practical examples, and best practices to effectively use Fields in your Noir programs. -keywords: - [ - noir, - field type, - methods, - examples, - best practices, - ] -sidebar_position: 0 ---- - -The field type corresponds to the native field type of the proving backend. - -The size of a Noir field depends on the elliptic curve's finite field for the proving backend -adopted. For example, a field would be a 254-bit integer when paired with the default backend that -spans the Grumpkin curve. - -Fields support integer arithmetic and are often used as the default numeric type in Noir: - -```rust -fn main(x : Field, y : Field) { - let z = x + y; -} -``` - -`x`, `y` and `z` are all private fields in this example. Using the `let` keyword we defined a new -private value `z` constrained to be equal to `x + y`. - -If proving efficiency is of priority, fields should be used as a default for solving problems. -Smaller integer types (e.g. `u64`) incur extra range constraints. - -## Methods - -After declaring a Field, you can use these common methods on it: - -### to_le_bits - -Transforms the field into an array of bits, Little Endian. - -```rust title="to_le_bits" showLineNumbers -pub fn to_le_bits(self: Self) -> [u1; N] {} -``` -> Source code: noir_stdlib/src/field/mod.nr#L32-L34 - - -example: - -```rust title="to_le_bits_example" showLineNumbers -fn test_to_le_bits() { - let field = 2; - let bits: [u1; 8] = field.to_le_bits(); - assert_eq(bits, [0, 1, 0, 0, 0, 0, 0, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L276-L282 - - - -### to_be_bits - -Transforms the field into an array of bits, Big Endian. - -```rust title="to_be_bits" showLineNumbers -pub fn to_be_bits(self: Self) -> [u1; N] {} -``` -> Source code: noir_stdlib/src/field/mod.nr#L48-L50 - - -example: - -```rust title="to_be_bits_example" showLineNumbers -fn test_to_be_bits() { - let field = 2; - let bits: [u1; 8] = field.to_be_bits(); - assert_eq(bits, [0, 0, 0, 0, 0, 0, 1, 0]); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L267-L273 - - - -### to_le_bytes - -Transforms into an array of bytes, Little Endian - -```rust title="to_le_bytes" showLineNumbers -pub fn to_le_bytes(self: Self) -> [u8; N] { -``` -> Source code: noir_stdlib/src/field/mod.nr#L61-L63 - - -example: - -```rust title="to_le_bytes_example" showLineNumbers -fn test_to_le_bytes() { - let field = 2; - let bytes: [u8; 8] = field.to_le_bytes(); - assert_eq(bytes, [2, 0, 0, 0, 0, 0, 0, 0]); - assert_eq(Field::from_le_bytes::<8>(bytes), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L295-L302 - - -### to_be_bytes - -Transforms into an array of bytes, Big Endian - -```rust title="to_be_bytes" showLineNumbers -pub fn to_be_bytes(self: Self) -> [u8; N] { -``` -> Source code: noir_stdlib/src/field/mod.nr#L94-L96 - - -example: - -```rust title="to_be_bytes_example" showLineNumbers -fn test_to_be_bytes() { - let field = 2; - let bytes: [u8; 8] = field.to_be_bytes(); - assert_eq(bytes, [0, 0, 0, 0, 0, 0, 0, 2]); - assert_eq(Field::from_be_bytes::<8>(bytes), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L285-L292 - - - -### to_le_radix - -Decomposes into an array over the specified base, Little Endian - -```rust title="to_le_radix" showLineNumbers -pub fn to_le_radix(self: Self, radix: u32) -> [u8; N] { - // Brillig does not need an immediate radix - if !crate::runtime::is_unconstrained() { - crate::assert_constant(radix); - } - self.__to_le_radix(radix) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L118-L126 - - - -example: - -```rust title="to_le_radix_example" showLineNumbers -fn test_to_le_radix() { - let field = 2; - let bytes: [u8; 8] = field.to_le_radix(256); - assert_eq(bytes, [2, 0, 0, 0, 0, 0, 0, 0]); - assert_eq(Field::from_le_bytes::<8>(bytes), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L315-L322 - - - -### to_be_radix - -Decomposes into an array over the specified base, Big Endian - -```rust title="to_be_radix" showLineNumbers -pub fn to_be_radix(self: Self, radix: u32) -> [u8; N] { - // Brillig does not need an immediate radix - if !crate::runtime::is_unconstrained() { - crate::assert_constant(radix); - } - self.__to_be_radix(radix) - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L128-L136 - - -example: - -```rust title="to_be_radix_example" showLineNumbers -fn test_to_be_radix() { - let field = 2; - let bytes: [u8; 8] = field.to_be_radix(256); - assert_eq(bytes, [0, 0, 0, 0, 0, 0, 0, 2]); - assert_eq(Field::from_be_bytes::<8>(bytes), field); - } -``` -> Source code: noir_stdlib/src/field/mod.nr#L305-L312 - - - -### pow_32 - -Returns the value to the power of the specified exponent - -```rust -fn pow_32(self, exponent: Field) -> Field -``` - -example: - -```rust -fn main() { - let field = 2 - let pow = field.pow_32(4); - assert(pow == 16); -} -``` - -### assert_max_bit_size - -Adds a constraint to specify that the field can be represented with `bit_size` number of bits - -```rust title="assert_max_bit_size" showLineNumbers -pub fn assert_max_bit_size(self) { -``` -> Source code: noir_stdlib/src/field/mod.nr#L10-L12 - - -example: - -```rust -fn main() { - let field = 2 - field.assert_max_bit_size(32); -} -``` - -### sgn0 - -Parity of (prime) Field element, i.e. sgn0(x mod p) = 0 if x ∈ \{0, ..., p-1\} is even, otherwise sgn0(x mod p) = 1. - -```rust -fn sgn0(self) -> u1 -``` - - -### lt - -Returns true if the field is less than the other field - -```rust -pub fn lt(self, another: Field) -> bool -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/function_types.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/function_types.md deleted file mode 100644 index f6121af17e2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/function_types.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Function types -sidebar_position: 10 ---- - -Noir supports higher-order functions. The syntax for a function type is as follows: - -```rust -fn(arg1_type, arg2_type, ...) -> return_type -``` - -Example: - -```rust -fn assert_returns_100(f: fn() -> Field) { // f takes no args and returns a Field - assert(f() == 100); -} - -fn main() { - assert_returns_100(|| 100); // ok - assert_returns_100(|| 150); // fails -} -``` - -A function type also has an optional capture environment - this is necessary to support closures. -See [Lambdas](../lambdas.md) for more details. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/index.md deleted file mode 100644 index 0f2db2b2d75..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/index.md +++ /dev/null @@ -1,126 +0,0 @@ ---- -title: Data Types -description: - Get a clear understanding of the two categories of Noir data types - primitive types and compound - types. Learn about their characteristics, differences, and how to use them in your Noir - programming. -keywords: - [ - noir, - data types, - primitive types, - compound types, - private types, - public types, - ] ---- - -Every value in Noir has a type, which determines which operations are valid for it. - -All values in Noir are fundamentally composed of `Field` elements. For a more approachable -developing experience, abstractions are added on top to introduce different data types in Noir. - -Noir has two category of data types: primitive types (e.g. `Field`, integers, `bool`) and compound -types that group primitive types (e.g. arrays, tuples, structs). Each value can either be private or -public. - -## Private & Public Types - -A **private value** is known only to the Prover, while a **public value** is known by both the -Prover and Verifier. Mark values as `private` when the value should only be known to the prover. All -primitive types (including individual fields of compound types) in Noir are private by default, and -can be marked public when certain values are intended to be revealed to the Verifier. - -> **Note:** For public values defined in Noir programs paired with smart contract verifiers, once -> the proofs are verified on-chain the values can be considered known to everyone that has access to -> that blockchain. - -Public data types are treated no differently to private types apart from the fact that their values -will be revealed in proofs generated. Simply changing the value of a public type will not change the -circuit (where the same goes for changing values of private types as well). - -_Private values_ are also referred to as _witnesses_ sometimes. - -> **Note:** The terms private and public when applied to a type (e.g. `pub Field`) have a different -> meaning than when applied to a function (e.g. `pub fn foo() {}`). -> -> The former is a visibility modifier for the Prover to interpret if a value should be made known to -> the Verifier, while the latter is a visibility modifier for the compiler to interpret if a -> function should be made accessible to external Noir programs like in other languages. - -### pub Modifier - -All data types in Noir are private by default. Types are explicitly declared as public using the -`pub` modifier: - -```rust -fn main(x : Field, y : pub Field) -> pub Field { - x + y -} -``` - -In this example, `x` is **private** while `y` and `x + y` (the return value) are **public**. Note -that visibility is handled **per variable**, so it is perfectly valid to have one input that is -private and another that is public. - -> **Note:** Public types can only be declared through parameters on `main`. - -## Type Aliases - -A type alias is a new name for an existing type. Type aliases are declared with the keyword `type`: - -```rust -type Id = u8; - -fn main() { - let id: Id = 1; - let zero: u8 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can also be used with [generics](../generics.md): - -```rust -type Id = Size; - -fn main() { - let id: Id = 1; - let zero: u32 = 0; - assert(zero + 1 == id); -} -``` - -Type aliases can even refer to other aliases. An error will be issued if they form a cycle: - -```rust -// Ok! -type A = B; -type B = Field; - -type Bad1 = Bad2; - -// error: Dependency cycle found -type Bad2 = Bad1; -// ^^^^^^^^^^^ 'Bad2' recursively depends on itself: Bad2 -> Bad1 -> Bad2 -``` - -By default, like functions, type aliases are private to the module they exist in. You can use `pub` -to make the type alias public or `pub(crate)` to make it public to just its crate: - -```rust -// This type alias is now public -pub type Id = u8; -``` - -## Wildcard Type -Noir can usually infer the type of the variable from the context, so specifying the type of a variable is only required when it cannot be inferred. However, specifying a complex type can be tedious, especially when it has multiple generic arguments. Often some of the generic types can be inferred from the context, and Noir only needs a hint to properly infer the other types. We can partially specify a variable's type by using `_` as a marker, indicating where we still want the compiler to infer the type. - -```rust -let a: [_; 4] = foo(b); -``` - - -### BigInt - -You can achieve BigInt functionality using the [Noir BigInt](https://github.com/shuklaayush/noir-bigint) library. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/integers.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/integers.md deleted file mode 100644 index a1d59bf3166..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/integers.md +++ /dev/null @@ -1,156 +0,0 @@ ---- -title: Integers -description: Explore the Integer data type in Noir. Learn about its methods, see real-world examples, and grasp how to efficiently use Integers in your Noir code. -keywords: [noir, integer types, methods, examples, arithmetic] -sidebar_position: 1 ---- - -An integer type is a range constrained field type. -The Noir frontend supports both unsigned and signed integer types. -The allowed sizes are 1, 8, 16, 32 and 64 bits. - -:::info - -When an integer is defined in Noir without a specific type, it will default to `Field`. - -The one exception is for loop indices which default to `u64` since comparisons on `Field`s are not possible. - -::: - -## Unsigned Integers - -An unsigned integer type is specified first with the letter `u` (indicating its unsigned nature) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: u8 = 1; - let y: u8 = 1; - let z = x + y; - assert (z == 2); -} -``` - -The bit size determines the maximum value the integer type can store. For example, a `u8` variable can store a value in the range of 0 to 255 (i.e. $\\2^{8}-1\\$). - -## Signed Integers - -A signed integer type is specified first with the letter `i` (which stands for integer) followed by its bit size (e.g. `8`): - -```rust -fn main() { - let x: i8 = -1; - let y: i8 = -1; - let z = x + y; - assert (z == -2); -} -``` - -The bit size determines the maximum and minimum range of value the integer type can store. For example, an `i8` variable can store a value in the range of -128 to 127 (i.e. $\\-2^{7}\\$ to $\\2^{7}-1\\$). - -## 128 bits Unsigned Integers - -The built-in structure `U128` allows you to use 128-bit unsigned integers almost like a native integer type. However, there are some differences to keep in mind: -- You cannot cast between a native integer and `U128` -- There is a higher performance cost when using `U128`, compared to a native type. - -Conversion between unsigned integer types and U128 are done through the use of `from_integer` and `to_integer` functions. `from_integer` also accepts the `Field` type as input. - -```rust -fn main() { - let x = U128::from_integer(23); - let y = U128::from_hex("0x7"); - let z = x + y; - assert(z.to_integer() == 30); -} -``` - -`U128` is implemented with two 64 bits limbs, representing the low and high bits, which explains the performance cost. You should expect `U128` to be twice more costly for addition and four times more costly for multiplication. -You can construct a U128 from its limbs: -```rust -fn main(x: u64, y: u64) { - let x = U128::from_u64s_be(x,y); - assert(z.hi == x as Field); - assert(z.lo == y as Field); -} -``` - -Note that the limbs are stored as Field elements in order to avoid unnecessary conversions. -Apart from this, most operations will work as usual: - -```rust -fn main(x: U128, y: U128) { - // multiplication - let c = x * y; - // addition and subtraction - let c = c - x + y; - // division - let c = x / y; - // bit operation; - let c = x & y | y; - // bit shift - let c = x << y; - // comparisons; - let c = x < y; - let c = x == y; -} -``` - -## Overflows - -Computations that exceed the type boundaries will result in overflow errors. This happens with both signed and unsigned integers. For example, attempting to prove: - -```rust -fn main(x: u8, y: u8) { - let z = x + y; -} -``` - -With: - -```toml -x = "255" -y = "1" -``` - -Would result in: - -``` -$ nargo execute -error: Assertion failed: 'attempt to add with overflow' -┌─ ~/src/main.nr:9:13 -│ -│ let z = x + y; -│ ----- -│ -= Call stack: - ... -``` - -A similar error would happen with signed integers: - -```rust -fn main() { - let x: i8 = -118; - let y: i8 = -11; - let z = x + y; -} -``` - -### Wrapping methods - -Although integer overflow is expected to error, some use-cases rely on wrapping. For these use-cases, the standard library provides `wrapping` variants of certain common operations: - -```rust -fn wrapping_add(x: T, y: T) -> T; -fn wrapping_sub(x: T, y: T) -> T; -fn wrapping_mul(x: T, y: T) -> T; -``` - -Example of how it is used: - -```rust - -fn main(x: u8, y: u8) -> pub u8 { - std::wrapping_add(x, y) -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/references.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/references.md deleted file mode 100644 index a5293d11cfb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/references.md +++ /dev/null @@ -1,23 +0,0 @@ ---- -title: References -sidebar_position: 9 ---- - -Noir supports first-class references. References are a bit like pointers: they point to a specific address that can be followed to access the data stored at that address. You can use Rust-like syntax to use pointers in Noir: the `&` operator references the variable, the `*` operator dereferences it. - -Example: - -```rust -fn main() { - let mut x = 2; - - // you can reference x as &mut and pass it to multiplyBy2 - multiplyBy2(&mut x); -} - -// you can access &mut here -fn multiplyBy2(x: &mut Field) { - // and dereference it with * - *x = *x * 2; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/slices.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/slices.mdx deleted file mode 100644 index cfee564a302..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/slices.mdx +++ /dev/null @@ -1,358 +0,0 @@ ---- -title: Slices -description: Explore the Slice data type in Noir. Understand its methods, see real-world examples, and learn how to effectively use Slices in your Noir programs. -keywords: [noir, slice type, methods, examples, subarrays] -sidebar_position: 5 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A slice is a dynamically-sized view into a sequence of elements. They can be resized at runtime, but because they don't own the data, they cannot be returned from a circuit. You can treat slices as arrays without a constrained size. - -```rust -fn main() -> pub u32 { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -To write a slice literal, use a preceding ampersand as in: `&[0; 2]` or -`&[1, 2, 3]`. - -It is important to note that slices are not references to arrays. In Noir, -`&[..]` is more similar to an immutable, growable vector. - -View the corresponding test file [here][test-file]. - -[test-file]: https://github.com/noir-lang/noir/blob/f387ec1475129732f72ba294877efdf6857135ac/crates/nargo_cli/tests/test_data_ssa_refactor/slices/src/main.nr - -## Methods - -For convenience, the STD provides some ready-to-use, common methods for slices: - -### push_back - -Pushes a new element to the end of the slice, returning a new slice with a length one greater than the original unmodified slice. - -```rust -fn push_back(_self: [T], _elem: T) -> [T] -``` - -example: - -```rust -fn main() -> pub Field { - let mut slice: [Field] = &[0; 2]; - - let mut new_slice = slice.push_back(6); - new_slice.len() -} -``` - -View the corresponding test file [here][test-file]. - -### push_front - -Returns a new array with the specified element inserted at index 0. The existing elements indexes are incremented by 1. - -```rust -fn push_front(_self: Self, _elem: T) -> Self -``` - -Example: - -```rust -let mut new_slice: [Field] = &[]; -new_slice = new_slice.push_front(20); -assert(new_slice[0] == 20); // returns true -``` - -View the corresponding test file [here][test-file]. - -### pop_front - -Returns a tuple of two items, the first element of the array and the rest of the array. - -```rust -fn pop_front(_self: Self) -> (T, Self) -``` - -Example: - -```rust -let (first_elem, rest_of_slice) = slice.pop_front(); -``` - -View the corresponding test file [here][test-file]. - -### pop_back - -Returns a tuple of two items, the beginning of the array with the last element omitted and the last element. - -```rust -fn pop_back(_self: Self) -> (Self, T) -``` - -Example: - -```rust -let (popped_slice, last_elem) = slice.pop_back(); -``` - -View the corresponding test file [here][test-file]. - -### append - -Loops over a slice and adds it to the end of another. - -```rust -fn append(mut self, other: Self) -> Self -``` - -Example: - -```rust -let append = &[1, 2].append(&[3, 4, 5]); -``` - -### insert - -Inserts an element at a specified index and shifts all following elements by 1. - -```rust -fn insert(_self: Self, _index: Field, _elem: T) -> Self -``` - -Example: - -```rust -new_slice = rest_of_slice.insert(2, 100); -assert(new_slice[2] == 100); -``` - -View the corresponding test file [here][test-file]. - -### remove - -Remove an element at a specified index, shifting all elements after it to the left, returning the altered slice and the removed element. - -```rust -fn remove(_self: Self, _index: Field) -> (Self, T) -``` - -Example: - -```rust -let (remove_slice, removed_elem) = slice.remove(3); -``` - -### len - -Returns the length of a slice - -```rust -fn len(self) -> Field -``` - -Example: - -```rust -fn main() { - let slice = &[42, 42]; - assert(slice.len() == 2); -} -``` - -### as_array - -Converts this slice into an array. - -Make sure to specify the size of the resulting array. -Panics if the resulting array length is different than the slice's length. - -```rust -fn as_array(self) -> [T; N] -``` - -Example: - -```rust -fn main() { - let slice = &[5, 6]; - - // Always specify the length of the resulting array! - let array: [Field; 2] = slice.as_array(); - - assert(array[0] == slice[0]); - assert(array[1] == slice[1]); -} -``` - -### map - -Applies a function to each element of the slice, returning a new slice containing the mapped elements. - -```rust -fn map(self, f: fn[Env](T) -> U) -> [U] -``` - -example - -```rust -let a = &[1, 2, 3]; -let b = a.map(|a| a * 2); // b is now &[2, 4, 6] -``` - -### fold - -Applies a function to each element of the slice, returning the final accumulated value. The first -parameter is the initial value. - -```rust -fn fold(self, mut accumulator: U, f: fn[Env](U, T) -> U) -> U -``` - -This is a left fold, so the given function will be applied to the accumulator and first element of -the slice, then the second, and so on. For a given call the expected result would be equivalent to: - -```rust -let a1 = &[1]; -let a2 = &[1, 2]; -let a3 = &[1, 2, 3]; - -let f = |a, b| a - b; -a1.fold(10, f) //=> f(10, 1) -a2.fold(10, f) //=> f(f(10, 1), 2) -a3.fold(10, f) //=> f(f(f(10, 1), 2), 3) -``` - -example: - -```rust - -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let folded = slice.fold(0, |a, b| a + b); - assert(folded == 10); -} - -``` - -### reduce - -Same as fold, but uses the first element as the starting element. - -```rust -fn reduce(self, f: fn[Env](T, T) -> T) -> T -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let reduced = slice.reduce(|a, b| a + b); - assert(reduced == 10); -} -``` - -### filter - -Returns a new slice containing only elements for which the given predicate returns true. - -```rust -fn filter(self, f: fn[Env](T) -> bool) -> Self -``` - -example: - -```rust -fn main() { - let slice = &[1, 2, 3, 4, 5]; - let odds = slice.filter(|x| x % 2 == 1); - assert_eq(odds, &[1, 3, 5]); -} -``` - -### join - -Flatten each element in the slice into one value, separated by `separator`. - -Note that although slices implement `Append`, `join` cannot be used on slice -elements since nested slices are prohibited. - -```rust -fn join(self, separator: T) -> T where T: Append -``` - -example: - -```rust -struct Accumulator { - total: Field, -} - -// "Append" two accumulators by adding them -impl Append for Accumulator { - fn empty() -> Self { - Self { total: 0 } - } - - fn append(self, other: Self) -> Self { - Self { total: self.total + other.total } - } -} - -fn main() { - let slice = &[1, 2, 3, 4, 5].map(|total| Accumulator { total }); - - let result = slice.join(Accumulator::empty()); - assert_eq(result, Accumulator { total: 15 }); - - // We can use a non-empty separator to insert additional elements to sum: - let separator = Accumulator { total: 10 }; - let result = slice.join(separator); - assert_eq(result, Accumulator { total: 55 }); -} -``` - -### all - -Returns true if all the elements satisfy the given predicate - -```rust -fn all(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 2]; - let all = slice.all(|a| a == 2); - assert(all); -} -``` - -### any - -Returns true if any of the elements satisfy the given predicate - -```rust -fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -example: - -```rust -fn main() { - let slice = &[2, 2, 2, 2, 5]; - let any = slice.any(|a| a == 5); - assert(any); -} - -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/strings.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/strings.md deleted file mode 100644 index 1fdee42425e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/strings.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Strings -description: - Discover the String data type in Noir. Learn about its methods, see real-world examples, and understand how to effectively manipulate and use Strings in Noir. -keywords: - [ - noir, - string type, - methods, - examples, - concatenation, - ] -sidebar_position: 3 ---- - - -The string type is a fixed length value defined with `str`. - -You can use strings in `assert()` functions or print them with -`println()`. See more about [Logging](../../standard_library/logging.md). - -```rust - -fn main(message : pub str<11>, hex_as_string : str<4>) { - println(message); - assert(message == "hello world"); - assert(hex_as_string == "0x41"); -} -``` - -You can convert a `str` to a byte array by calling `as_bytes()` -or a vector by calling `as_bytes_vec()`. - -```rust -fn main() { - let message = "hello world"; - let message_bytes = message.as_bytes(); - let mut message_vec = message.as_bytes_vec(); - assert(message_bytes.len() == 11); - assert(message_bytes[0] == 104); - assert(message_bytes[0] == message_vec.get(0)); -} -``` - -## Escape characters - -You can use escape characters for your strings: - -| Escape Sequence | Description | -|-----------------|-----------------| -| `\r` | Carriage Return | -| `\n` | Newline | -| `\t` | Tab | -| `\0` | Null Character | -| `\"` | Double Quote | -| `\\` | Backslash | - -Example: - -```rust -let s = "Hello \"world" // prints "Hello "world" -let s = "hey \tyou"; // prints "hey you" -``` - -## Raw strings - -A raw string begins with the letter `r` and is optionally delimited by a number of hashes `#`. - -Escape characters are *not* processed within raw strings. All contents are interpreted literally. - -Example: - -```rust -let s = r"Hello world"; -let s = r#"Simon says "hello world""#; - -// Any number of hashes may be used (>= 1) as long as the string also terminates with the same number of hashes -let s = r#####"One "#, Two "##, Three "###, Four "####, Five will end the string."#####; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/structs.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/structs.md deleted file mode 100644 index 29951ae843a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/structs.md +++ /dev/null @@ -1,96 +0,0 @@ ---- -title: Structs -description: - Explore the Struct data type in Noir. Learn about its methods, see real-world examples, and grasp how to effectively define and use Structs in your Noir programs. -keywords: - [ - noir, - struct type, - methods, - examples, - data structures, - ] -sidebar_position: 8 ---- - -A struct also allows for grouping multiple values of different types. Unlike tuples, we can also -name each field. - -> **Note:** The usage of _field_ here refers to each element of the struct and is unrelated to the -> field type of Noir. - -Defining a struct requires giving it a name and listing each field within as `: ` pairs: - -```rust -struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` - -An instance of a struct can then be created with actual values in `: ` pairs in any -order. Struct fields are accessible using their given names: - -```rust -fn main() { - let legs = 4; - - let dog = Animal { - eyes: 2, - hands: 0, - legs, - }; - - let zero = dog.hands; -} -``` - -Structs can also be destructured in a pattern, binding each field to a new variable: - -```rust -fn main() { - let Animal { hands, legs: feet, eyes } = get_octopus(); - - let ten = hands + feet + eyes as u8; -} - -fn get_octopus() -> Animal { - let octopus = Animal { - hands: 0, - legs: 8, - eyes: 2, - }; - - octopus -} -``` - -The new variables can be bound with names different from the original struct field names, as -showcased in the `legs --> feet` binding in the example above. - -### Visibility - -By default, like functions, structs are private to the module they exist in. You can use `pub` -to make the struct public or `pub(crate)` to make it public to just its crate: - -```rust -// This struct is now public -pub struct Animal { - hands: Field, - legs: Field, - eyes: u8, -} -``` - -The same applies to struct fields: by default they are private to the module they exist in, -but they can be made `pub` or `pub(crate)`: - -```rust -// This struct is now public -pub struct Animal { - hands: Field, // private to its module - pub(crate) legs: Field, // accessible from the entire crate - pub eyes: u8, // accessible from anywhere -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/tuples.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/tuples.md deleted file mode 100644 index 2ec5c9c4113..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/data_types/tuples.md +++ /dev/null @@ -1,48 +0,0 @@ ---- -title: Tuples -description: - Dive into the Tuple data type in Noir. Understand its methods, practical examples, and best practices for efficiently using Tuples in your Noir code. -keywords: - [ - noir, - tuple type, - methods, - examples, - multi-value containers, - ] -sidebar_position: 7 ---- - -A tuple collects multiple values like an array, but with the added ability to collect values of -different types: - -```rust -fn main() { - let tup: (u8, u64, Field) = (255, 500, 1000); -} -``` - -One way to access tuple elements is via destructuring using pattern matching: - -```rust -fn main() { - let tup = (1, 2); - - let (one, two) = tup; - - let three = one + two; -} -``` - -Another way to access tuple elements is via direct member access, using a period (`.`) followed by -the index of the element we want to access. Index `0` corresponds to the first tuple element, `1` to -the second and so on: - -```rust -fn main() { - let tup = (5, 6, 7, 8); - - let five = tup.0; - let eight = tup.3; -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/functions.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/functions.md deleted file mode 100644 index f656cdfd97a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/functions.md +++ /dev/null @@ -1,226 +0,0 @@ ---- -title: Functions -description: - Learn how to declare functions and methods in Noir, a programming language with Rust semantics. - This guide covers parameter declaration, return types, call expressions, and more. -keywords: [Noir, Rust, functions, methods, parameter declaration, return types, call expressions] -sidebar_position: 1 ---- - -Functions in Noir follow the same semantics of Rust, though Noir does not support early returns. - -To declare a function the `fn` keyword is used. - -```rust -fn foo() {} -``` - -By default, functions are visible only within the package they are defined. To make them visible outside of that package (for example, as part of a [library](../modules_packages_crates/crates_and_packages.md#libraries)), you should mark them as `pub`: - -```rust -pub fn foo() {} -``` - -You can also restrict the visibility of the function to only the crate it was defined in, by specifying `pub(crate)`: - -```rust -pub(crate) fn foo() {} //foo can only be called within its crate -``` - -All parameters in a function must have a type and all types are known at compile time. The parameter -is pre-pended with a colon and the parameter type. Multiple parameters are separated using a comma. - -```rust -fn foo(x : Field, y : Field){} -``` - -The return type of a function can be stated by using the `->` arrow notation. The function below -states that the foo function must return a `Field`. If the function returns no value, then the arrow -is omitted. - -```rust -fn foo(x : Field, y : Field) -> Field { - x + y -} -``` - -Note that a `return` keyword is unneeded in this case - the last expression in a function's body is -returned. - -## Main function - -If you're writing a binary, the `main` function is the starting point of your program. You can pass all types of expressions to it, as long as they have a fixed size at compile time: - -```rust -fn main(x : Field) // this is fine: passing a Field -fn main(x : [Field; 2]) // this is also fine: passing a Field with known size at compile-time -fn main(x : (Field, bool)) // 👌: passing a (Field, bool) tuple means size 2 -fn main(x : str<5>) // this is fine, as long as you pass a string of size 5 - -fn main(x : Vec) // can't compile, has variable size -fn main(x : [Field]) // can't compile, has variable size -fn main(....// i think you got it by now -``` - -Keep in mind [tests](../../tooling/testing.md) don't differentiate between `main` and any other function. The following snippet passes tests, but won't compile or prove: - -```rust -fn main(x : [Field]) { - assert(x[0] == 1); -} - -#[test] -fn test_one() { - main(&[1, 2]); -} -``` - -```bash -$ nargo test -[testing] Running 1 test functions -[testing] Testing test_one... ok -[testing] All tests passed - -$ nargo check -The application panicked (crashed). -Message: Cannot have variable sized arrays as a parameter to main -``` - -## Call Expressions - -Calling a function in Noir is executed by using the function name and passing in the necessary -arguments. - -Below we show how to call the `foo` function from the `main` function using a call expression: - -```rust -fn main(x : Field, y : Field) { - let z = foo(x); -} - -fn foo(x : Field) -> Field { - x + x -} -``` - -## Methods - -You can define methods in Noir on any struct type in scope. - -```rust -struct MyStruct { - foo: Field, - bar: Field, -} - -impl MyStruct { - fn new(foo: Field) -> MyStruct { - MyStruct { - foo, - bar: 2, - } - } - - fn sum(self) -> Field { - self.foo + self.bar - } -} - -fn main() { - let s = MyStruct::new(40); - assert(s.sum() == 42); -} -``` - -Methods are just syntactic sugar for functions, so if we wanted to we could also call `sum` as -follows: - -```rust -assert(MyStruct::sum(s) == 42); -``` - -It is also possible to specialize which method is chosen depending on the [generic](./generics.md) type that is used. In this example, the `foo` function returns different values depending on its type: - -```rust -struct Foo {} - -impl Foo { - fn foo(self) -> Field { 1 } -} - -impl Foo { - fn foo(self) -> Field { 2 } -} - -fn main() { - let f1: Foo = Foo{}; - let f2: Foo = Foo{}; - assert(f1.foo() + f2.foo() == 3); -} -``` - -Also note that impls with the same method name defined in them cannot overlap. For example, if we already have `foo` defined for `Foo` and `Foo` like we do above, we cannot also define `foo` in an `impl Foo` since it would be ambiguous which version of `foo` to choose. - -```rust -// Including this impl in the same project as the above snippet would -// cause an overlapping impls error -impl Foo { - fn foo(self) -> Field { 3 } -} -``` - -## Lambdas - -Lambdas are anonymous functions. They follow the syntax of Rust - `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -See [Lambdas](./lambdas.md) for more details. - -## Attributes - -Attributes are metadata that can be applied to a function, using the following syntax: `#[attribute(value)]`. - -Supported attributes include: - -- **builtin**: the function is implemented by the compiler, for efficiency purposes. -- **deprecated**: mark the function as _deprecated_. Calling the function will generate a warning: `warning: use of deprecated function` -- **field**: Used to enable conditional compilation of code depending on the field size. See below for more details -- **oracle**: mark the function as _oracle_; meaning it is an external unconstrained function, implemented in noir_js. See [Unconstrained](./unconstrained.md) and [NoirJS](../../reference/NoirJS/noir_js/index.md) for more details. -- **test**: mark the function as unit tests. See [Tests](../../tooling/testing.md) for more details - -### Field Attribute - -The field attribute defines which field the function is compatible for. The function is conditionally compiled, under the condition that the field attribute matches the Noir native field. -The field can be defined implicitly, by using the name of the elliptic curve usually associated to it - for instance bn254, bls12_381 - or explicitly by using the field (prime) order, in decimal or hexadecimal form. -As a result, it is possible to define multiple versions of a function with each version specialized for a different field attribute. This can be useful when a function requires different parameters depending on the underlying elliptic curve. - -Example: we define the function `foo()` three times below. Once for the default Noir bn254 curve, once for the field $\mathbb F_{23}$, which will normally never be used by Noir, and once again for the bls12_381 curve. - -```rust -#[field(bn254)] -fn foo() -> u32 { - 1 -} - -#[field(23)] -fn foo() -> u32 { - 2 -} - -// This commented code would not compile as foo would be defined twice because it is the same field as bn254 -// #[field(21888242871839275222246405745257275088548364400416034343698204186575808495617)] -// fn foo() -> u32 { -// 2 -// } - -#[field(bls12_381)] -fn foo() -> u32 { - 3 -} -``` - -If the field name is not known to Noir, it will discard the function. Field names are case insensitive. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/generics.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/generics.md deleted file mode 100644 index c180a0ce7e6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/generics.md +++ /dev/null @@ -1,251 +0,0 @@ ---- -title: Generics -description: Learn how to use Generics in Noir -keywords: [Noir, Rust, generics, functions, structs] -sidebar_position: 7 ---- - -Generics allow you to use the same functions with multiple different concrete data types. You can -read more about the concept of generics in the Rust documentation -[here](https://doc.rust-lang.org/book/ch10-01-syntax.html). - -Here is a trivial example showing the identity function that supports any type. In Rust, it is -common to refer to the most general type as `T`. We follow the same convention in Noir. - -```rust -fn id(x: T) -> T { - x -} -``` - -## Numeric Generics - -If we want to be generic over array lengths (which are type-level integers), we can use numeric -generics. Using these looks similar to using regular generics, but introducing them into scope -requires declaring them with `let MyGenericName: IntegerType`. This can be done anywhere a normal -generic is declared. Instead of types, these generics resolve to integers at compile-time. -Here's an example of a struct that is generic over the size of the array it contains internally: - -```rust -struct BigInt { - limbs: [u32; N], -} - -impl BigInt { - // `N` is in scope of all methods in the impl - fn first(first: BigInt, second: BigInt) -> Self { - assert(first.limbs != second.limbs); - first - - fn second(first: BigInt, second: Self) -> Self { - assert(first.limbs != second.limbs); - second - } -} -``` - -## In Structs - -Generics are useful for specifying types in structs. For example, we can specify that a field in a -struct will be of a certain generic type. In this case `value` is of type `T`. - -```rust -struct RepeatedValue { - value: T, - count: Field, -} - -impl RepeatedValue { - fn print(self) { - for _i in 0 .. self.count { - println(self.value); - } - } -} - -fn main() { - let repeated = RepeatedValue { value: "Hello!", count: 2 }; - repeated.print(); -} -``` - -The `print` function will print `Hello!` an arbitrary number of times, twice in this case. - -## Calling functions on generic parameters - -Since a generic type `T` can represent any type, how can we call functions on the underlying type? -In other words, how can we go from "any type `T`" to "any type `T` that has certain methods available?" - -This is what [traits](../concepts/traits.md) are for in Noir. Here's an example of a function generic over -any type `T` that implements the `Eq` trait for equality: - -```rust -fn first_element_is_equal(array1: [T; N], array2: [T; N]) -> bool - where T: Eq -{ - if (array1.len() == 0) | (array2.len() == 0) { - true - } else { - array1[0] == array2[0] - } -} - -fn main() { - assert(first_element_is_equal([1, 2, 3], [1, 5, 6])); - - // We can use first_element_is_equal for arrays of any type - // as long as we have an Eq impl for the types we pass in - let array = [MyStruct::new(), MyStruct::new()]; - assert(array_eq(array, array, MyStruct::eq)); -} - -impl Eq for MyStruct { - fn eq(self, other: MyStruct) -> bool { - self.foo == other.foo - } -} -``` - -You can find more details on traits and trait implementations on the [traits page](../concepts/traits.md). - -## Manually Specifying Generics with the Turbofish Operator - -There are times when the compiler cannot reasonably infer what type should be used for a generic, or when the developer themselves may want to manually distinguish generic type parameters. This is where the `::<>` turbofish operator comes into play. - -The `::<>` operator can follow a variable or path and can be used to manually specify generic arguments within the angle brackets. -The name "turbofish" comes from that `::<>` looks like a little fish. - -Examples: -```rust -fn main() { - let mut slice = []; - slice = slice.push_back(1); - slice = slice.push_back(2); - // Without turbofish a type annotation would be needed on the left hand side - let array = slice.as_array::<2>(); -} -``` - - -```rust -trait MyTrait { - fn ten() -> Self; -} - -impl MyTrait for Field { - fn ten() -> Self { 10 } -} - -struct Foo { - inner: T -} - -impl Foo { - fn generic_method(_self: Self) -> U where U: MyTrait { - U::ten() - } -} - -fn example() { - let foo: Foo = Foo { inner: 1 }; - // Using a type other than `Field` here (e.g. u32) would fail as - // there is no matching impl for `u32: MyTrait`. - // - // Substituting the `10` on the left hand side of this assert - // with `10 as u32` would also fail with a type mismatch as we - // are expecting a `Field` from the right hand side. - assert(10 as u32 == foo.generic_method::()); -} -``` - -## Arithmetic Generics - -In addition to numeric generics, Noir also allows a limited form of arithmetic on generics. -When you have a numeric generic such as `N`, you can use the following operators on it in a -type position: `+`, `-`, `*`, `/`, and `%`. - -Note that type checking arithmetic generics is a best effort guess from the compiler and there -are many cases of types that are equal that the compiler may not see as such. For example, -we know that `T * (N + M)` should be equal to `T*N + T*M` but the compiler does not currently -apply the distributive law and thus sees these as different types. - -Even with this limitation though, the compiler can handle common cases decently well: - -```rust -trait Serialize { - fn serialize(self) -> [Field; N]; -} - -impl Serialize<1> for Field { - fn serialize(self) -> [Field; 1] { - [self] - } -} - -impl Serialize for [T; N] - where T: Serialize { .. } - -impl Serialize for (T, U) - where T: Serialize, U: Serialize { .. } - -fn main() { - let data = (1, [2, 3, 4]); - assert_eq(data.serialize().len(), 4); -} -``` - -Note that if there is any over or underflow the types will fail to unify: - -```rust title="underflow-example" showLineNumbers -fn pop(array: [Field; N]) -> [Field; N - 1] { - let mut result: [Field; N - 1] = std::mem::zeroed(); - for i in 0..N - 1 { - result[i] = array[i]; - } - result -} - -fn main() { - // error: Could not determine array length `(0 - 1)` - pop([]); -} -``` -> Source code: test_programs/compile_failure/arithmetic_generics_underflow/src/main.nr#L1-L14 - - -This also applies if there is underflow in an intermediate calculation: - -```rust title="intermediate-underflow-example" showLineNumbers -fn main() { - // From main it looks like there's nothing sketchy going on - seems_fine([]); -} - -// Since `seems_fine` says it can receive and return any length N -fn seems_fine(array: [Field; N]) -> [Field; N] { - // But inside `seems_fine` we pop from the array which - // requires the length to be greater than zero. - - // error: Could not determine array length `(0 - 1)` - push_zero(pop(array)) -} - -fn pop(array: [Field; N]) -> [Field; N - 1] { - let mut result: [Field; N - 1] = std::mem::zeroed(); - for i in 0..N - 1 { - result[i] = array[i]; - } - result -} - -fn push_zero(array: [Field; N]) -> [Field; N + 1] { - let mut result: [Field; N + 1] = std::mem::zeroed(); - for i in 0..N { - result[i] = array[i]; - } - // index N is already zeroed - result -} -``` -> Source code: test_programs/compile_failure/arithmetic_generics_intermediate_underflow/src/main.nr#L1-L32 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/globals.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/globals.md deleted file mode 100644 index 6b8314399a2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/globals.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: Global Variables -description: - Learn about global variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, globals, global variables, constants] -sidebar_position: 8 ---- - -## Globals - - -Noir supports global variables. The global's type can be inferred by the compiler entirely: - -```rust -global N = 5; // Same as `global N: Field = 5` - -global TUPLE = (3, 2); - -fn main() { - assert(N == 5); - assert(N == TUPLE.0 + TUPLE.1); -} -``` - -:::info - -Globals can be defined as any expression, so long as they don't depend on themselves - otherwise there would be a dependency cycle! For example: - -```rust -global T = foo(T); // dependency error -``` - -::: - - -If they are initialized to a literal integer, globals can be used to specify an array's length: - -```rust -global N: u32 = 2; - -fn main(y : [Field; N]) { - assert(y[0] == y[1]) -} -``` - -A global from another module can be imported or referenced externally like any other name: - -```rust -global N = 20; - -fn main() { - assert(my_submodule::N != N); -} - -mod my_submodule { - global N: Field = 10; -} -``` - -When a global is used, Noir replaces the name with its definition on each occurrence. -This means globals defined using function calls will repeat the call each time they're used: - -```rust -global RESULT = foo(); - -fn foo() -> [Field; 100] { ... } -``` - -This is usually fine since Noir will generally optimize any function call that does not -refer to a program input into a constant. It should be kept in mind however, if the called -function performs side-effects like `println`, as these will still occur on each use. - -### Visibility - -By default, like functions, globals are private to the module they exist in. You can use `pub` -to make the global public or `pub(crate)` to make it public to just its crate: - -```rust -// This global is now public -pub global N = 5; -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/lambdas.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/lambdas.md deleted file mode 100644 index be3c7e0b5ca..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/lambdas.md +++ /dev/null @@ -1,81 +0,0 @@ ---- -title: Lambdas -description: Learn how to use anonymous functions in Noir programming language. -keywords: [Noir programming language, lambda, closure, function, anonymous function] -sidebar_position: 9 ---- - -## Introduction - -Lambdas are anonymous functions. The syntax is `|arg1, arg2, ..., argN| return_expression`. - -```rust -let add_50 = |val| val + 50; -assert(add_50(100) == 150); -``` - -A block can be used as the body of a lambda, allowing you to declare local variables inside it: - -```rust -let cool = || { - let x = 100; - let y = 100; - x + y -} - -assert(cool() == 200); -``` - -## Closures - -Inside the body of a lambda, you can use variables defined in the enclosing function. Such lambdas are called **closures**. In this example `x` is defined inside `main` and is accessed from within the lambda: - -```rust -fn main() { - let x = 100; - let closure = || x + 150; - assert(closure() == 250); -} -``` - -## Passing closures to higher-order functions - -It may catch you by surprise that the following code fails to compile: - -```rust -fn foo(f: fn () -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // error :( -} -``` - -The reason is that the closure's capture environment affects its type - we have a closure that captures two Fields and `foo` -expects a regular function as an argument - those are incompatible. -:::note - -Variables contained within the `||` are the closure's parameters, and the expression that follows it is the closure's body. The capture environment is comprised of any variables used in the closure's body that are not parameters. - -E.g. in |x| x + y, y would be a captured variable, but x would not be, since it is a parameter of the closure. - -::: -The syntax for the type of a closure is `fn[env](args) -> ret_type`, where `env` is the capture environment of the closure - -in this example that's `(Field, Field)`. - -The best solution in our case is to make `foo` generic over the environment type of its parameter, so that it can be called -with closures with any environment, as well as with regular functions: - -```rust -fn foo(f: fn[Env]() -> Field) -> Field { - f() -} - -fn main() { - let (x, y) = (50, 50); - assert(foo(|| x + y) == 100); // compiles fine - assert(foo(|| 60) == 60); // compiles fine -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/mutability.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/mutability.md deleted file mode 100644 index fdeef6a87c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/mutability.md +++ /dev/null @@ -1,121 +0,0 @@ ---- -title: Mutability -description: - Learn about mutable variables in Noir. Discover how - to declare, modify, and use them in your programs. -keywords: [noir programming language, mutability in noir, mutable variables] -sidebar_position: 8 ---- - -Variables in noir can be declared mutable via the `mut` keyword. Mutable variables can be reassigned -to via an assignment expression. - -```rust -let x = 2; -x = 3; // error: x must be mutable to be assigned to - -let mut y = 3; -let y = 4; // OK -``` - -The `mut` modifier can also apply to patterns: - -```rust -let (a, mut b) = (1, 2); -a = 11; // error: a must be mutable to be assigned to -b = 12; // OK - -let mut (c, d) = (3, 4); -c = 13; // OK -d = 14; // OK - -// etc. -let MyStruct { x: mut y } = MyStruct { x: a }; -// y is now in scope -``` - -Note that mutability in noir is local and everything is passed by value, so if a called function -mutates its parameters then the parent function will keep the old value of the parameters. - -```rust -fn main() -> pub Field { - let x = 3; - helper(x); - x // x is still 3 -} - -fn helper(mut x: i32) { - x = 4; -} -``` - -## Non-local mutability - -Non-local mutability can be achieved through the mutable reference type `&mut T`: - -```rust -fn set_to_zero(x: &mut Field) { - *x = 0; -} - -fn main() { - let mut y = 42; - set_to_zero(&mut y); - assert(*y == 0); -} -``` - -When creating a mutable reference, the original variable being referred to (`y` in this -example) must also be mutable. Since mutable references are a reference type, they must -be explicitly dereferenced via `*` to retrieve the underlying value. Note that this yields -a copy of the value, so mutating this copy will not change the original value behind the -reference: - -```rust -fn main() { - let mut x = 1; - let x_ref = &mut x; - - let mut y = *x_ref; - let y_ref = &mut y; - - x = 2; - *x_ref = 3; - - y = 4; - *y_ref = 5; - - assert(x == 3); - assert(*x_ref == 3); - assert(y == 5); - assert(*y_ref == 5); -} -``` - -Note that types in Noir are actually deeply immutable so the copy that occurs when -dereferencing is only a conceptual copy - no additional constraints will occur. - -Mutable references can also be stored within structs. Note that there is also -no lifetime parameter on these unlike rust. This is because the allocated memory -always lasts the entire program - as if it were an array of one element. - -```rust -struct Foo { - x: &mut Field -} - -impl Foo { - fn incr(mut self) { - *self.x += 1; - } -} - -fn main() { - let foo = Foo { x: &mut 0 }; - foo.incr(); - assert(*foo.x == 1); -} -``` - -In general, you should avoid non-local & shared mutability unless it is needed. Sticking -to only local mutability will improve readability and potentially improve compiler optimizations as well. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/ops.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/ops.md deleted file mode 100644 index c35c36c38a9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/ops.md +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: Logical Operations -description: - Learn about the supported arithmetic and logical operations in the Noir programming language. - Discover how to perform operations on private input types, integers, and booleans. -keywords: - [ - Noir programming language, - supported operations, - arithmetic operations, - logical operations, - predicate operators, - bitwise operations, - short-circuiting, - backend, - ] -sidebar_position: 3 ---- - -# Operations - -## Table of Supported Operations - -| Operation | Description | Requirements | -| :-------- | :------------------------------------------------------------: | -------------------------------------: | -| + | Adds two private input types together | Types must be private input | -| - | Subtracts two private input types together | Types must be private input | -| \* | Multiplies two private input types together | Types must be private input | -| / | Divides two private input types together | Types must be private input | -| ^ | XOR two private input types together | Types must be integer | -| & | AND two private input types together | Types must be integer | -| \| | OR two private input types together | Types must be integer | -| \<\< | Left shift an integer by another integer amount | Types must be integer, shift must be u8 | -| >> | Right shift an integer by another integer amount | Types must be integer, shift must be u8 | -| ! | Bitwise not of a value | Type must be integer or boolean | -| \< | returns a bool if one value is less than the other | Upper bound must have a known bit size | -| \<= | returns a bool if one value is less than or equal to the other | Upper bound must have a known bit size | -| > | returns a bool if one value is more than the other | Upper bound must have a known bit size | -| >= | returns a bool if one value is more than or equal to the other | Upper bound must have a known bit size | -| == | returns a bool if one value is equal to the other | Both types must not be constants | -| != | returns a bool if one value is not equal to the other | Both types must not be constants | - -### Predicate Operators - -`<,<=, !=, == , >, >=` are known as predicate/comparison operations because they compare two values. -This differs from the operations such as `+` where the operands are used in _computation_. - -### Bitwise Operations Example - -```rust -fn main(x : Field) { - let y = x as u32; - let z = y & y; -} -``` - -`z` is implicitly constrained to be the result of `y & y`. The `&` operand is used to denote bitwise -`&`. - -> `x & x` would not compile as `x` is a `Field` and not an integer type. - -### Logical Operators - -Noir has no support for the logical operators `||` and `&&`. This is because encoding the -short-circuiting that these operators require can be inefficient for Noir's backend. Instead you can -use the bitwise operators `|` and `&` which operate identically for booleans, just without the -short-circuiting. - -```rust -let my_val = 5; - -let mut flag = 1; -if (my_val > 6) | (my_val == 0) { - flag = 0; -} -assert(flag == 1); - -if (my_val != 10) & (my_val < 50) { - flag = 0; -} -assert(flag == 0); -``` - -### Shorthand operators - -Noir shorthand operators for most of the above operators, namely `+=, -=, *=, /=, %=, &=, |=, ^=, <<=`, and `>>=`. These allow for more concise syntax. For example: - -```rust -let mut i = 0; -i = i + 1; -``` - -could be written as: - -```rust -let mut i = 0; -i += 1; -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/oracles.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/oracles.mdx deleted file mode 100644 index 77a2ac1550a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/oracles.mdx +++ /dev/null @@ -1,29 +0,0 @@ ---- -title: Oracles -description: Dive into how Noir supports Oracles via RPC calls, and learn how to declare an Oracle in Noir with our comprehensive guide. -keywords: - - Noir - - Oracles - - RPC Calls - - Unconstrained Functions - - Programming - - Blockchain -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -Noir has support for Oracles via RPC calls. This means Noir will make an RPC call and use the return value for proof generation. - -Since Oracles are not resolved by Noir, they are [`unconstrained` functions](./unconstrained.md) - -You can declare an Oracle through the `#[oracle()]` flag. Example: - -```rust -#[oracle(get_number_sequence)] -unconstrained fn get_number_sequence(_size: Field) -> [Field] {} -``` - -The timeout for when using an external RPC oracle resolver can be set with the `NARGO_FOREIGN_CALL_TIMEOUT` environment variable. This timeout is in units of milliseconds. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/shadowing.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/shadowing.md deleted file mode 100644 index 5ce6130d201..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/shadowing.md +++ /dev/null @@ -1,44 +0,0 @@ ---- -title: Shadowing -sidebar_position: 12 ---- - -Noir allows for inheriting variables' values and re-declaring them with the same name similar to Rust, known as shadowing. - -For example, the following function is valid in Noir: - -```rust -fn main() { - let x = 5; - - { - let x = x * 2; - assert (x == 10); - } - - assert (x == 5); -} -``` - -In this example, a variable x is first defined with the value 5. - -The local scope that follows shadows the original x, i.e. creates a local mutable x based on the value of the original x. It is given a value of 2 times the original x. - -When we return to the main scope, x once again refers to just the original x, which stays at the value of 5. - -## Temporal mutability - -One way that shadowing is useful, in addition to ergonomics across scopes, is for temporarily mutating variables. - -```rust -fn main() { - let age = 30; - // age = age + 5; // Would error as `age` is immutable by default. - - let mut age = age + 5; // Temporarily mutates `age` with a new value. - - let age = age; // Locks `age`'s mutability again. - - assert (age == 35); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/unconstrained.md deleted file mode 100644 index b5221b8d2dd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/unconstrained.md +++ /dev/null @@ -1,104 +0,0 @@ ---- -title: Unconstrained Functions -description: "Learn about what unconstrained functions in Noir are, how to use them and when you'd want to." - -keywords: [Noir programming language, unconstrained, open] -sidebar_position: 5 ---- - -Unconstrained functions are functions which do not constrain any of the included computation and allow for non-deterministic computation. - -## Why? - -Zero-knowledge (ZK) domain-specific languages (DSL) enable developers to generate ZK proofs from their programs by compiling code down to the constraints of an NP complete language (such as R1CS or PLONKish languages). However, the hard bounds of a constraint system can be very limiting to the functionality of a ZK DSL. - -Enabling a circuit language to perform unconstrained execution is a powerful tool. Said another way, unconstrained execution lets developers generate witnesses from code that does not generate any constraints. Being able to execute logic outside of a circuit is critical for both circuit performance and constructing proofs on information that is external to a circuit. - -Fetching information from somewhere external to a circuit can also be used to enable developers to improve circuit efficiency. - -A ZK DSL does not just prove computation, but proves that some computation was handled correctly. Thus, it is necessary that when we switch from performing some operation directly inside of a circuit to inside of an unconstrained environment that the appropriate constraints are still laid down elsewhere in the circuit. - -## Example - -An in depth example might help drive the point home. This example comes from the excellent [post](https://discord.com/channels/1113924620781883405/1124022445054111926/1128747641853972590) by Tom in the Noir Discord. - -Let's look at how we can optimize a function to turn a `u72` into an array of `u8`s. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u72 & 0xff) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 91 -Backend circuit size: 3619 -``` - -A lot of the operations in this function are optimized away by the compiler (all the bit-shifts turn into divisions by constants). However we can save a bunch of gates by casting to u8 a bit earlier. This automatically truncates the bit-shifted value to fit in a u8 which allows us to remove the AND against 0xff. This saves us ~480 gates in total. - -```rust -fn main(num: u72) -> pub [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8)) as u8; - } - - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 75 -Backend circuit size: 3143 -``` - -Those are some nice savings already but we can do better. This code is all constrained so we're proving every step of calculating out using num, but we don't actually care about how we calculate this, just that it's correct. This is where brillig comes in. - -It turns out that truncating a u72 into a u8 is hard to do inside a snark, each time we do as u8 we lay down 4 ACIR opcodes which get converted into multiple gates. It's actually much easier to calculate num from out than the other way around. All we need to do is multiply each element of out by a constant and add them all together, both relatively easy operations inside a snark. - -We can then run `u72_to_u8` as unconstrained brillig code in order to calculate out, then use that result in our constrained function and assert that if we were to do the reverse calculation we'd get back num. This looks a little like the below: - -```rust -fn main(num: u72) -> pub [u8; 8] { - let out = unsafe { - u72_to_u8(num) - }; - - let mut reconstructed_num: u72 = 0; - for i in 0..8 { - reconstructed_num += (out[i] as u72 << (56 - (8 * i))); - } - assert(num == reconstructed_num); - out -} - -unconstrained fn u72_to_u8(num: u72) -> [u8; 8] { - let mut out: [u8; 8] = [0; 8]; - for i in 0..8 { - out[i] = (num >> (56 - (i * 8))) as u8; - } - out -} -``` - -``` -Total ACIR opcodes generated for language PLONKCSat { width: 3 }: 78 -Backend circuit size: 2902 -``` - -This ends up taking off another ~250 gates from our circuit! We've ended up with more ACIR opcodes than before but they're easier for the backend to prove (resulting in fewer gates). - -Note that in order to invoke unconstrained functions we need to wrap them in an `unsafe` block, -to make it clear that the call is unconstrained. - -Generally we want to use brillig whenever there's something that's easy to verify but hard to compute within the circuit. For example, if you wanted to calculate a square root of a number it'll be a much better idea to calculate this in brillig and then assert that if you square the result you get back your number. - -## Break and Continue - -In addition to loops over runtime bounds, `break` and `continue` are also available in unconstrained code. See [break and continue](../concepts/control_flow.md#break-and-continue) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/_category_.json deleted file mode 100644 index 1debcfe7675..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Modules, Packages and Crates", - "position": 2, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/crates_and_packages.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/crates_and_packages.md deleted file mode 100644 index 95ee9f52ab2..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/crates_and_packages.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Crates and Packages -description: Learn how to use Crates and Packages in your Noir project -keywords: [Nargo, dependencies, package management, crates, package] -sidebar_position: 0 ---- - -## Crates - -A crate is the smallest amount of code that the Noir compiler considers at a time. -Crates can contain modules, and the modules may be defined in other files that get compiled with the crate, as we’ll see in the coming sections. - -### Crate Types - -A Noir crate can come in several forms: binaries, libraries or contracts. - -#### Binaries - -_Binary crates_ are programs which you can compile to an ACIR circuit which you can then create proofs against. Each must have a function called `main` that defines the ACIR circuit which is to be proved. - -#### Libraries - -_Library crates_ don't have a `main` function and they don't compile down to ACIR. Instead they define functionality intended to be shared with multiple projects, and eventually included in a binary crate. - -#### Contracts - -Contract crates are similar to binary crates in that they compile to ACIR which you can create proofs against. They are different in that they do not have a single `main` function, but are a collection of functions to be deployed to the [Aztec network](https://aztec.network). You can learn more about the technical details of Aztec in the [monorepo](https://github.com/AztecProtocol/aztec-packages) or contract [examples](https://github.com/AztecProtocol/aztec-packages/tree/master/noir-projects/noir-contracts/contracts). - -### Crate Root - -Every crate has a root, which is the source file that the compiler starts, this is also known as the root module. The Noir compiler does not enforce any conditions on the name of the file which is the crate root, however if you are compiling via Nargo the crate root must be called `lib.nr` or `main.nr` for library or binary crates respectively. - -## Packages - -A Nargo _package_ is a collection of one of more crates that provides a set of functionality. A package must include a Nargo.toml file. - -A package _must_ contain either a library or a binary crate, but not both. - -### Differences from Cargo Packages - -One notable difference between Rust's Cargo and Noir's Nargo is that while Cargo allows a package to contain an unlimited number of binary crates and a single library crate, Nargo currently only allows a package to contain a single crate. - -In future this restriction may be lifted to allow a Nargo package to contain both a binary and library crate or multiple binary crates. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/dependencies.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/dependencies.md deleted file mode 100644 index 24e02de08fe..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/dependencies.md +++ /dev/null @@ -1,124 +0,0 @@ ---- -title: Dependencies -description: - Learn how to specify and manage dependencies in Nargo, allowing you to upload packages to GitHub - and use them easily in your project. -keywords: [Nargo, dependencies, GitHub, package management, versioning] -sidebar_position: 1 ---- - -Nargo allows you to upload packages to GitHub and use them as dependencies. - -## Specifying a dependency - -Specifying a dependency requires a tag to a specific commit and the git url to the url containing -the package. - -Currently, there are no requirements on the tag contents. If requirements are added, it would follow -semver 2.0 guidelines. - -> Note: Without a `tag` , there would be no versioning and dependencies would change each time you -> compile your project. - -For example, to add the [ecrecover-noir library](https://github.com/colinnielsen/ecrecover-noir) to your project, add it to `Nargo.toml`: - -```toml -# Nargo.toml - -[dependencies] -ecrecover = {tag = "v0.8.0", git = "https://github.com/colinnielsen/ecrecover-noir"} -``` - -If the module is in a subdirectory, you can define a subdirectory in your git repository, for example: - -```toml -# Nargo.toml - -[dependencies] -easy_private_token_contract = {tag ="v0.1.0-alpha62", git = "https://github.com/AztecProtocol/aztec-packages", directory = "noir-contracts/contracts/easy_private_token_contract"} -``` - -## Specifying a local dependency - -You can also specify dependencies that are local to your machine. - -For example, this file structure has a library and binary crate - -```tree -├── binary_crate -│   ├── Nargo.toml -│   └── src -│   └── main.nr -└── lib_a - ├── Nargo.toml - └── src - └── lib.nr -``` - -Inside of the binary crate, you can specify: - -```toml -# Nargo.toml - -[dependencies] -lib_a = { path = "../lib_a" } -``` - -## Importing dependencies - -You can import a dependency to a Noir file using the following syntax. For example, to import the -ecrecover-noir library and local lib_a referenced above: - -```rust -use ecrecover; -use lib_a; -``` - -You can also import only the specific parts of dependency that you want to use, like so: - -```rust -use std::hash::sha256; -use std::scalar_mul::fixed_base_embedded_curve; -``` - -Lastly, as demonstrated in the -[elliptic curve example](../standard_library/cryptographic_primitives/ec_primitives.md#examples), you -can import multiple items in the same line by enclosing them in curly braces: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; -``` - -We don't have a way to consume libraries from inside a [workspace](./workspaces.md) as external dependencies right now. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -## Dependencies of Dependencies - -Note that when you import a dependency, you also get access to all of the dependencies of that package. - -For example, the [phy_vector](https://github.com/resurgencelabs/phy_vector) library imports an [fraction](https://github.com/resurgencelabs/fraction) library. If you're importing the phy_vector library, then you can access the functions in fractions library like so: - -```rust -use phy_vector; - -fn main(x : Field, y : pub Field) { - //... - let f = phy_vector::fraction::toFraction(true, 2, 1); - //... -} -``` - -## Available Libraries - -Noir does not currently have an official package manager. You can find a list of available Noir libraries in the [awesome-noir repo here](https://github.com/noir-lang/awesome-noir#libraries). - -Some libraries that are available today include: - -- [Standard Library](https://github.com/noir-lang/noir/tree/master/noir_stdlib) - the Noir Standard Library -- [Ethereum Storage Proof Verification](https://github.com/aragonzkresearch/noir-trie-proofs) - a library that contains the primitives necessary for RLP decoding (in the form of look-up table construction) and Ethereum state and storage proof verification (or verification of any trie proof involving 32-byte long keys) -- [BigInt](https://github.com/shuklaayush/noir-bigint) - a library that provides a custom BigUint56 data type, allowing for computations on large unsigned integers -- [ECrecover](https://github.com/colinnielsen/ecrecover-noir/tree/main) - a library to verify an ECDSA signature and return the source Ethereum address -- [Sparse Merkle Tree Verifier](https://github.com/vocdoni/smtverifier-noir/tree/main) - a library for verification of sparse Merkle trees -- [Signed Int](https://github.com/resurgencelabs/signed_int) - a library for accessing a custom Signed Integer data type, allowing access to negative numbers on Noir -- [Fraction](https://github.com/resurgencelabs/fraction) - a library for accessing fractional number data type in Noir, allowing results that aren't whole numbers diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/modules.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/modules.md deleted file mode 100644 index 14aa1f0579a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/modules.md +++ /dev/null @@ -1,221 +0,0 @@ ---- -title: Modules -description: - Learn how to organize your files using modules in Noir, following the same convention as Rust's - module system. Examples included. -keywords: [Noir, Rust, modules, organizing files, sub-modules] -sidebar_position: 2 ---- - -Noir's module system follows the same convention as the _newer_ version of Rust's module system. - -## Purpose of Modules - -Modules are used to organize files. Without modules all of your code would need to live in a single -file. In Noir, the compiler does not automatically scan all of your files to detect modules. This -must be done explicitly by the developer. - -## Examples - -### Importing a module in the crate root - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo.nr` - -```rust -fn from_foo() {} -``` - -In the above snippet, the crate root is the `src/main.nr` file. The compiler sees the module -declaration `mod foo` which prompts it to look for a foo.nr file. - -Visually this module hierarchy looks like the following : - -``` -crate - ├── main - │ - └── foo - └── from_foo - -``` - -The module filename may also be the name of the module as a directory with the contents in a -file named `mod.nr` within that directory. The above example can alternatively be expressed like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::hello_world(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -fn from_foo() {} -``` - -Note that it's an error to have both files `src/foo.nr` and `src/foo/mod.nr` in the filesystem. - -### Importing a module throughout the tree - -All modules are accessible from the `crate::` namespace. - -``` -crate - ├── bar - ├── foo - └── main - -``` - -In the above snippet, if `bar` would like to use functions in `foo`, it can do so by `use crate::foo::function_name`. - -### Sub-modules - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -fn from_bar() {} -``` - -In the above snippet, we have added an extra module to the module tree; `bar`. `bar` is a submodule -of `foo` hence we declare bar in `foo.nr` with `mod bar`. Since `foo` is not the crate root, the -compiler looks for the file associated with the `bar` module in `src/foo/bar.nr` - -Visually the module hierarchy looks as follows: - -``` -crate - ├── main - │ - └── foo - ├── from_foo - └── bar - └── from_bar -``` - -Similar to importing a module in the crate root, modules can be placed in a `mod.nr` file, like this: - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo/mod.nr` - -```rust -mod bar; -fn from_foo() {} -``` - -Filename : `src/foo/bar/mod.nr` - -```rust -fn from_bar() {} -``` - -### Referencing a parent module - -Given a submodule, you can refer to its parent module using the `super` keyword. - -Filename : `src/main.nr` - -```rust -mod foo; - -fn main() { - foo::from_foo(); -} -``` - -Filename : `src/foo.nr` - -```rust -mod bar; - -fn from_foo() {} -``` - -Filename : `src/foo/bar.nr` - -```rust -// Same as bar::from_foo -use super::from_foo; - -fn from_bar() { - from_foo(); // invokes super::from_foo(), which is bar::from_foo() - super::from_foo(); // also invokes bar::from_foo() -} -``` - -### `use` visibility - -`use` declarations are private to the containing module, by default. However, like functions, -they can be marked as `pub` or `pub(crate)`. Such a use declaration serves to _re-export_ a name. -A public `use` declaration can therefore redirect some public name to a different target definition: -even a definition with a private canonical path, inside a different module. - -An example of re-exporting: - -```rust -mod some_module { - pub use foo::{bar, baz}; - mod foo { - pub fn bar() {} - pub fn baz() {} - } -} - -fn main() { - some_module::bar(); - some_module::baz(); -} -``` - -In this example, the module `some_module` re-exports two public names defined in `foo`. - -### Visibility - -By default, like functions, modules are private to the module (or crate) they exist in. You can use `pub` -to make the module public or `pub(crate)` to make it public to just its crate: - -```rust -// This module is now public and can be seen by other crates. -pub mod foo; -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/workspaces.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/workspaces.md deleted file mode 100644 index 513497f12bf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/modules_packages_crates/workspaces.md +++ /dev/null @@ -1,42 +0,0 @@ ---- -title: Workspaces -sidebar_position: 3 ---- - -Workspaces are a feature of nargo that allow you to manage multiple related Noir packages in a single repository. A workspace is essentially a group of related projects that share common build output directories and configurations. - -Each Noir project (with it's own Nargo.toml file) can be thought of as a package. Each package is expected to contain exactly one "named circuit", being the "name" defined in Nargo.toml with the program logic defined in `./src/main.nr`. - -For a project with the following structure: - -```tree -├── crates -│ ├── a -│ │ ├── Nargo.toml -│ │ └── Prover.toml -│ │ └── src -│ │ └── main.nr -│ └── b -│ ├── Nargo.toml -│ └── Prover.toml -│ └── src -│ └── main.nr -│ -└── Nargo.toml -``` - -You can define a workspace in Nargo.toml like so: - -```toml -[workspace] -members = ["crates/a", "crates/b"] -default-member = "crates/a" -``` - -`members` indicates which packages are included in the workspace. As such, all member packages of a workspace will be processed when the `--workspace` flag is used with various commands or if a `default-member` is not specified. - -`default-member` indicates which package various commands process by default. - -Libraries can be defined in a workspace. Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. - -Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/_category_.json deleted file mode 100644 index af04c0933fd..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Standard Library", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/black_box_fns.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/black_box_fns.md deleted file mode 100644 index d6079ab182c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/black_box_fns.md +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Black Box Functions -description: Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. -keywords: [noir, black box functions] ---- - -Black box functions are functions in Noir that rely on backends implementing support for specialized constraints. This makes certain zk-snark unfriendly computations cheaper than if they were implemented in Noir. - -The ACVM spec defines a set of blackbox functions which backends will be expected to implement. This allows backends to use optimized implementations of these constraints if they have them, however they may also fallback to less efficient naive implementations if not. - -## Function list - -Here is a list of the current black box functions: - -- [AES128](./cryptographic_primitives/ciphers.mdx#aes128) -- [SHA256](./cryptographic_primitives/hashes.mdx#sha256) -- [Schnorr signature verification](./cryptographic_primitives/schnorr.mdx) -- [Blake2s](./cryptographic_primitives/hashes.mdx#blake2s) -- [Blake3](./cryptographic_primitives/hashes.mdx#blake3) -- [Pedersen Hash](./cryptographic_primitives/hashes.mdx#pedersen_hash) -- [Pedersen Commitment](./cryptographic_primitives/hashes.mdx#pedersen_commitment) -- [ECDSA signature verification](./cryptographic_primitives/ecdsa_sig_verification.mdx) -- [Embedded curve operations (MSM, addition, ...)](./cryptographic_primitives/embedded_curve_ops.mdx) -- AND -- XOR -- RANGE -- [Keccak256](./cryptographic_primitives/hashes.mdx#keccak256) -- [Recursive proof verification](./recursion.mdx) - -Most black box functions are included as part of the Noir standard library, however `AND`, `XOR` and `RANGE` are used as part of the Noir language syntax. For instance, using the bitwise operator `&` will invoke the `AND` black box function. - -You can view the black box functions defined in the ACVM code [here](https://github.com/noir-lang/noir/blob/master/acvm-repo/acir/src/circuit/black_box_functions.rs). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/bn254.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/bn254.md deleted file mode 100644 index 3294f005dbb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/bn254.md +++ /dev/null @@ -1,46 +0,0 @@ ---- -title: Bn254 Field Library ---- - -Noir provides a module in standard library with some optimized functions for bn254 Fr in `std::field::bn254`. - -## decompose - -```rust -fn decompose(x: Field) -> (Field, Field) {} -``` - -Decomposes a single field into two fields, low and high. The low field contains the lower 16 bytes of the input field and the high field contains the upper 16 bytes of the input field. Both field results are range checked to 128 bits. - - -## assert_gt - -```rust -fn assert_gt(a: Field, b: Field) {} -``` - -Asserts that a > b. This will generate less constraints than using `assert(gt(a, b))`. - -## assert_lt - -```rust -fn assert_lt(a: Field, b: Field) {} -``` - -Asserts that a < b. This will generate less constraints than using `assert(lt(a, b))`. - -## gt - -```rust -fn gt(a: Field, b: Field) -> bool {} -``` - -Returns true if a > b. - -## lt - -```rust -fn lt(a: Field, b: Field) -> bool {} -``` - -Returns true if a < b. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/boundedvec.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/boundedvec.md deleted file mode 100644 index 509b214bf3a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/boundedvec.md +++ /dev/null @@ -1,419 +0,0 @@ ---- -title: Bounded Vectors -keywords: [noir, vector, bounded vector, slice] -sidebar_position: 1 ---- - -A `BoundedVec` is a growable storage similar to a `Vec` except that it -is bounded with a maximum possible length. Unlike `Vec`, `BoundedVec` is not implemented -via slices and thus is not subject to the same restrictions slices are (notably, nested -slices - and thus nested vectors as well - are disallowed). - -Since a BoundedVec is backed by a normal array under the hood, growing the BoundedVec by -pushing an additional element is also more efficient - the length only needs to be increased -by one. - -For these reasons `BoundedVec` should generally be preferred over `Vec` when there -is a reasonable maximum bound that can be placed on the vector. - -Example: - -```rust -let mut vector: BoundedVec = BoundedVec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -assert(vector.max_len() == 10); -``` - -## Methods - -### new - -```rust -pub fn new() -> Self -``` - -Creates a new, empty vector of length zero. - -Since this container is backed by an array internally, it still needs an initial value -to give each element. To resolve this, each element is zeroed internally. This value -is guaranteed to be inaccessible unless `get_unchecked` is used. - -Example: - -```rust -let empty_vector: BoundedVec = BoundedVec::new(); -assert(empty_vector.len() == 0); -``` - -Note that whenever calling `new` the maximum length of the vector should always be specified -via a type signature: - -```rust title="new_example" showLineNumbers -fn good() -> BoundedVec { - // Ok! MaxLen is specified with a type annotation - let v1: BoundedVec = BoundedVec::new(); - let v2 = BoundedVec::new(); - - // Ok! MaxLen is known from the type of `good`'s return value - v2 -} - -fn bad() { - // Error: Type annotation needed - // The compiler can't infer `MaxLen` from this code. - let mut v3 = BoundedVec::new(); - v3.push(5); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L11-L27 - - -This defaulting of `MaxLen` (and numeric generics in general) to zero may change in future noir versions -but for now make sure to use type annotations when using bounded vectors. Otherwise, you will receive a constraint failure at runtime when the vec is pushed to. - -### get - -```rust -pub fn get(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this -will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - let last = v.get(v.len() - 1); - assert(first != last); -} -``` - -### get_unchecked - -```rust -pub fn get_unchecked(self, index: u64) -> T { -``` - -Retrieves an element from the vector at the given index, starting from zero, without -performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, -it is unsafe! Use at your own risk! - -Example: - -```rust title="get_unchecked_example" showLineNumbers -fn sum_of_first_three(v: BoundedVec) -> u32 { - // Always ensure the length is larger than the largest - // index passed to get_unchecked - assert(v.len() > 2); - let first = v.get_unchecked(0); - let second = v.get_unchecked(1); - let third = v.get_unchecked(2); - first + second + third -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L54-L64 - - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -If the given index is equal to or greater than the length of the vector, this will issue a constraint failure. - -Example: - -```rust -fn foo(v: BoundedVec) { - let first = v.get(0); - assert(first != 42); - v.set(0, 42); - let new_first = v.get(0); - assert(new_first == 42); -} -``` - -### set_unchecked - -```rust -pub fn set_unchecked(&mut self: Self, index: u64, value: T) -> T { -``` - -Writes an element to the vector at the given index, starting from zero, without performing a bounds check. - -Since this function does not perform a bounds check on length before accessing the element, it is unsafe! Use at your own risk! - -Example: - -```rust title="set_unchecked_example" showLineNumbers -fn set_unchecked_example() { - let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([1, 2]); - - // Here we're safely writing within the valid range of `vec` - // `vec` now has the value [42, 2] - vec.set_unchecked(0, 42); - - // We can then safely read this value back out of `vec`. - // Notice that we use the checked version of `get` which would prevent reading unsafe values. - assert_eq(vec.get(0), 42); - - // We've now written past the end of `vec`. - // As this index is still within the maximum potential length of `v`, - // it won't cause a constraint failure. - vec.set_unchecked(2, 42); - println(vec); - - // This will write past the end of the maximum potential length of `vec`, - // it will then trigger a constraint failure. - vec.set_unchecked(5, 42); - println(vec); -} -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L67-L91 - - - -### push - -```rust -pub fn push(&mut self, elem: T) { -``` - -Pushes an element to the end of the vector. This increases the length -of the vector by one. - -Panics if the new length of the vector will be greater than the max length. - -Example: - -```rust title="bounded-vec-push-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - v.push(1); - v.push(2); - - // Panics with failed assertion "push out of bounds" - v.push(3); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L95-L103 - - -### pop - -```rust -pub fn pop(&mut self) -> T -``` - -Pops the element at the end of the vector. This will decrease the length -of the vector by one. - -Panics if the vector is empty. - -Example: - -```rust title="bounded-vec-pop-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.push(1); - v.push(2); - - let two = v.pop(); - let one = v.pop(); - - assert(two == 2); - assert(one == 1); - // error: cannot pop from an empty vector - // let _ = v.pop(); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L108-L120 - - -### len - -```rust -pub fn len(self) -> u64 { -``` - -Returns the current length of this vector - -Example: - -```rust title="bounded-vec-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - assert(v.len() == 0); - - v.push(100); - assert(v.len() == 1); - - v.push(200); - v.push(300); - v.push(400); - assert(v.len() == 4); - - let _ = v.pop(); - let _ = v.pop(); - assert(v.len() == 2); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L125-L140 - - -### max_len - -```rust -pub fn max_len(_self: BoundedVec) -> u64 { -``` - -Returns the maximum length of this vector. This is always -equal to the `MaxLen` parameter this vector was initialized with. - -Example: - -```rust title="bounded-vec-max-len-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.max_len() == 5); - v.push(10); - assert(v.max_len() == 5); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L145-L151 - - -### storage - -```rust -pub fn storage(self) -> [T; MaxLen] { -``` - -Returns the internal array within this vector. -Since arrays in Noir are immutable, mutating the returned storage array will not mutate -the storage held internally by this vector. - -Note that uninitialized elements may be zeroed out! - -Example: - -```rust title="bounded-vec-storage-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - - assert(v.storage() == [0, 0, 0, 0, 0]); - - v.push(57); - assert(v.storage() == [57, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L156-L163 - - -### extend_from_array - -```rust -pub fn extend_from_array(&mut self, array: [T; Len]) -``` - -Pushes each element from the given array to this vector. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-array-example" showLineNumbers -let mut vec: BoundedVec = BoundedVec::new(); - vec.extend_from_array([2, 4]); - - assert(vec.len == 2); - assert(vec.get(0) == 2); - assert(vec.get(1) == 4); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L168-L175 - - -### extend_from_bounded_vec - -```rust -pub fn extend_from_bounded_vec(&mut self, vec: BoundedVec) -``` - -Pushes each element from the other vector to this vector. The length of -the other vector is left unchanged. - -Panics if pushing each element would cause the length of this vector -to exceed the maximum length. - -Example: - -```rust title="bounded-vec-extend-from-bounded-vec-example" showLineNumbers -let mut v1: BoundedVec = BoundedVec::new(); - let mut v2: BoundedVec = BoundedVec::new(); - - v2.extend_from_array([1, 2, 3]); - v1.extend_from_bounded_vec(v2); - - assert(v1.storage() == [1, 2, 3, 0, 0]); - assert(v2.storage() == [1, 2, 3, 0, 0, 0, 0]); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L180-L189 - - -### from_array - -```rust -pub fn from_array(array: [T; Len]) -> Self -``` - -Creates a new vector, populating it with values derived from an array input. -The maximum length of the vector is determined based on the type signature. - -Example: -```rust -let bounded_vec: BoundedVec = BoundedVec::from_array([1, 2, 3]) -``` - -### map - -```rust -pub fn map(self, f: fn[Env](T) -> U) -> BoundedVec -``` - -Creates a new vector of equal size by calling a closure on each element in this vector. - -Example: - -```rust title="bounded-vec-map-example" showLineNumbers -let vec: BoundedVec = BoundedVec::from_array([1, 2, 3, 4]); - let result = vec.map(|value| value * 2); -``` -> Source code: noir_stdlib/src/collections/bounded_vec.nr#L495-L498 - - -### any - -```rust -pub fn any(self, predicate: fn[Env](T) -> bool) -> bool -``` - -Returns true if the given predicate returns true for any element -in this vector. - -Example: - -```rust title="bounded-vec-any-example" showLineNumbers -let mut v: BoundedVec = BoundedVec::new(); - v.extend_from_array([2, 4, 6]); - - let all_even = !v.any(|elem: u32| elem % 2 != 0); - assert(all_even); -``` -> Source code: test_programs/noir_test_success/bounded_vec/src/main.nr#L256-L262 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/index.md deleted file mode 100644 index ea84c6d5c21..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/index.md +++ /dev/null @@ -1,5 +0,0 @@ ---- -title: Containers -description: Container types provided by Noir's standard library for storing and retrieving data -keywords: [containers, data types, vec, hashmap] ---- diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/vec.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/vec.mdx deleted file mode 100644 index 475011922f8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/vec.mdx +++ /dev/null @@ -1,170 +0,0 @@ ---- -title: Vectors -description: Delve into the Vec data type in Noir. Learn about its methods, practical examples, and best practices for using Vectors in your Noir code. -keywords: [noir, vector type, methods, examples, dynamic arrays] -sidebar_position: 6 ---- - -import Experimental from '@site/src/components/Notes/_experimental.mdx'; - - - -A vector is a collection type similar to Rust's `Vec` type. In Noir, it is a convenient way to use slices as mutable arrays. - -Example: - -```rust -let mut vector: Vec = Vec::new(); -for i in 0..5 { - vector.push(i); -} -assert(vector.len() == 5); -``` - -## Methods - -### new - -Creates a new, empty vector. - -```rust -pub fn new() -> Self -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### from_slice - -Creates a vector containing each element from a given slice. Mutations to the resulting vector will not affect the original slice. - -```rust -pub fn from_slice(slice: [T]) -> Self -``` - -Example: - -```rust -let slice: [Field] = &[1, 2, 3]; -let vector_from_slice = Vec::from_slice(slice); -assert(vector_from_slice.len() == 3); -``` - -### len - -Returns the number of elements in the vector. - -```rust -pub fn len(self) -> Field -``` - -Example: - -```rust -let empty_vector: Vec = Vec::new(); -assert(empty_vector.len() == 0); -``` - -### get - -Retrieves an element from the vector at a given index. Panics if the index points beyond the vector's end. - -```rust -pub fn get(self, index: Field) -> T -``` - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -``` - -### set - -```rust -pub fn set(&mut self: Self, index: u64, value: T) { -``` - -Writes an element to the vector at the given index, starting from zero. - -Panics if the index points beyond the vector's end. - -Example: - -```rust -let vector: Vec = Vec::from_slice(&[10, 20, 30]); -assert(vector.get(1) == 20); -vector.set(1, 42); -assert(vector.get(1) == 42); -``` - -### push - -Adds a new element to the vector's end, returning a new vector with a length one greater than the original unmodified vector. - -```rust -pub fn push(&mut self, elem: T) -``` - -Example: - -```rust -let mut vector: Vec = Vec::new(); -vector.push(10); -assert(vector.len() == 1); -``` - -### pop - -Removes an element from the vector's end, returning a new vector with a length one less than the original vector, along with the removed element. Panics if the vector's length is zero. - -```rust -pub fn pop(&mut self) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20]); -let popped_elem = vector.pop(); -assert(popped_elem == 20); -assert(vector.len() == 1); -``` - -### insert - -Inserts an element at a specified index, shifting subsequent elements to the right. - -```rust -pub fn insert(&mut self, index: Field, elem: T) -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 30]); -vector.insert(1, 20); -assert(vector.get(1) == 20); -``` - -### remove - -Removes an element at a specified index, shifting subsequent elements to the left, and returns the removed element. - -```rust -pub fn remove(&mut self, index: Field) -> T -``` - -Example: - -```rust -let mut vector = Vec::from_slice(&[10, 20, 30]); -let removed_elem = vector.remove(1); -assert(removed_elem == 20); -assert(vector.len() == 2); -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/_category_.json deleted file mode 100644 index 5d694210bbf..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 0, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ciphers.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ciphers.mdx deleted file mode 100644 index d6a5e1a79eb..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ciphers.mdx +++ /dev/null @@ -1,32 +0,0 @@ ---- -title: Ciphers -description: - Learn about the implemented ciphers ready to use for any Noir project -keywords: - [ciphers, Noir project, aes128, encrypt] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## aes128 - -Given a plaintext as an array of bytes, returns the corresponding aes128 ciphertext (CBC mode). Input padding is automatically performed using PKCS#7, so that the output length is `input.len() + (16 - input.len() % 16)`. - -```rust title="aes128" showLineNumbers -pub fn aes128_encrypt(input: [u8; N], iv: [u8; 16], key: [u8; 16]) -> [u8] {} -``` -> Source code: noir_stdlib/src/aes128.nr#L2-L4 - - -```rust -fn main() { - let input: [u8; 4] = [0, 12, 3, 15] // Random bytes, will be padded to 16 bytes. - let iv: [u8; 16] = [0; 16]; // Initialisation vector - let key: [u8; 16] = [0; 16] // AES key - let ciphertext = std::aes128::aes128_encrypt(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()); // In this case, the output length will be 16 bytes. -} -``` - - - \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ec_primitives.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ec_primitives.md deleted file mode 100644 index f262d8160d6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ec_primitives.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: Elliptic Curve Primitives -keywords: [cryptographic primitives, Noir project] -sidebar_position: 4 ---- - -Data structures and methods on them that allow you to carry out computations involving elliptic -curves over the (mathematical) field corresponding to `Field`. For the field currently at our -disposal, applications would involve a curve embedded in BN254, e.g. the -[Baby Jubjub curve](https://eips.ethereum.org/EIPS/eip-2494). - -## Data structures - -### Elliptic curve configurations - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Curve`), i.e. the specific elliptic -curve you want to use, which would be specified using any one of the methods -`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::new` which take the coefficients in the -defining equation together with a generator point as parameters. You can find more detail in the -comments in -[`noir_stdlib/src/ec/mod.nr`](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr), but -the gist of it is that the elliptic curves of interest are usually expressed in one of the standard -forms implemented here (Twisted Edwards, Montgomery and Short Weierstraß), and in addition to that, -you could choose to use `affine` coordinates (Cartesian coordinates - the usual (x,y) - possibly -together with a point at infinity) or `curvegroup` coordinates (some form of projective coordinates -requiring more coordinates but allowing for more efficient implementations of elliptic curve -operations). Conversions between all of these forms are provided, and under the hood these -conversions are done whenever an operation is more efficient in a different representation (or a -mixed coordinate representation is employed). - -### Points - -(`std::ec::{tecurve,montcurve,swcurve}::{affine,curvegroup}::Point`), i.e. points lying on the -elliptic curve. For a curve configuration `c` and a point `p`, it may be checked that `p` -does indeed lie on `c` by calling `c.contains(p1)`. - -## Methods - -(given a choice of curve representation, e.g. use `std::ec::tecurve::affine::Curve` and use -`std::ec::tecurve::affine::Point`) - -- The **zero element** is given by `Point::zero()`, and we can verify whether a point `p: Point` is - zero by calling `p.is_zero()`. -- **Equality**: Points `p1: Point` and `p2: Point` may be checked for equality by calling - `p1.eq(p2)`. -- **Addition**: For `c: Curve` and points `p1: Point` and `p2: Point` on the curve, adding these two - points is accomplished by calling `c.add(p1,p2)`. -- **Negation**: For a point `p: Point`, `p.negate()` is its negation. -- **Subtraction**: For `c` and `p1`, `p2` as above, subtracting `p2` from `p1` is accomplished by - calling `c.subtract(p1,p2)`. -- **Scalar multiplication**: For `c` as above, `p: Point` a point on the curve and `n: Field`, - scalar multiplication is given by `c.mul(n,p)`. If instead `n :: [u1; N]`, i.e. `n` is a bit - array, the `bit_mul` method may be used instead: `c.bit_mul(n,p)` -- **Multi-scalar multiplication**: For `c` as above and arrays `n: [Field; N]` and `p: [Point; N]`, - multi-scalar multiplication is given by `c.msm(n,p)`. -- **Coordinate representation conversions**: The `into_group` method converts a point or curve - configuration in the affine representation to one in the CurveGroup representation, and - `into_affine` goes in the other direction. -- **Curve representation conversions**: `tecurve` and `montcurve` curves and points are equivalent - and may be converted between one another by calling `into_montcurve` or `into_tecurve` on their - configurations or points. `swcurve` is more general and a curve c of one of the other two types - may be converted to this representation by calling `c.into_swcurve()`, whereas a point `p` lying - on the curve given by `c` may be mapped to its corresponding `swcurve` point by calling - `c.map_into_swcurve(p)`. -- **Map-to-curve methods**: The Elligator 2 method of mapping a field element `n: Field` into a - `tecurve` or `montcurve` with configuration `c` may be called as `c.elligator2_map(n)`. For all of - the curve configurations, the SWU map-to-curve method may be called as `c.swu_map(z,n)`, where - `z: Field` depends on `Field` and `c` and must be chosen by the user (the conditions it needs to - satisfy are specified in the comments - [here](https://github.com/noir-lang/noir/blob/master/noir_stdlib/src/ec/mod.nr)). - -## Examples - -The -[ec_baby_jubjub test](https://github.com/noir-lang/noir/blob/master/test_programs/compile_success_empty/ec_baby_jubjub/src/main.nr) -illustrates all of the above primitives on various forms of the Baby Jubjub curve. A couple of more -interesting examples in Noir would be: - -Public-key cryptography: Given an elliptic curve and a 'base point' on it, determine the public key -from the private key. This is a matter of using scalar multiplication. In the case of Baby Jubjub, -for example, this code would do: - -```rust -use std::ec::tecurve::affine::{Curve, Point}; - -fn bjj_pub_key(priv_key: Field) -> Point -{ - - let bjj = Curve::new(168700, 168696, G::new(995203441582195749578291179787384436505546430278305826713579947235728471134,5472060717959818805561601436314318772137091100104008585924551046643952123905)); - - let base_pt = Point::new(5299619240641551281634865583518297030282874472190772894086521144482721001553, 16950150798460657717958625567821834550301663161624707787222815936182638968203); - - bjj.mul(priv_key,base_pt) -} -``` - -This would come in handy in a Merkle proof. - -- EdDSA signature verification: This is a matter of combining these primitives with a suitable hash - function. See - [feat(stdlib): EdDSA sig verification noir#1136](https://github.com/noir-lang/noir/pull/1136) for - the case of Baby Jubjub and the Poseidon hash function. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx deleted file mode 100644 index 8d96027b42c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx +++ /dev/null @@ -1,98 +0,0 @@ ---- -title: ECDSA Signature Verification -description: Learn about the cryptographic primitives regarding ECDSA over the secp256k1 and secp256r1 curves -keywords: [cryptographic primitives, Noir project, ecdsa, secp256k1, secp256r1, signatures] -sidebar_position: 3 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -Noir supports ECDSA signatures verification over the secp256k1 and secp256r1 curves. - -## ecdsa_secp256k1::verify_signature - -Verifier for ECDSA Secp256k1 signatures. -See ecdsa_secp256k1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256k1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N], -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256k1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256k1::verify_signature_slice - -Verifier for ECDSA Secp256k1 signatures where the message is a slice. - -```rust title="ecdsa_secp256k1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8], -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256k1.nr#L13-L20 - - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures. -See ecdsa_secp256r1::verify_signature_slice for a version that accepts slices directly. - -```rust title="ecdsa_secp256r1" showLineNumbers -pub fn verify_signature( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8; N], -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L2-L9 - - -example: - -```rust -fn main(hashed_message : [u8;32], pub_key_x : [u8;32], pub_key_y : [u8;32], signature : [u8;64]) { - let valid_signature = std::ecdsa_secp256r1::verify_signature(pub_key_x, pub_key_y, signature, hashed_message); - assert(valid_signature); -} -``` - - - -## ecdsa_secp256r1::verify_signature - -Verifier for ECDSA Secp256r1 signatures where the message is a slice. - -```rust title="ecdsa_secp256r1_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: [u8; 32], - public_key_y: [u8; 32], - signature: [u8; 64], - message_hash: [u8], -) -> bool -``` -> Source code: noir_stdlib/src/ecdsa_secp256r1.nr#L13-L20 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/eddsa.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/eddsa.mdx deleted file mode 100644 index b283de693c8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/eddsa.mdx +++ /dev/null @@ -1,37 +0,0 @@ ---- -title: EdDSA Verification -description: Learn about the cryptographic primitives regarding EdDSA -keywords: [cryptographic primitives, Noir project, eddsa, signatures] -sidebar_position: 5 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## eddsa::eddsa_poseidon_verify - -Verifier for EdDSA signatures - -```rust -fn eddsa_poseidon_verify(public_key_x : Field, public_key_y : Field, signature_s: Field, signature_r8_x: Field, signature_r8_y: Field, message: Field) -> bool -``` - -It is also possible to specify the hash algorithm used for the signature by using the `eddsa_verify` function by passing a type implementing the Hasher trait with the turbofish operator. -For instance, if you want to use Poseidon2 instead, you can do the following: -```rust -use std::hash::poseidon2::Poseidon2Hasher; - -eddsa_verify::(pub_key_a.x, pub_key_a.y, s_a, r8_a.x, r8_a.y, msg); -``` - - - -## eddsa::eddsa_to_pub - -Private to public key conversion. - -Returns `(pub_key_x, pub_key_y)` - -```rust -fn eddsa_to_pub(secret : Field) -> (Field, Field) -``` - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx deleted file mode 100644 index 482a36932b9..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx +++ /dev/null @@ -1,95 +0,0 @@ ---- -title: Scalar multiplication -description: See how you can perform scalar multiplication in Noir -keywords: [cryptographic primitives, Noir project, scalar multiplication] -sidebar_position: 1 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -The following functions perform operations over the embedded curve whose coordinates are defined by the configured noir field. -For the BN254 scalar field, this is BabyJubJub or Grumpkin. - -:::note -Suffixes `_low` and `_high` denote low and high limbs of a scalar. -::: - -## embedded_curve_ops::multi_scalar_mul - -Performs multi scalar multiplication over the embedded curve. -The function accepts arbitrary amount of point-scalar pairs on the input, it multiplies the individual pairs over -the curve and returns a sum of the resulting points. - -Points represented as x and y coordinates [x1, y1, x2, y2, ...], scalars as low and high limbs [low1, high1, low2, high2, ...]. - -```rust title="multi_scalar_mul" showLineNumbers -pub fn multi_scalar_mul( - points: [EmbeddedCurvePoint; N], - scalars: [EmbeddedCurveScalar; N], -) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L103-L108 - - -example - -```rust -fn main(point_x: Field, point_y: Field, scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::multi_scalar_mul([point_x, point_y], [scalar_low, scalar_high]); - println(point); -} -``` - -## embedded_curve_ops::fixed_base_scalar_mul - -Performs fixed base scalar multiplication over the embedded curve (multiplies input scalar with a generator point). -The function accepts a single scalar on the input represented as 2 fields. - -```rust title="fixed_base_scalar_mul" showLineNumbers -pub fn fixed_base_scalar_mul(scalar: EmbeddedCurveScalar) -> EmbeddedCurvePoint -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L120-L122 - - -example - -```rust -fn main(scalar_low: Field, scalar_high: Field) { - let point = std::embedded_curve_ops::fixed_base_scalar_mul(scalar_low, scalar_high); - println(point); -} -``` - -## embedded_curve_ops::embedded_curve_add - -Adds two points on the embedded curve. -This function takes two `EmbeddedCurvePoint` structures as parameters, representing points on the curve, and returns a new `EmbeddedCurvePoint` structure that represents their sum. - -### Parameters: -- `point1` (`EmbeddedCurvePoint`): The first point to add. -- `point2` (`EmbeddedCurvePoint`): The second point to add. - -### Returns: -- `EmbeddedCurvePoint`: The resulting point after the addition of `point1` and `point2`. - -```rust title="embedded_curve_add" showLineNumbers -pub fn embedded_curve_add( - point1: EmbeddedCurvePoint, - point2: EmbeddedCurvePoint, -) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/embedded_curve_ops.nr#L136-L141 - - -example - -```rust -fn main() { - let point1 = EmbeddedCurvePoint { x: 1, y: 2 }; - let point2 = EmbeddedCurvePoint { x: 3, y: 4 }; - let result = std::embedded_curve_ops::embedded_curve_add(point1, point2); - println!("Resulting Point: ({}, {})", result.x, result.y); -} -``` - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/hashes.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/hashes.mdx deleted file mode 100644 index 541a1971561..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/hashes.mdx +++ /dev/null @@ -1,227 +0,0 @@ ---- -title: Hash methods -description: - Learn about the cryptographic primitives ready to use for any Noir project, including sha256, - blake2s and pedersen -keywords: - [cryptographic primitives, Noir project, sha256, blake2s, pedersen, hash] -sidebar_position: 0 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## sha256 - -Given an array of bytes, returns the resulting sha256 hash. -Specify a message_size to hash only the first `message_size` bytes of the input. - -```rust title="sha256" showLineNumbers -pub fn sha256(input: [u8; N]) -> HASH -``` -> Source code: noir_stdlib/src/hash/sha256.nr#L47-L49 - - -example: -```rust title="sha256_var" showLineNumbers -let digest = std::hash::sha256_var([x as u8], 1); -``` -> Source code: test_programs/execution_success/sha256/src/main.nr#L15-L17 - - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::sha256::sha256_var(x, 4); -} -``` - - - - -## blake2s - -Given an array of bytes, returns an array with the Blake2 hash - -```rust title="blake2s" showLineNumbers -pub fn blake2s(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L18-L20 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake2s(x); -} -``` - - - -## blake3 - -Given an array of bytes, returns an array with the Blake3 hash - -```rust title="blake3" showLineNumbers -pub fn blake3(input: [u8; N]) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L24-L26 - - -example: - -```rust -fn main() { - let x = [163, 117, 178, 149]; // some random bytes - let hash = std::hash::blake3(x); -} -``` - - - -## pedersen_hash - -Given an array of Fields, returns the Pedersen hash. - -```rust title="pedersen_hash" showLineNumbers -pub fn pedersen_hash(input: [Field; N]) -> Field -``` -> Source code: noir_stdlib/src/hash/mod.nr#L49-L51 - - -example: - -```rust title="pedersen-hash" showLineNumbers -fn main(x: Field, y: Field, expected_hash: Field) { - let hash = std::hash::pedersen_hash([x, y]); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/pedersen_hash/src/main.nr#L1-L6 - - - - -## pedersen_commitment - -Given an array of Fields, returns the Pedersen commitment. - -```rust title="pedersen_commitment" showLineNumbers -pub fn pedersen_commitment(input: [Field; N]) -> EmbeddedCurvePoint { -``` -> Source code: noir_stdlib/src/hash/mod.nr#L29-L31 - - -example: - -```rust title="pedersen-commitment" showLineNumbers -fn main(x: Field, y: Field, expected_commitment: std::embedded_curve_ops::EmbeddedCurvePoint) { - let commitment = std::hash::pedersen_commitment([x, y]); - assert_eq(commitment.x, expected_commitment.x); - assert_eq(commitment.y, expected_commitment.y); -} -``` -> Source code: test_programs/execution_success/pedersen_commitment/src/main.nr#L1-L7 - - - - -## keccak256 - -Given an array of bytes (`u8`), returns the resulting keccak hash as an array of -32 bytes (`[u8; 32]`). Specify a message_size to hash only the first -`message_size` bytes of the input. - -```rust title="keccak256" showLineNumbers -pub fn keccak256(input: [u8; N], message_size: u32) -> [u8; 32] -``` -> Source code: noir_stdlib/src/hash/mod.nr#L116-L118 - - -example: - -```rust title="keccak256" showLineNumbers -fn main(x: Field, result: [u8; 32]) { - // We use the `as` keyword here to denote the fact that we want to take just the first byte from the x Field - // The padding is taken care of by the program - let digest = std::hash::keccak256([x as u8], 1); - assert(digest == result); - - //#1399: variable message size - let message_size = 4; - let hash_a = std::hash::keccak256([1, 2, 3, 4], message_size); - let hash_b = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size); - - assert(hash_a == hash_b); - - let message_size_big = 8; - let hash_c = std::hash::keccak256([1, 2, 3, 4, 0, 0, 0, 0], message_size_big); - - assert(hash_a != hash_c); -} -``` -> Source code: test_programs/execution_success/keccak256/src/main.nr#L1-L20 - - - - -## poseidon - -Given an array of Fields, returns a new Field with the Poseidon Hash. Mind that you need to specify -how many inputs are there to your Poseidon function. - -```rust -// example for hash_1, hash_2 accepts an array of length 2, etc -fn hash_1(input: [Field; 1]) -> Field -``` - -example: - -```rust title="poseidon" showLineNumbers -use std::hash::poseidon; - -fn main(x1: [Field; 2], y1: pub Field, x2: [Field; 4], y2: pub Field) { - let hash1 = poseidon::bn254::hash_2(x1); - assert(hash1 == y1); - - let hash2 = poseidon::bn254::hash_4(x2); - assert(hash2 == y2); -} -``` -> Source code: test_programs/execution_success/poseidon_bn254_hash/src/main.nr#L1-L11 - - -## poseidon 2 - -Given an array of Fields, returns a new Field with the Poseidon2 Hash. Contrary to the Poseidon -function, there is only one hash and you can specify a message_size to hash only the first -`message_size` bytes of the input, - -```rust -// example for hashing the first three elements of the input -Poseidon2::hash(input, 3); -``` - -example: - -```rust title="poseidon2" showLineNumbers -use std::hash::poseidon2; - -fn main(inputs: [Field; 4], expected_hash: Field) { - let hash = poseidon2::Poseidon2::hash(inputs, inputs.len()); - assert_eq(hash, expected_hash); -} -``` -> Source code: test_programs/execution_success/poseidon2/src/main.nr#L1-L8 - - -## hash_to_field - -```rust -fn hash_to_field(_input : [Field]) -> Field {} -``` - -Calculates the `blake2s` hash of the inputs and returns the hash modulo the field modulus to return -a value which can be represented as a `Field`. - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/index.md deleted file mode 100644 index 650f30165d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/index.md +++ /dev/null @@ -1,14 +0,0 @@ ---- -title: Cryptographic Primitives -description: - Learn about the cryptographic primitives ready to use for any Noir project -keywords: - [ - cryptographic primitives, - Noir project, - ] ---- - -The Noir team is progressively adding new cryptographic primitives to the standard library. Reach out for news or if you would be interested in adding more of these calculations in Noir. - -Some methods are available thanks to the Aztec backend, not being performed using Noir. When using other backends, these methods may or may not be supplied. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/schnorr.mdx b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/schnorr.mdx deleted file mode 100644 index 030452645c5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/cryptographic_primitives/schnorr.mdx +++ /dev/null @@ -1,64 +0,0 @@ ---- -title: Schnorr Signatures -description: Learn how you can verify Schnorr signatures using Noir -keywords: [cryptographic primitives, Noir project, schnorr, signatures] -sidebar_position: 2 ---- - -import BlackBoxInfo from '@site/src/components/Notes/_blackbox'; - -## schnorr::verify_signature - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin). -See schnorr::verify_signature_slice for a version that works directly on slices. - -```rust title="schnorr_verify" showLineNumbers -pub fn verify_signature( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8; N], -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L4-L11 - - -where `_signature` can be generated like so using the npm package -[@noir-lang/barretenberg](https://www.npmjs.com/package/@noir-lang/barretenberg) - -```js -const { BarretenbergWasm } = require('@noir-lang/barretenberg/dest/wasm'); -const { Schnorr } = require('@noir-lang/barretenberg/dest/crypto/schnorr'); - -... - -const barretenberg = await BarretenbergWasm.new(); -const schnorr = new Schnorr(barretenberg); -const pubKey = schnorr.computePublicKey(privateKey); -const message = ... -const signature = Array.from( - schnorr.constructSignature(hash, privateKey).toBuffer() -); - -... -``` - - - -## schnorr::verify_signature_slice - -Verifier for Schnorr signatures over the embedded curve (for BN254 it is Grumpkin) -where the message is a slice. - -```rust title="schnorr_verify_slice" showLineNumbers -pub fn verify_signature_slice( - public_key_x: Field, - public_key_y: Field, - signature: [u8; 64], - message: [u8], -) -> bool -``` -> Source code: noir_stdlib/src/schnorr.nr#L15-L22 - - - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/fmtstr.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/fmtstr.md deleted file mode 100644 index 19809d60261..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/fmtstr.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: fmtstr ---- - -`fmtstr` is the type resulting from using format string (`f"..."`). - -## Methods - -### quoted_contents - -```rust title="quoted_contents" showLineNumbers -pub comptime fn quoted_contents(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/format_string.nr#L3-L5 - - -Returns the format string contents (that is, without the leading and trailing double quotes) as a `Quoted` value. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/is_unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/is_unconstrained.md deleted file mode 100644 index 51bb1bda8f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/is_unconstrained.md +++ /dev/null @@ -1,69 +0,0 @@ ---- -title: Is Unconstrained Function -description: - The is_unconstrained function returns wether the context at that point of the program is unconstrained or not. -keywords: - [ - unconstrained - ] ---- - -It's very common for functions in circuits to take unconstrained hints of an expensive computation and then verify it. This is done by running the hint in an unconstrained context and then verifying the result in a constrained context. - -When a function is marked as unconstrained, any subsequent functions that it calls will also be run in an unconstrained context. However, if we are implementing a library function, other users might call it within an unconstrained context or a constrained one. Generally, in an unconstrained context we prefer just computing the result instead of taking a hint of it and verifying it, since that'd mean doing the same computation twice: - -```rust - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - my_expensive_computation_hint(); - // verify my_expensive_computation: If external_interface is called from unconstrained, this is redundant - ... -} - -``` - -In order to improve the performance in an unconstrained context you can use the function at `std::runtime::is_unconstrained() -> bool`: - - -```rust -use dep::std::runtime::is_unconstrained; - -fn my_expensive_computation(){ - ... -} - -unconstrained fn my_expensive_computation_hint(){ - my_expensive_computation() -} - -pub fn external_interface(){ - if is_unconstrained() { - my_expensive_computation(); - } else { - my_expensive_computation_hint(); - // verify my_expensive_computation - ... - } -} - -``` - -The is_unconstrained result is resolved at compile time, so in unconstrained contexts the compiler removes the else branch, and in constrained contexts the compiler removes the if branch, reducing the amount of compute necessary to run external_interface. - -Note that using `is_unconstrained` in a `comptime` context will also return `true`: - -``` -fn main() { - comptime { - assert(is_unconstrained()); - } -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/logging.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/logging.md deleted file mode 100644 index db75ef9f86f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/logging.md +++ /dev/null @@ -1,78 +0,0 @@ ---- -title: Logging -description: - Learn how to use the println statement for debugging in Noir with this tutorial. Understand the - basics of logging in Noir and how to implement it in your code. -keywords: - [ - noir logging, - println statement, - print statement, - debugging in noir, - noir std library, - logging tutorial, - basic logging in noir, - noir logging implementation, - noir debugging techniques, - rust, - ] ---- - -The standard library provides two familiar statements you can use: `println` and `print`. Despite being a limited implementation of rust's `println!` and `print!` macros, these constructs can be useful for debugging. - -You can print the output of both statements in your Noir code by using the `nargo execute` command or the `--show-output` flag when using `nargo test` (provided there are print statements in your tests). - -It is recommended to use `nargo execute` if you want to debug failing constraints with `println` or `print` statements. This is due to every input in a test being a constant rather than a witness, so we issue an error during compilation while we only print during execution (which comes after compilation). Neither `println`, nor `print` are callable for failed constraints caught at compile time. - -Both `print` and `println` are generic functions which can work on integers, fields, strings, and even structs or expressions. Note however, that slices are currently unsupported. For example: - -```rust -struct Person { - age: Field, - height: Field, -} - -fn main(age: Field, height: Field) { - let person = Person { - age: age, - height: height, - }; - println(person); - println(age + height); - println("Hello world!"); -} -``` - -You can print different types in the same statement (including strings) with a type called `fmtstr`. It can be specified in the same way as a normal string, just prepended with an "f" character: - -```rust - let fmt_str = f"i: {i}, j: {j}"; - println(fmt_str); - - let s = myStruct { y: x, x: y }; - println(s); - - println(f"i: {i}, s: {s}"); - - println(x); - println([x, y]); - - let foo = fooStruct { my_struct: s, foo: 15 }; - println(f"s: {s}, foo: {foo}"); - - println(15); // prints 0x0f, implicit Field - println(-1 as u8); // prints 255 - println(-1 as i8); // prints -1 -``` - -Examples shown above are interchangeable between the two `print` statements: - -```rust -let person = Person { age : age, height : height }; - -println(person); -print(person); - -println("Hello world!"); // Prints with a newline at the end of the input -print("Hello world!"); // Prints the input and keeps cursor on the same line -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/mem.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/mem.md deleted file mode 100644 index 95d36ac2a72..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/mem.md +++ /dev/null @@ -1,52 +0,0 @@ ---- -title: Memory Module -description: - This module contains functions which manipulate memory in a low-level way -keywords: - [ - mem, memory, zeroed, transmute, checked_transmute - ] ---- - -# `std::mem::zeroed` - -```rust -fn zeroed() -> T -``` - -Returns a zeroed value of any type. -This function is generally unsafe to use as the zeroed bit pattern is not guaranteed to be valid for all types. -It can however, be useful in cases when the value is guaranteed not to be used such as in a BoundedVec library implementing a growable vector, up to a certain length, backed by an array. -The array can be initialized with zeroed values which are guaranteed to be inaccessible until the vector is pushed to. -Similarly, enumerations in noir can be implemented using this method by providing zeroed values for the unused variants. - -This function currently supports the following types: - -- Field -- Bool -- Uint -- Array -- Slice -- String -- Tuple -- Functions - -Using it on other types could result in unexpected behavior. - -# `std::mem::checked_transmute` - -```rust -fn checked_transmute(value: T) -> U -``` - -Transmutes a value of one type into the same value but with a new type `U`. - -This function is safe to use since both types are asserted to be equal later during compilation after the concrete values for generic types become known. -This function is useful for cases where the compiler may fails a type check that is expected to pass where -a user knows the two types to be equal. For example, when using arithmetic generics there are cases the compiler -does not see as equal, such as `[Field; N*(A + B)]` and `[Field; N*A + N*B]`, which users may know to be equal. -In these cases, `checked_transmute` can be used to cast the value to the desired type while also preserving safety -by checking this equality once `N`, `A`, `B` are fully resolved. - -Note that since this safety check is performed after type checking rather than during, no error is issued if the function -containing `checked_transmute` is never called. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/merkle_trees.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/merkle_trees.md deleted file mode 100644 index 6a9ebf72ada..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/merkle_trees.md +++ /dev/null @@ -1,58 +0,0 @@ ---- -title: Merkle Trees -description: Learn about Merkle Trees in Noir with this tutorial. Explore the basics of computing a merkle root using a proof, with examples. -keywords: - [ - Merkle trees in Noir, - Noir programming language, - check membership, - computing root from leaf, - Noir Merkle tree implementation, - Merkle tree tutorial, - Merkle tree code examples, - Noir libraries, - pedersen hash., - ] ---- - -## compute_merkle_root - -Returns the root of the tree from the provided leaf and its hash path, using a [Pedersen hash](./cryptographic_primitives/hashes.mdx#pedersen_hash). - -```rust -fn compute_merkle_root(leaf : Field, index : Field, hash_path: [Field]) -> Field -``` - -example: - -```rust -/** - // these values are for this example only - index = "0" - priv_key = "0x000000000000000000000000000000000000000000000000000000616c696365" - secret = "0x1929ea3ab8d9106a899386883d9428f8256cfedb3c4f6b66bf4aa4d28a79988f" - note_hash_path = [ - "0x1e61bdae0f027b1b2159e1f9d3f8d00fa668a952dddd822fda80dc745d6f65cc", - "0x0e4223f3925f98934393c74975142bd73079ab0621f4ee133cee050a3c194f1a", - "0x2fd7bb412155bf8693a3bd2a3e7581a679c95c68a052f835dddca85fa1569a40" - ] - */ -fn main(index: Field, priv_key: Field, secret: Field, note_hash_path: [Field; 3]) { - - let pubkey = std::scalar_mul::fixed_base_embedded_curve(priv_key); - let pubkey_x = pubkey[0]; - let pubkey_y = pubkey[1]; - let note_commitment = std::hash::pedersen(&[pubkey_x, pubkey_y, secret]); - - let root = std::merkle::compute_merkle_root(note_commitment[0], index, note_hash_path.as_slice()); - println(root); -} -``` - -To check merkle tree membership: - -1. Include a merkle root as a program input. -2. Compute the merkle root of a given leaf, index and hash path. -3. Assert the merkle roots are equal. - -For more info about merkle trees, see the Wikipedia [page](https://en.wikipedia.org/wiki/Merkle_tree). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/ctstring.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/ctstring.md deleted file mode 100644 index b76f873ca03..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/ctstring.md +++ /dev/null @@ -1,100 +0,0 @@ ---- -title: CtString ---- - -`std::meta::ctstring` contains methods on the built-in `CtString` type which is -a compile-time, dynamically-sized string type. Compared to `str` and `fmtstr`, -`CtString` is useful because its size does not need to be specified in its type. This -can be used for formatting items at compile-time or general string handling in `comptime` -code. - -Since `fmtstr`s can be converted into `CtString`s, you can make use of their formatting -abilities in CtStrings by formatting in `fmtstr`s then converting the result to a CtString -afterward. - -## Traits - -### AsCtString - -```rust title="as-ctstring" showLineNumbers -pub trait AsCtString { - comptime fn as_ctstring(self) -> CtString; -} -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L43-L47 - - -Converts an object into a compile-time string. - -Implementations: - -```rust -impl AsCtString for str { ... } -impl AsCtString for fmtstr { ... } -``` - -## Methods - -### new - -```rust title="new" showLineNumbers -pub comptime fn new() -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L4-L6 - - -Creates an empty `CtString`. - -### append_str - -```rust title="append_str" showLineNumbers -pub comptime fn append_str(self, s: str) -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L11-L13 - - -Returns a new CtString with the given str appended onto the end. - -### append_fmtstr - -```rust title="append_fmtstr" showLineNumbers -pub comptime fn append_fmtstr(self, s: fmtstr) -> Self { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L17-L19 - - -Returns a new CtString with the given fmtstr appended onto the end. - -### as_quoted_str - -```rust title="as_quoted_str" showLineNumbers -pub comptime fn as_quoted_str(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L26-L28 - - -Returns a quoted string literal from this string's contents. - -There is no direct conversion from a `CtString` to a `str` since -the size would not be known. To get around this, this function can -be used in combination with macro insertion (`!`) to insert this string -literal at this function's call site. - -Example: - -```rust title="as_quoted_str_example" showLineNumbers -let my_ctstring = "foo bar".as_ctstring(); - let my_str = my_ctstring.as_quoted_str!(); - - assert_eq(crate::meta::type_of(my_str), quote { str<7> }.as_type()); -``` -> Source code: noir_stdlib/src/meta/ctstring.nr#L92-L97 - - -## Trait Implementations - -```rust -impl Eq for CtString -impl Hash for CtString -impl Append for CtString -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/expr.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/expr.md deleted file mode 100644 index b6d395c6700..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/expr.md +++ /dev/null @@ -1,380 +0,0 @@ ---- -title: Expr ---- - -`std::meta::expr` contains methods on the built-in `Expr` type for quoted, syntactically valid expressions. - -## Methods - -### as_array - -```rust title="as_array" showLineNumbers -pub comptime fn as_array(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L10-L12 - - -If this expression is an array, this returns a slice of each element in the array. - -### as_assert - -```rust title="as_assert" showLineNumbers -pub comptime fn as_assert(self) -> Option<(Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L16-L18 - - -If this expression is an assert, this returns the assert expression and the optional message. - -### as_assert_eq - -```rust title="as_assert_eq" showLineNumbers -pub comptime fn as_assert_eq(self) -> Option<(Expr, Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L23-L25 - - -If this expression is an assert_eq, this returns the left-hand-side and right-hand-side -expressions, together with the optional message. - -### as_assign - -```rust title="as_assign" showLineNumbers -pub comptime fn as_assign(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L30-L32 - - -If this expression is an assignment, this returns a tuple with the left hand side -and right hand side in order. - -### as_binary_op - -```rust title="as_binary_op" showLineNumbers -pub comptime fn as_binary_op(self) -> Option<(Expr, BinaryOp, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L37-L39 - - -If this expression is a binary operator operation ` `, -return the left-hand side, operator, and the right-hand side of the operation. - -### as_block - -```rust title="as_block" showLineNumbers -pub comptime fn as_block(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L44-L46 - - -If this expression is a block `{ stmt1; stmt2; ...; stmtN }`, return -a slice containing each statement. - -### as_bool - -```rust title="as_bool" showLineNumbers -pub comptime fn as_bool(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L50-L52 - - -If this expression is a boolean literal, return that literal. - -### as_cast - -```rust title="as_cast" showLineNumbers -#[builtin(expr_as_cast)] - pub comptime fn as_cast(self) -> Option<(Expr, UnresolvedType)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L56-L59 - - -If this expression is a cast expression (`expr as type`), returns the casted -expression and the type to cast to. - -### as_comptime - -```rust title="as_comptime" showLineNumbers -pub comptime fn as_comptime(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L64-L66 - - -If this expression is a `comptime { stmt1; stmt2; ...; stmtN }` block, -return each statement in the block. - -### as_constructor - -```rust title="as_constructor" showLineNumbers -pub comptime fn as_constructor(self) -> Option<(UnresolvedType, [(Quoted, Expr)])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L71-L73 - - -If this expression is a constructor `Type { field1: expr1, ..., fieldN: exprN }`, -return the type and the fields. - -### as_for - -```rust title="as_for" showLineNumbers -pub comptime fn as_for(self) -> Option<(Quoted, Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L78-L80 - - -If this expression is a for statement over a single expression, return the identifier, -the expression and the for loop body. - -### as_for_range - -```rust title="as_for" showLineNumbers -pub comptime fn as_for(self) -> Option<(Quoted, Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L78-L80 - - -If this expression is a for statement over a range, return the identifier, -the range start, the range end and the for loop body. - -### as_function_call - -```rust title="as_function_call" showLineNumbers -pub comptime fn as_function_call(self) -> Option<(Expr, [Expr])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L92-L94 - - -If this expression is a function call `foo(arg1, ..., argN)`, return -the function and a slice of each argument. - -### as_if - -```rust title="as_if" showLineNumbers -pub comptime fn as_if(self) -> Option<(Expr, Expr, Option)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L100-L102 - - -If this expression is an `if condition { then_branch } else { else_branch }`, -return the condition, then branch, and else branch. If there is no else branch, -`None` is returned for that branch instead. - -### as_index - -```rust title="as_index" showLineNumbers -pub comptime fn as_index(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L107-L109 - - -If this expression is an index into an array `array[index]`, return the -array and the index. - -### as_integer - -```rust title="as_integer" showLineNumbers -pub comptime fn as_integer(self) -> Option<(Field, bool)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L114-L116 - - -If this expression is an integer literal, return the integer as a field -as well as whether the integer is negative (true) or not (false). - -### as_lambda - -```rust title="as_lambda" showLineNumbers -pub comptime fn as_lambda( - self, - ) -> Option<([(Expr, Option)], Option, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L120-L124 - - -If this expression is a lambda, returns the parameters, return type and body. - -### as_let - -```rust title="as_let" showLineNumbers -pub comptime fn as_let(self) -> Option<(Expr, Option, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L129-L131 - - -If this expression is a let statement, returns the let pattern as an `Expr`, -the optional type annotation, and the assigned expression. - -### as_member_access - -```rust title="as_member_access" showLineNumbers -pub comptime fn as_member_access(self) -> Option<(Expr, Quoted)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L136-L138 - - -If this expression is a member access `foo.bar`, return the struct/tuple -expression and the field. The field will be represented as a quoted value. - -### as_method_call - -```rust title="as_method_call" showLineNumbers -pub comptime fn as_method_call(self) -> Option<(Expr, Quoted, [UnresolvedType], [Expr])> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L143-L145 - - -If this expression is a method call `foo.bar::(arg1, ..., argN)`, return -the receiver, method name, a slice of each generic argument, and a slice of each argument. - -### as_repeated_element_array - -```rust title="as_repeated_element_array" showLineNumbers -pub comptime fn as_repeated_element_array(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L150-L152 - - -If this expression is a repeated element array `[elem; length]`, return -the repeated element and the length expressions. - -### as_repeated_element_slice - -```rust title="as_repeated_element_slice" showLineNumbers -pub comptime fn as_repeated_element_slice(self) -> Option<(Expr, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L157-L159 - - -If this expression is a repeated element slice `[elem; length]`, return -the repeated element and the length expressions. - -### as_slice - -```rust title="as_slice" showLineNumbers -pub comptime fn as_slice(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L164-L166 - - -If this expression is a slice literal `&[elem1, ..., elemN]`, -return each element of the slice. - -### as_tuple - -```rust title="as_tuple" showLineNumbers -pub comptime fn as_tuple(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L171-L173 - - -If this expression is a tuple `(field1, ..., fieldN)`, -return each element of the tuple. - -### as_unary_op - -```rust title="as_unary_op" showLineNumbers -pub comptime fn as_unary_op(self) -> Option<(UnaryOp, Expr)> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L178-L180 - - -If this expression is a unary operation ` `, -return the unary operator as well as the right-hand side expression. - -### as_unsafe - -```rust title="as_unsafe" showLineNumbers -pub comptime fn as_unsafe(self) -> Option<[Expr]> {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L185-L187 - - -If this expression is an `unsafe { stmt1; ...; stmtN }` block, -return each statement inside in a slice. - -### has_semicolon - -```rust title="has_semicolon" showLineNumbers -pub comptime fn has_semicolon(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L206-L208 - - -`true` if this expression is trailed by a semicolon. E.g. - -``` -comptime { - let expr1 = quote { 1 + 2 }.as_expr().unwrap(); - let expr2 = quote { 1 + 2; }.as_expr().unwrap(); - - assert(expr1.as_binary_op().is_some()); - assert(expr2.as_binary_op().is_some()); - - assert(!expr1.has_semicolon()); - assert(expr2.has_semicolon()); -} -``` - -### is_break - -```rust title="is_break" showLineNumbers -pub comptime fn is_break(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L212-L214 - - -`true` if this expression is `break`. - -### is_continue - -```rust title="is_continue" showLineNumbers -pub comptime fn is_continue(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L218-L220 - - -`true` if this expression is `continue`. - -### modify - -```rust title="modify" showLineNumbers -pub comptime fn modify(self, f: fn[Env](Expr) -> Option) -> Expr { -``` -> Source code: noir_stdlib/src/meta/expr.nr#L229-L231 - - -Applies a mapping function to this expression and to all of its sub-expressions. -`f` will be applied to each sub-expression first, then applied to the expression itself. - -This happens recursively for every expression within `self`. - -For example, calling `modify` on `(&[1], &[2, 3])` with an `f` that returns `Option::some` -for expressions that are integers, doubling them, would return `(&[2], &[4, 6])`. - -### quoted - -```rust title="quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/expr.nr#L266-L268 - - -Returns this expression as a `Quoted` value. It's the same as `quote { $self }`. - -### resolve - -```rust title="resolve" showLineNumbers -pub comptime fn resolve(self, in_function: Option) -> TypedExpr {} -``` -> Source code: noir_stdlib/src/meta/expr.nr#L282-L284 - - -Resolves and type-checks this expression and returns the result as a `TypedExpr`. - -The `in_function` argument specifies where the expression is resolved: -- If it's `none`, the expression is resolved in the function where `resolve` was called -- If it's `some`, the expression is resolved in the given function - -If any names used by this expression are not in scope or if there are any type errors, -this will give compiler errors as if the expression was written directly into -the current `comptime` function. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/function_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/function_def.md deleted file mode 100644 index b7f2ebdb889..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/function_def.md +++ /dev/null @@ -1,166 +0,0 @@ ---- -title: FunctionDefinition ---- - -`std::meta::function_def` contains methods on the built-in `FunctionDefinition` type representing -a function definition in the source program. - -## Methods - -### add_attribute - -```rust title="add_attribute" showLineNumbers -pub comptime fn add_attribute(self, attribute: str) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L3-L5 - - -Adds an attribute to the function. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### body - -```rust title="body" showLineNumbers -pub comptime fn body(self) -> Expr {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L8-L10 - - -Returns the body of the function as an expression. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -pub comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L13-L15 - - -Returns true if this function has a custom attribute with the given name. - -### is_unconstrained - -```rust title="is_unconstrained" showLineNumbers -pub comptime fn is_unconstrained(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L18-L20 - - -Returns true if this function is unconstrained. - -### module - -```rust title="module" showLineNumbers -pub comptime fn module(self) -> Module {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L23-L25 - - -Returns the module where the function is defined. - -### name - -```rust title="name" showLineNumbers -pub comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L28-L30 - - -Returns the name of the function. - -### parameters - -```rust title="parameters" showLineNumbers -pub comptime fn parameters(self) -> [(Quoted, Type)] {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L33-L35 - - -Returns each parameter of the function as a tuple of (parameter pattern, parameter type). - -### return_type - -```rust title="return_type" showLineNumbers -pub comptime fn return_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L38-L40 - - -The return type of the function. - -### set_body - -```rust title="set_body" showLineNumbers -pub comptime fn set_body(self, body: Expr) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L43-L45 - - -Mutate the function body to a new expression. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_parameters - -```rust title="set_parameters" showLineNumbers -pub comptime fn set_parameters(self, parameters: [(Quoted, Type)]) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L48-L50 - - -Mutates the function's parameters to a new set of parameters. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -Expects a slice of (parameter pattern, parameter type) for each parameter. Requires -each parameter pattern to be a syntactically valid parameter. - -### set_return_type - -```rust title="set_return_type" showLineNumbers -pub comptime fn set_return_type(self, return_type: Type) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L53-L55 - - -Mutates the function's return type to a new type. This is only valid -on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_return_public - -```rust title="set_return_public" showLineNumbers -pub comptime fn set_return_public(self, public: bool) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L58-L60 - - -Mutates the function's return visibility to public (if `true` is given) or private (if `false` is given). -This is only valid on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -### set_unconstrained - -```rust title="set_unconstrained" showLineNumbers -pub comptime fn set_unconstrained(self, value: bool) {} -``` -> Source code: noir_stdlib/src/meta/function_def.nr#L66-L68 - - -Mutates the function to be unconstrained (if `true` is given) or not (if `false` is given). -This is only valid on functions in the current crate which have not yet been resolved. -This means any functions called at compile-time are invalid targets for this method. - -## Trait Implementations - -```rust -impl Eq for FunctionDefinition -impl Hash for FunctionDefinition -``` - -Note that each function is assigned a unique ID internally and this is what is used for -equality and hashing. So even functions with identical signatures and bodies may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/index.md deleted file mode 100644 index 14544c07442..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/index.md +++ /dev/null @@ -1,224 +0,0 @@ ---- -title: Metaprogramming -description: Noir's Metaprogramming API -keywords: [metaprogramming, comptime, macros, macro, quote, unquote] ---- - -`std::meta` is the entry point for Noir's metaprogramming API. This consists of `comptime` functions -and types used for inspecting and modifying Noir programs. - -## Functions - -### type_of - -```rust title="type_of" showLineNumbers -pub comptime fn type_of(x: T) -> Type {} -``` -> Source code: noir_stdlib/src/meta/mod.nr#L27-L29 - - -Returns the type of a variable at compile-time. - -Example: -```rust -comptime { - let x: i32 = 1; - let x_type: Type = std::meta::type_of(x); - - assert_eq(x_type, quote { i32 }.as_type()); -} -``` - -### unquote - -```rust title="unquote" showLineNumbers -pub comptime fn unquote(code: Quoted) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L19-L21 - - -Unquotes the passed-in token stream where this function was called. - -Example: -```rust -comptime { - let code = quote { 1 + 2 }; - - // let x = 1 + 2; - let x = unquote!(code); -} -``` - -### derive - -```rust title="derive" showLineNumbers -#[varargs] -pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L48-L51 - - -Attribute placed on struct definitions. - -Creates a trait impl for each trait passed in as an argument. -To do this, the trait must have a derive handler registered -with `derive_via` beforehand. The traits in the stdlib that -can be derived this way are `Eq`, `Ord`, `Default`, and `Hash`. - -Example: -```rust -#[derive(Eq, Default)] -struct Foo { - x: i32, - y: T, -} - -fn main() { - let foo1 = Foo::default(); - let foo2 = Foo { x: 0, y: &[0] }; - assert_eq(foo1, foo2); -} -``` - -### derive_via - -```rust title="derive_via_signature" showLineNumbers -pub comptime fn derive_via(t: TraitDefinition, f: DeriveFunction) { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L68-L70 - - -Attribute placed on trait definitions. - -Registers a function to create impls for the given trait -when the trait is used in a `derive` call. Users may use -this to register their own functions to enable their traits -to be derived by `derive`. - -Because this function requires a function as an argument which -should produce a trait impl for any given struct, users may find -it helpful to use a function like `std::meta::make_trait_impl` to -help creating these impls. - -Example: -```rust -#[derive_via(derive_do_nothing)] -trait DoNothing { - fn do_nothing(self); -} - -comptime fn derive_do_nothing(s: StructDefinition) -> Quoted { - let typ = s.as_type(); - quote { - impl DoNothing for $typ { - fn do_nothing(self) { - println("Nothing"); - } - } - } -} -``` - -As another example, `derive_eq` in the stdlib is used to derive the `Eq` -trait for any struct. It makes use of `make_trait_impl` to do this: - -```rust title="derive_eq" showLineNumbers -comptime fn derive_eq(s: StructDefinition) -> Quoted { - let signature = quote { fn eq(_self: Self, _other: Self) -> bool }; - let for_each_field = |name| quote { (_self.$name == _other.$name) }; - let body = |fields| { - if s.fields().len() == 0 { - quote { true } - } else { - fields - } - }; - crate::meta::make_trait_impl( - s, - quote { Eq }, - signature, - for_each_field, - quote { & }, - body, - ) -} -``` -> Source code: noir_stdlib/src/cmp.nr#L10-L30 - - -### make_trait_impl - -```rust title="make_trait_impl" showLineNumbers -pub comptime fn make_trait_impl( - s: StructDefinition, - trait_name: Quoted, - function_signature: Quoted, - for_each_field: fn[Env1](Quoted) -> Quoted, - join_fields_with: Quoted, - body: fn[Env2](Quoted) -> Quoted, -) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/mod.nr#L87-L96 - - -A helper function to more easily create trait impls while deriving traits. - -Note that this function only works for traits which: -1. Have only one method -2. Have no generics on the trait itself. - - E.g. Using this on a trait such as `trait Foo { ... }` will result in the - generated impl incorrectly missing the `T` generic. - -If your trait fits these criteria then `make_trait_impl` is likely the easiest -way to write your derive handler. The arguments are as follows: - -- `s`: The struct to make the impl for -- `trait_name`: The name of the trait to derive. E.g. `quote { Eq }`. -- `function_signature`: The signature of the trait method to derive. E.g. `fn eq(self, other: Self) -> bool`. -- `for_each_field`: An operation to be performed on each field. E.g. `|name| quote { (self.$name == other.$name) }`. -- `join_fields_with`: A separator to join each result of `for_each_field` with. - E.g. `quote { & }`. You can also use an empty `quote {}` for no separator. -- `body`: The result of the field operations are passed into this function for any final processing. - This is the place to insert any setup/teardown code the trait requires. If the trait doesn't require - any such code, you can return the body as-is: `|body| body`. - -Example deriving `Hash`: - -```rust title="derive_hash" showLineNumbers -comptime fn derive_hash(s: StructDefinition) -> Quoted { - let name = quote { Hash }; - let signature = quote { fn hash(_self: Self, _state: &mut H) where H: std::hash::Hasher }; - let for_each_field = |name| quote { _self.$name.hash(_state); }; - crate::meta::make_trait_impl( - s, - name, - signature, - for_each_field, - quote {}, - |fields| fields, - ) -} -``` -> Source code: noir_stdlib/src/hash/mod.nr#L137-L151 - - -Example deriving `Ord`: - -```rust title="derive_ord" showLineNumbers -comptime fn derive_ord(s: StructDefinition) -> Quoted { - let signature = quote { fn cmp(_self: Self, _other: Self) -> std::cmp::Ordering }; - let for_each_field = |name| quote { - if result == std::cmp::Ordering::equal() { - result = _self.$name.cmp(_other.$name); - } - }; - let body = |fields| quote { - let mut result = std::cmp::Ordering::equal(); - $fields - result - }; - crate::meta::make_trait_impl(s, quote { Ord }, signature, for_each_field, quote {}, body) -} -``` -> Source code: noir_stdlib/src/cmp.nr#L216-L231 - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/module.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/module.md deleted file mode 100644 index f47231972b7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/module.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: Module ---- - -`std::meta::module` contains methods on the built-in `Module` type which represents a module in the source program. -Note that this type represents a module generally, it isn't limited to only `mod my_submodule { ... }` -declarations in the source program. - -## Methods - -### add_item - -```rust title="add_item" showLineNumbers -pub comptime fn add_item(self, item: Quoted) {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L3-L5 - - -Adds a top-level item (a function, a struct, a global, etc.) to the module. -Adding multiple items in one go is also valid if the `Quoted` value has multiple items in it. -Note that the items are type-checked as if they are inside the module they are being added to. - -### functions - -```rust title="functions" showLineNumbers -pub comptime fn functions(self) -> [FunctionDefinition] {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L18-L20 - - -Returns each function defined in the module. - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -pub comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L8-L10 - - -Returns true if this module has a custom attribute with the given name. - -### is_contract - -```rust title="is_contract" showLineNumbers -pub comptime fn is_contract(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L13-L15 - - -`true` if this module is a contract module (was declared via `contract foo { ... }`). - -### name - -```rust title="name" showLineNumbers -pub comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L28-L30 - - -Returns the name of the module. - -### structs - -```rust title="structs" showLineNumbers -pub comptime fn structs(self) -> [StructDefinition] {} -``` -> Source code: noir_stdlib/src/meta/module.nr#L23-L25 - - -Returns each struct defined in the module. - -## Trait Implementations - -```rust -impl Eq for Module -impl Hash for Module -``` - -Note that each module is assigned a unique ID internally and this is what is used for -equality and hashing. So even modules with identical names and contents may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/op.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/op.md deleted file mode 100644 index 03ea49ad8ec..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/op.md +++ /dev/null @@ -1,244 +0,0 @@ ---- -title: UnaryOp and BinaryOp ---- - -`std::meta::op` contains the `UnaryOp` and `BinaryOp` types as well as methods on them. -These types are used to represent a unary or binary operator respectively in Noir source code. - -## Types - -### UnaryOp - -Represents a unary operator. One of `-`, `!`, `&mut`, or `*`. - -### Methods - -#### is_minus - -```rust title="is_minus" showLineNumbers -pub fn is_minus(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L24-L26 - - -Returns `true` if this operator is `-`. - -#### is_not - -```rust title="is_not" showLineNumbers -pub fn is_not(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L30-L32 - - -`true` if this operator is `!` - -#### is_mutable_reference - -```rust title="is_mutable_reference" showLineNumbers -pub fn is_mutable_reference(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L36-L38 - - -`true` if this operator is `&mut` - -#### is_dereference - -```rust title="is_dereference" showLineNumbers -pub fn is_dereference(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L42-L44 - - -`true` if this operator is `*` - -#### quoted - -```rust title="unary_quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/op.nr#L48-L50 - - -Returns this operator as a `Quoted` value. - -### Trait Implementations - -```rust -impl Eq for UnaryOp -impl Hash for UnaryOp -``` - -### BinaryOp - -Represents a binary operator. One of `+`, `-`, `*`, `/`, `%`, `==`, `!=`, `<`, `<=`, `>`, `>=`, `&`, `|`, `^`, `>>`, or `<<`. - -### Methods - -#### is_add - -```rust title="is_add" showLineNumbers -pub fn is_add(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L86-L88 - - -`true` if this operator is `+` - -#### is_subtract - -```rust title="is_subtract" showLineNumbers -pub fn is_subtract(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L92-L94 - - -`true` if this operator is `-` - -#### is_multiply - -```rust title="is_multiply" showLineNumbers -pub fn is_multiply(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L98-L100 - - -`true` if this operator is `*` - -#### is_divide - -```rust title="is_divide" showLineNumbers -pub fn is_divide(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L104-L106 - - -`true` if this operator is `/` - -#### is_modulo - -```rust title="is_modulo" showLineNumbers -pub fn is_modulo(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L176-L178 - - -`true` if this operator is `%` - -#### is_equal - -```rust title="is_equal" showLineNumbers -pub fn is_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L110-L112 - - -`true` if this operator is `==` - -#### is_not_equal - -```rust title="is_not_equal" showLineNumbers -pub fn is_not_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L116-L118 - - -`true` if this operator is `!=` - -#### is_less_than - -```rust title="is_less_than" showLineNumbers -pub fn is_less_than(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L122-L124 - - -`true` if this operator is `<` - -#### is_less_than_or_equal - -```rust title="is_less_than_or_equal" showLineNumbers -pub fn is_less_than_or_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L128-L130 - - -`true` if this operator is `<=` - -#### is_greater_than - -```rust title="is_greater_than" showLineNumbers -pub fn is_greater_than(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L134-L136 - - -`true` if this operator is `>` - -#### is_greater_than_or_equal - -```rust title="is_greater_than_or_equal" showLineNumbers -pub fn is_greater_than_or_equal(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L140-L142 - - -`true` if this operator is `>=` - -#### is_and - -```rust title="is_and" showLineNumbers -pub fn is_and(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L146-L148 - - -`true` if this operator is `&` - -#### is_or - -```rust title="is_or" showLineNumbers -pub fn is_or(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L152-L154 - - -`true` if this operator is `|` - -#### is_shift_right - -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L164-L166 - - -`true` if this operator is `>>` - -#### is_shift_left - -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { -``` -> Source code: noir_stdlib/src/meta/op.nr#L164-L166 - - -`true` if this operator is `<<` - -#### quoted - -```rust title="binary_quoted" showLineNumbers -pub comptime fn quoted(self) -> Quoted { -``` -> Source code: noir_stdlib/src/meta/op.nr#L182-L184 - - -Returns this operator as a `Quoted` value. - -### Trait Implementations - -```rust -impl Eq for BinaryOp -impl Hash for BinaryOp -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/quoted.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/quoted.md deleted file mode 100644 index d7acf23bc07..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/quoted.md +++ /dev/null @@ -1,141 +0,0 @@ ---- -title: Quoted ---- - -`std::meta::quoted` contains methods on the built-in `Quoted` type which represents -quoted token streams and is the result of the `quote { ... }` expression. - -## Methods - -### as_expr - -```rust title="as_expr" showLineNumbers -pub comptime fn as_expr(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L6-L8 - - -Parses the quoted token stream as an expression. Returns `Option::none()` if -the expression failed to parse. - -Example: - -```rust title="as_expr_example" showLineNumbers -#[test] - fn test_expr_as_function_call() { - comptime - { - let expr = quote { foo(42) }.as_expr().unwrap(); - let (_function, args) = expr.as_function_call().unwrap(); - assert_eq(args.len(), 1); - assert_eq(args[0].as_integer().unwrap(), (42, false)); - } - } -``` -> Source code: test_programs/noir_test_success/comptime_expr/src/main.nr#L360-L371 - - -### as_module - -```rust title="as_module" showLineNumbers -pub comptime fn as_module(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L11-L13 - - -Interprets this token stream as a module path leading to the name of a module. -Returns `Option::none()` if the module isn't found or this token stream cannot be parsed as a path. - -Example: - -```rust title="as_module_example" showLineNumbers -mod baz { - pub mod qux {} -} - -#[test] -fn as_module_test() { - comptime { - let my_mod = quote { baz::qux }.as_module().unwrap(); - assert_eq(my_mod.name(), quote { qux }); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_module/src/main.nr#L115-L127 - - -### as_trait_constraint - -```rust title="as_trait_constraint" showLineNumbers -pub comptime fn as_trait_constraint(self) -> TraitConstraint {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L16-L18 - - -Interprets this token stream as a trait constraint (without an object type). -Note that this function panics instead of returning `Option::none()` if the token -stream does not parse and resolve to a valid trait constraint. - -Example: - -```rust title="implements_example" showLineNumbers -pub fn function_with_where(_x: T) -where - T: SomeTrait, -{ - comptime { - let t = quote { T }.as_type(); - let some_trait_i32 = quote { SomeTrait }.as_trait_constraint(); - assert(t.implements(some_trait_i32)); - - assert(t.get_trait_impl(some_trait_i32).is_none()); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L160-L173 - - -### as_type - -```rust title="as_type" showLineNumbers -pub comptime fn as_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L21-L23 - - -Interprets this token stream as a resolved type. Panics if the token -stream doesn't parse to a type or if the type isn't a valid type in scope. - -```rust title="implements_example" showLineNumbers -pub fn function_with_where(_x: T) -where - T: SomeTrait, -{ - comptime { - let t = quote { T }.as_type(); - let some_trait_i32 = quote { SomeTrait }.as_trait_constraint(); - assert(t.implements(some_trait_i32)); - - assert(t.get_trait_impl(some_trait_i32).is_none()); - } -} -``` -> Source code: test_programs/compile_success_empty/comptime_type/src/main.nr#L160-L173 - - -### tokens - -```rust title="tokens" showLineNumbers -pub comptime fn tokens(self) -> [Quoted] {} -``` -> Source code: noir_stdlib/src/meta/quoted.nr#L26-L28 - - -Returns a slice of the individual tokens that form this token stream. - -## Trait Implementations - -```rust -impl Eq for Quoted -impl Hash for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/struct_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/struct_def.md deleted file mode 100644 index fd609942f4e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/struct_def.md +++ /dev/null @@ -1,177 +0,0 @@ ---- -title: StructDefinition ---- - -`std::meta::struct_def` contains methods on the built-in `StructDefinition` type. -This type corresponds to `struct Name { field1: Type1, ... }` items in the source program. - -## Methods - -### add_attribute - -```rust title="add_attribute" showLineNumbers -pub comptime fn add_attribute(self, attribute: str) {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L3-L5 - - -Adds an attribute to the struct. - -### add_generic - -```rust title="add_generic" showLineNumbers -pub comptime fn add_generic(self, generic_name: str) -> Type {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L8-L10 - - -Adds an generic to the struct. Returns the new generic type. -Errors if the given generic name isn't a single identifier or if -the struct already has a generic with the same name. - -This method should be used carefully, if there is existing code referring -to the struct type it may be checked before this function is called and -see the struct with the original number of generics. This method should -thus be preferred to use on code generated from other macros and structs -that are not used in function signatures. - -Example: - -```rust title="add-generic-example" showLineNumbers -comptime fn add_generic(s: StructDefinition) { - assert_eq(s.generics().len(), 0); - let new_generic = s.add_generic("T"); - - let generics = s.generics(); - assert_eq(generics.len(), 1); - assert_eq(generics[0], new_generic); - } -``` -> Source code: test_programs/compile_success_empty/comptime_struct_definition/src/main.nr#L35-L44 - - -### as_type - -```rust title="as_type" showLineNumbers -pub comptime fn as_type(self) -> Type {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L15-L17 - - -Returns this struct as a type in the source program. If this struct has -any generics, the generics are also included as-is. - -### generics - -```rust title="generics" showLineNumbers -pub comptime fn generics(self) -> [Type] {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L26-L28 - - -Returns each generic on this struct. - -Example: - -``` -#[example] -struct Foo { - bar: [T; 2], - baz: Baz, -} - -comptime fn example(foo: StructDefinition) { - assert_eq(foo.generics().len(), 2); - - // Fails because `T` isn't in scope - // let t = quote { T }.as_type(); - // assert_eq(foo.generics()[0], t); -} -``` - -### fields - -```rust title="fields" showLineNumbers -pub comptime fn fields(self) -> [(Quoted, Type)] {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L33-L35 - - -Returns each field of this struct as a pair of (field name, field type). - -### has_named_attribute - -```rust title="has_named_attribute" showLineNumbers -pub comptime fn has_named_attribute(self, name: str) -> bool {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L20-L22 - - -Returns true if this struct has a custom attribute with the given name. - -### module - -```rust title="module" showLineNumbers -pub comptime fn module(self) -> Module {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L38-L40 - - -Returns the module where the struct is defined. - -### name - -```rust title="name" showLineNumbers -pub comptime fn name(self) -> Quoted {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L43-L45 - - -Returns the name of this struct - -Note that the returned quoted value will be just the struct name, it will -not be the full path to the struct, nor will it include any generics. - -### set_fields - -```rust title="set_fields" showLineNumbers -pub comptime fn set_fields(self, new_fields: [(Quoted, Type)]) {} -``` -> Source code: noir_stdlib/src/meta/struct_def.nr#L52-L54 - - -Sets the fields of this struct to the given fields list where each element -is a pair of the field's name and the field's type. Expects each field name -to be a single identifier. Note that this will override any previous fields -on this struct. If those should be preserved, use `.fields()` to retrieve the -current fields on the struct type and append the new fields from there. - -Example: - -```rust -// Change this struct to: -// struct Foo { -// a: u32, -// b: i8, -// } -#[mangle_fields] -struct Foo { x: Field } - -comptime fn mangle_fields(s: StructDefinition) { - s.set_fields(&[ - (quote { a }, quote { u32 }.as_type()), - (quote { b }, quote { i8 }.as_type()), - ]); -} -``` - -## Trait Implementations - -```rust -impl Eq for StructDefinition -impl Hash for StructDefinition -``` - -Note that each struct is assigned a unique ID internally and this is what is used for -equality and hashing. So even structs with identical generics and fields may not -be equal in this sense if they were originally different items in the source program. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_constraint.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_constraint.md deleted file mode 100644 index 3106f732b5a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_constraint.md +++ /dev/null @@ -1,17 +0,0 @@ ---- -title: TraitConstraint ---- - -`std::meta::trait_constraint` contains methods on the built-in `TraitConstraint` type which represents -a trait constraint that can be used to search for a trait implementation. This is similar -syntactically to just the trait itself, but can also contain generic arguments. E.g. `Eq`, `Default`, -`BuildHasher`. - -This type currently has no public methods but it can be used alongside `Type` in `implements` or `get_trait_impl`. - -## Trait Implementations - -```rust -impl Eq for TraitConstraint -impl Hash for TraitConstraint -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_def.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_def.md deleted file mode 100644 index e661d3af7f1..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_def.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: TraitDefinition ---- - -`std::meta::trait_def` contains methods on the built-in `TraitDefinition` type. This type -represents trait definitions such as `trait Foo { .. }` at the top-level of a program. - -## Methods - -### as_trait_constraint - -```rust title="as_trait_constraint" showLineNumbers -pub comptime fn as_trait_constraint(_self: Self) -> TraitConstraint {} -``` -> Source code: noir_stdlib/src/meta/trait_def.nr#L6-L8 - - -Converts this trait into a trait constraint. If there are any generics on this -trait, they will be kept as-is without instantiating or replacing them. - -## Trait Implementations - -```rust -impl Eq for TraitDefinition -impl Hash for TraitDefinition -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_impl.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_impl.md deleted file mode 100644 index a527617c1e6..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/trait_impl.md +++ /dev/null @@ -1,60 +0,0 @@ ---- -title: TraitImpl ---- - -`std::meta::trait_impl` contains methods on the built-in `TraitImpl` type which represents a trait -implementation such as `impl Foo for Bar { ... }`. - -## Methods - -### trait_generic_args - -```rust title="trait_generic_args" showLineNumbers -pub comptime fn trait_generic_args(self) -> [Type] {} -``` -> Source code: noir_stdlib/src/meta/trait_impl.nr#L3-L5 - - -Returns any generic arguments on the trait of this trait implementation, if any. - -```rs -impl Foo for Bar { ... } - -comptime { - let bar_type = quote { Bar }.as_type(); - let foo = quote { Foo }.as_trait_constraint(); - - let my_impl: TraitImpl = bar_type.get_trait_impl(foo).unwrap(); - - let generics = my_impl.trait_generic_args(); - assert_eq(generics.len(), 2); - - assert_eq(generics[0], quote { i32 }.as_type()); - assert_eq(generics[1], quote { Field }.as_type()); -} -``` - -### methods - -```rust title="methods" showLineNumbers -pub comptime fn methods(self) -> [FunctionDefinition] {} -``` -> Source code: noir_stdlib/src/meta/trait_impl.nr#L8-L10 - - -Returns each method in this trait impl. - -Example: - -```rs -comptime { - let i32_type = quote { i32 }.as_type(); - let eq = quote { Eq }.as_trait_constraint(); - - let impl_eq_for_i32: TraitImpl = i32_type.get_trait_impl(eq).unwrap(); - let methods = impl_eq_for_i32.methods(); - - assert_eq(methods.len(), 1); - assert_eq(methods[0].name(), quote { eq }); -} -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/typed_expr.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/typed_expr.md deleted file mode 100644 index 0db7dbfef61..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/typed_expr.md +++ /dev/null @@ -1,27 +0,0 @@ ---- -title: TypedExpr ---- - -`std::meta::typed_expr` contains methods on the built-in `TypedExpr` type for resolved and type-checked expressions. - -## Methods - -### get_type - -```rust title="as_function_definition" showLineNumbers -pub comptime fn as_function_definition(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typed_expr.nr#L7-L9 - - -If this expression refers to a function definitions, returns it. Otherwise returns `Option::none()`. - -### get_type - -```rust title="get_type" showLineNumbers -pub comptime fn get_type(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/typed_expr.nr#L13-L15 - - -Returns the type of the expression, or `Option::none()` if there were errors when the expression was previously resolved. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/unresolved_type.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/unresolved_type.md deleted file mode 100644 index 2826ec5ec0f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/unresolved_type.md +++ /dev/null @@ -1,57 +0,0 @@ ---- -title: UnresolvedType ---- - -`std::meta::unresolved_type` contains methods on the built-in `UnresolvedType` type for the syntax of types. - -## Methods - -### as_mutable_reference - -```rust title="as_mutable_reference" showLineNumbers -comptime fn as_mutable_reference(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L8-L10 - - -If this is a mutable reference type `&mut T`, returns the mutable type `T`. - -### as_slice - -```rust title="as_slice" showLineNumbers -comptime fn as_slice(self) -> Option {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L14-L16 - - -If this is a slice `&[T]`, returns the element type `T`. - -### is_bool - -```rust title="is_bool" showLineNumbers -comptime fn is_bool(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L20-L22 - - -Returns `true` if this type is `bool`. - -### is_field - -```rust title="is_field" showLineNumbers -pub comptime fn is_field(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L26-L28 - - -Returns true if this type refers to the Field type. - -### is_unit - -```rust title="is_unit" showLineNumbers -comptime fn is_unit(self) -> bool {} -``` -> Source code: noir_stdlib/src/meta/unresolved_type.nr#L32-L34 - - -Returns true if this type is the unit `()` type. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/options.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/options.md deleted file mode 100644 index a1bd4e1de5f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/options.md +++ /dev/null @@ -1,101 +0,0 @@ ---- -title: Option Type ---- - -The `Option` type is a way to express that a value might be present (`Some(T))` or absent (`None`). It's a safer way to handle potential absence of values, compared to using nulls in many other languages. - -```rust -struct Option { - None, - Some(T), -} -``` - -The `Option` type, already imported into your Noir program, can be used directly: - -```rust -fn main() { - let none = Option::none(); - let some = Option::some(3); -} -``` - -See [this test](https://github.com/noir-lang/noir/blob/5cbfb9c4a06c8865c98ff2b594464b037d821a5c/crates/nargo_cli/tests/test_data/option/src/main.nr) for a more comprehensive set of examples of each of the methods described below. - -## Methods - -### none - -Constructs a none value. - -### some - -Constructs a some wrapper around a given value. - -### is_none - -Returns true if the Option is None. - -### is_some - -Returns true of the Option is Some. - -### unwrap - -Asserts `self.is_some()` and returns the wrapped value. - -### unwrap_unchecked - -Returns the inner value without asserting `self.is_some()`. This method can be useful within an if condition when we already know that `option.is_some()`. If the option is None, there is no guarantee what value will be returned, only that it will be of type T for an `Option`. - -### unwrap_or - -Returns the wrapped value if `self.is_some()`. Otherwise, returns the given default value. - -### unwrap_or_else - -Returns the wrapped value if `self.is_some()`. Otherwise, calls the given function to return a default value. - -### expect - -Asserts `self.is_some()` with a provided custom message and returns the contained `Some` value. The custom message is expected to be a format string. - -### map - -If self is `Some(x)`, this returns `Some(f(x))`. Otherwise, this returns `None`. - -### map_or - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns the given default value. - -### map_or_else - -If self is `Some(x)`, this returns `f(x)`. Otherwise, this returns `default()`. - -### and - -Returns None if self is None. Otherwise, this returns `other`. - -### and_then - -If self is None, this returns None. Otherwise, this calls the given function with the Some value contained within self, and returns the result of that call. In some languages this function is called `flat_map` or `bind`. - -### or - -If self is Some, return self. Otherwise, return `other`. - -### or_else - -If self is Some, return self. Otherwise, return `default()`. - -### xor - -If only one of the two Options is Some, return that option. Otherwise, if both options are Some or both are None, None is returned. - -### filter - -Returns `Some(x)` if self is `Some(x)` and `predicate(x)` is true. Otherwise, this returns `None`. - -### flatten - -Flattens an `Option>` into a `Option`. This returns `None` if the outer Option is None. Otherwise, this returns the inner Option. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/traits.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/traits.md deleted file mode 100644 index ee20f9cd949..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/traits.md +++ /dev/null @@ -1,628 +0,0 @@ ---- -title: Traits -description: Noir's stdlib provides a few commonly used traits. -keywords: [traits, trait, interface, protocol, default, add, eq] ---- - -## `std::default` - -### `std::default::Default` - -```rust title="default-trait" showLineNumbers -pub trait Default { - fn default() -> Self; -} -``` -> Source code: noir_stdlib/src/default.nr#L4-L8 - - -Constructs a default value of a type. - -Implementations: -```rust -impl Default for Field { .. } - -impl Default for i8 { .. } -impl Default for i16 { .. } -impl Default for i32 { .. } -impl Default for i64 { .. } - -impl Default for u8 { .. } -impl Default for u16 { .. } -impl Default for u32 { .. } -impl Default for u64 { .. } - -impl Default for () { .. } -impl Default for bool { .. } - -impl Default for [T; N] - where T: Default { .. } - -impl Default for [T] { .. } - -impl Default for (A, B) - where A: Default, B: Default { .. } - -impl Default for (A, B, C) - where A: Default, B: Default, C: Default { .. } - -impl Default for (A, B, C, D) - where A: Default, B: Default, C: Default, D: Default { .. } - -impl Default for (A, B, C, D, E) - where A: Default, B: Default, C: Default, D: Default, E: Default { .. } -``` - -For primitive integer types, the return value of `default` is `0`. Container -types such as arrays are filled with default values of their element type, -except slices whose length is unknown and thus defaulted to zero. - ---- - -## `std::convert` - -### `std::convert::From` - -```rust title="from-trait" showLineNumbers -pub trait From { - fn from(input: T) -> Self; -} -``` -> Source code: noir_stdlib/src/convert.nr#L1-L5 - - -The `From` trait defines how to convert from a given type `T` to the type on which the trait is implemented. - -The Noir standard library provides a number of implementations of `From` between primitive types. -```rust title="from-impls" showLineNumbers -// Unsigned integers - -impl From for u32 { - fn from(value: u8) -> u32 { - value as u32 - } -} - -impl From for u64 { - fn from(value: u8) -> u64 { - value as u64 - } -} -impl From for u64 { - fn from(value: u32) -> u64 { - value as u64 - } -} - -impl From for Field { - fn from(value: u8) -> Field { - value as Field - } -} -impl From for Field { - fn from(value: u32) -> Field { - value as Field - } -} -impl From for Field { - fn from(value: u64) -> Field { - value as Field - } -} - -// Signed integers - -impl From for i32 { - fn from(value: i8) -> i32 { - value as i32 - } -} - -impl From for i64 { - fn from(value: i8) -> i64 { - value as i64 - } -} -impl From for i64 { - fn from(value: i32) -> i64 { - value as i64 - } -} - -// Booleans -impl From for u8 { - fn from(value: bool) -> u8 { - value as u8 - } -} -impl From for u32 { - fn from(value: bool) -> u32 { - value as u32 - } -} -impl From for u64 { - fn from(value: bool) -> u64 { - value as u64 - } -} -impl From for i8 { - fn from(value: bool) -> i8 { - value as i8 - } -} -impl From for i32 { - fn from(value: bool) -> i32 { - value as i32 - } -} -impl From for i64 { - fn from(value: bool) -> i64 { - value as i64 - } -} -impl From for Field { - fn from(value: bool) -> Field { - value as Field - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L28-L119 - - -#### When to implement `From` - -As a general rule of thumb, `From` may be implemented in the [situations where it would be suitable in Rust](https://doc.rust-lang.org/std/convert/trait.From.html#when-to-implement-from): - -- The conversion is *infallible*: Noir does not provide an equivalent to Rust's `TryFrom`, if the conversion can fail then provide a named method instead. -- The conversion is *lossless*: semantically, it should not lose or discard information. For example, `u32: From` can losslessly convert any `u16` into a valid `u32` such that the original `u16` can be recovered. On the other hand, `u16: From` should not be implemented as `2**16` is a `u32` which cannot be losslessly converted into a `u16`. -- The conversion is *value-preserving*: the conceptual kind and meaning of the resulting value is the same, even though the Noir type and technical representation might be different. While it's possible to infallibly and losslessly convert a `u8` into a `str<2>` hex representation, `4u8` and `"04"` are too different for `str<2>: From` to be implemented. -- The conversion is *obvious*: it's the only reasonable conversion between the two types. If there's ambiguity on how to convert between them such that the same input could potentially map to two different values then a named method should be used. For instance rather than implementing `U128: From<[u8; 16]>`, the methods `U128::from_le_bytes` and `U128::from_be_bytes` are used as otherwise the endianness of the array would be ambiguous, resulting in two potential values of `U128` from the same byte array. - -One additional recommendation specific to Noir is: -- The conversion is *efficient*: it's relatively cheap to convert between the two types. Due to being a ZK DSL, it's more important to avoid unnecessary computation compared to Rust. If the implementation of `From` would encourage users to perform unnecessary conversion, resulting in additional proving time, then it may be preferable to expose functionality such that this conversion may be avoided. - -### `std::convert::Into` - -The `Into` trait is defined as the reciprocal of `From`. It should be easy to convince yourself that if we can convert to type `A` from type `B`, then it's possible to convert type `B` into type `A`. - -For this reason, implementing `From` on a type will automatically generate a matching `Into` implementation. One should always prefer implementing `From` over `Into` as implementing `Into` will not generate a matching `From` implementation. - -```rust title="into-trait" showLineNumbers -pub trait Into { - fn into(self) -> T; -} - -impl Into for U -where - T: From, -{ - fn into(self) -> T { - T::from(self) - } -} -``` -> Source code: noir_stdlib/src/convert.nr#L13-L26 - - -`Into` is most useful when passing function arguments where the types don't quite match up with what the function expects. In this case, the compiler has enough type information to perform the necessary conversion by just appending `.into()` onto the arguments in question. - ---- - -## `std::cmp` - -### `std::cmp::Eq` - -```rust title="eq-trait" showLineNumbers -pub trait Eq { - fn eq(self, other: Self) -> bool; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L4-L8 - - -Returns `true` if `self` is equal to `other`. Implementing this trait on a type -allows the type to be used with `==` and `!=`. - -Implementations: -```rust -impl Eq for Field { .. } - -impl Eq for i8 { .. } -impl Eq for i16 { .. } -impl Eq for i32 { .. } -impl Eq for i64 { .. } - -impl Eq for u8 { .. } -impl Eq for u16 { .. } -impl Eq for u32 { .. } -impl Eq for u64 { .. } - -impl Eq for () { .. } -impl Eq for bool { .. } - -impl Eq for [T; N] - where T: Eq { .. } - -impl Eq for [T] - where T: Eq { .. } - -impl Eq for (A, B) - where A: Eq, B: Eq { .. } - -impl Eq for (A, B, C) - where A: Eq, B: Eq, C: Eq { .. } - -impl Eq for (A, B, C, D) - where A: Eq, B: Eq, C: Eq, D: Eq { .. } - -impl Eq for (A, B, C, D, E) - where A: Eq, B: Eq, C: Eq, D: Eq, E: Eq { .. } -``` - -### `std::cmp::Ord` - -```rust title="ord-trait" showLineNumbers -pub trait Ord { - fn cmp(self, other: Self) -> Ordering; -} -``` -> Source code: noir_stdlib/src/cmp.nr#L210-L214 - - -`a.cmp(b)` compares two values returning `Ordering::less()` if `a < b`, -`Ordering::equal()` if `a == b`, or `Ordering::greater()` if `a > b`. -Implementing this trait on a type allows `<`, `<=`, `>`, and `>=` to be -used on values of the type. - -`std::cmp` also provides `max` and `min` functions for any type which implements the `Ord` trait. - -Implementations: - -```rust -impl Ord for u8 { .. } -impl Ord for u16 { .. } -impl Ord for u32 { .. } -impl Ord for u64 { .. } - -impl Ord for i8 { .. } -impl Ord for i16 { .. } -impl Ord for i32 { .. } - -impl Ord for i64 { .. } - -impl Ord for () { .. } -impl Ord for bool { .. } - -impl Ord for [T; N] - where T: Ord { .. } - -impl Ord for [T] - where T: Ord { .. } - -impl Ord for (A, B) - where A: Ord, B: Ord { .. } - -impl Ord for (A, B, C) - where A: Ord, B: Ord, C: Ord { .. } - -impl Ord for (A, B, C, D) - where A: Ord, B: Ord, C: Ord, D: Ord { .. } - -impl Ord for (A, B, C, D, E) - where A: Ord, B: Ord, C: Ord, D: Ord, E: Ord { .. } -``` - ---- - -## `std::ops` - -### `std::ops::Add`, `std::ops::Sub`, `std::ops::Mul`, and `std::ops::Div` - -These traits abstract over addition, subtraction, multiplication, and division respectively. -Implementing these traits for a given type will also allow that type to be used with the corresponding operator -for that trait (`+` for Add, etc) in addition to the normal method names. - -```rust title="add-trait" showLineNumbers -pub trait Add { - fn add(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L1-L5 - -```rust title="sub-trait" showLineNumbers -pub trait Sub { - fn sub(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L60-L64 - -```rust title="mul-trait" showLineNumbers -pub trait Mul { - fn mul(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L119-L123 - -```rust title="div-trait" showLineNumbers -pub trait Div { - fn div(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L178-L182 - - -The implementations block below is given for the `Add` trait, but the same types that implement -`Add` also implement `Sub`, `Mul`, and `Div`. - -Implementations: -```rust -impl Add for Field { .. } - -impl Add for i8 { .. } -impl Add for i16 { .. } -impl Add for i32 { .. } -impl Add for i64 { .. } - -impl Add for u8 { .. } -impl Add for u16 { .. } -impl Add for u32 { .. } -impl Add for u64 { .. } -``` - -### `std::ops::Rem` - -```rust title="rem-trait" showLineNumbers -pub trait Rem { - fn rem(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L237-L241 - - -`Rem::rem(a, b)` is the remainder function returning the result of what is -left after dividing `a` and `b`. Implementing `Rem` allows the `%` operator -to be used with the implementation type. - -Unlike other numeric traits, `Rem` is not implemented for `Field`. - -Implementations: -```rust -impl Rem for u8 { fn rem(self, other: u8) -> u8 { self % other } } -impl Rem for u16 { fn rem(self, other: u16) -> u16 { self % other } } -impl Rem for u32 { fn rem(self, other: u32) -> u32 { self % other } } -impl Rem for u64 { fn rem(self, other: u64) -> u64 { self % other } } - -impl Rem for i8 { fn rem(self, other: i8) -> i8 { self % other } } -impl Rem for i16 { fn rem(self, other: i16) -> i16 { self % other } } -impl Rem for i32 { fn rem(self, other: i32) -> i32 { self % other } } -impl Rem for i64 { fn rem(self, other: i64) -> i64 { self % other } } -``` - -### `std::ops::Neg` - -```rust title="neg-trait" showLineNumbers -pub trait Neg { - fn neg(self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L290-L294 - - -`Neg::neg` is equivalent to the unary negation operator `-`. - -Implementations: -```rust title="neg-trait-impls" showLineNumbers -impl Neg for Field { - fn neg(self) -> Field { - -self - } -} - -impl Neg for i8 { - fn neg(self) -> i8 { - -self - } -} -impl Neg for i16 { - fn neg(self) -> i16 { - -self - } -} -impl Neg for i32 { - fn neg(self) -> i32 { - -self - } -} -impl Neg for i64 { - fn neg(self) -> i64 { - -self - } -} -``` -> Source code: noir_stdlib/src/ops/arith.nr#L296-L323 - - -### `std::ops::Not` - -```rust title="not-trait" showLineNumbers -pub trait Not { - fn not(self: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L1-L5 - - -`Not::not` is equivalent to the unary bitwise NOT operator `!`. - -Implementations: -```rust title="not-trait-impls" showLineNumbers -impl Not for bool { - fn not(self) -> bool { - !self - } -} - -impl Not for u64 { - fn not(self) -> u64 { - !self - } -} -impl Not for u32 { - fn not(self) -> u32 { - !self - } -} -impl Not for u16 { - fn not(self) -> u16 { - !self - } -} -impl Not for u8 { - fn not(self) -> u8 { - !self - } -} -impl Not for u1 { - fn not(self) -> u1 { - !self - } -} - -impl Not for i8 { - fn not(self) -> i8 { - !self - } -} -impl Not for i16 { - fn not(self) -> i16 { - !self - } -} -impl Not for i32 { - fn not(self) -> i32 { - !self - } -} -impl Not for i64 { - fn not(self) -> i64 { - !self - } -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L7-L60 - - -### `std::ops::{ BitOr, BitAnd, BitXor }` - -```rust title="bitor-trait" showLineNumbers -pub trait BitOr { - fn bitor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L62-L66 - -```rust title="bitand-trait" showLineNumbers -pub trait BitAnd { - fn bitand(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L121-L125 - -```rust title="bitxor-trait" showLineNumbers -pub trait BitXor { - fn bitxor(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L180-L184 - - -Traits for the bitwise operations `|`, `&`, and `^`. - -Implementing `BitOr`, `BitAnd` or `BitXor` for a type allows the `|`, `&`, or `^` operator respectively -to be used with the type. - -The implementations block below is given for the `BitOr` trait, but the same types that implement -`BitOr` also implement `BitAnd` and `BitXor`. - -Implementations: -```rust -impl BitOr for bool { fn bitor(self, other: bool) -> bool { self | other } } - -impl BitOr for u8 { fn bitor(self, other: u8) -> u8 { self | other } } -impl BitOr for u16 { fn bitor(self, other: u16) -> u16 { self | other } } -impl BitOr for u32 { fn bitor(self, other: u32) -> u32 { self | other } } -impl BitOr for u64 { fn bitor(self, other: u64) -> u64 { self | other } } - -impl BitOr for i8 { fn bitor(self, other: i8) -> i8 { self | other } } -impl BitOr for i16 { fn bitor(self, other: i16) -> i16 { self | other } } -impl BitOr for i32 { fn bitor(self, other: i32) -> i32 { self | other } } -impl BitOr for i64 { fn bitor(self, other: i64) -> i64 { self | other } } -``` - -### `std::ops::{ Shl, Shr }` - -```rust title="shl-trait" showLineNumbers -pub trait Shl { - fn shl(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L239-L243 - -```rust title="shr-trait" showLineNumbers -pub trait Shr { - fn shr(self, other: u8) -> Self; -} -``` -> Source code: noir_stdlib/src/ops/bit.nr#L292-L296 - - -Traits for a bit shift left and bit shift right. - -Implementing `Shl` for a type allows the left shift operator (`<<`) to be used with the implementation type. -Similarly, implementing `Shr` allows the right shift operator (`>>`) to be used with the type. - -Note that bit shifting is not currently implemented for signed types. - -The implementations block below is given for the `Shl` trait, but the same types that implement -`Shl` also implement `Shr`. - -Implementations: -```rust -impl Shl for u8 { fn shl(self, other: u8) -> u8 { self << other } } -impl Shl for u16 { fn shl(self, other: u16) -> u16 { self << other } } -impl Shl for u32 { fn shl(self, other: u32) -> u32 { self << other } } -impl Shl for u64 { fn shl(self, other: u64) -> u64 { self << other } } -``` - ---- - -## `std::append` - -### `std::append::Append` - -`Append` can abstract over types that can be appended to - usually container types: - -```rust title="append-trait" showLineNumbers -pub trait Append { - fn empty() -> Self; - fn append(self, other: Self) -> Self; -} -``` -> Source code: noir_stdlib/src/append.nr#L9-L14 - - -`Append` requires two methods: - -- `empty`: Constructs an empty value of `Self`. -- `append`: Append two values together, returning the result. - -Additionally, it is expected that for any implementation: - -- `T::empty().append(x) == x` -- `x.append(T::empty()) == x` - -Implementations: -```rust -impl Append for [T] -impl Append for Quoted -``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/classes/Noir.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/classes/Noir.md deleted file mode 100644 index ead255bc504..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/classes/Noir.md +++ /dev/null @@ -1,52 +0,0 @@ -# Noir - -## Constructors - -### new Noir(circuit) - -```ts -new Noir(circuit): Noir -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `circuit` | `CompiledCircuit` | - -#### Returns - -[`Noir`](Noir.md) - -## Methods - -### execute() - -```ts -execute(inputs, foreignCallHandler?): Promise -``` - -#### Parameters - -| Parameter | Type | -| :------ | :------ | -| `inputs` | `InputMap` | -| `foreignCallHandler`? | [`ForeignCallHandler`](../type-aliases/ForeignCallHandler.md) | - -#### Returns - -`Promise`\<`object`\> - -#### Description - -Allows to execute a circuit to get its witness and return value. - -#### Example - -```typescript -async execute(inputs) -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/and.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/and.md deleted file mode 100644 index c783283e396..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/and.md +++ /dev/null @@ -1,22 +0,0 @@ -# and() - -```ts -and(lhs, rhs): string -``` - -Performs a bitwise AND operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/blake2s256.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/blake2s256.md deleted file mode 100644 index 7882d0da8d5..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/blake2s256.md +++ /dev/null @@ -1,21 +0,0 @@ -# blake2s256() - -```ts -blake2s256(inputs): Uint8Array -``` - -Calculates the Blake2s256 hash of the input bytes - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `inputs` | `Uint8Array` | | - -## Returns - -`Uint8Array` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md deleted file mode 100644 index 5e3cd53e9d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256k1\_verify() - -```ts -ecdsa_secp256k1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256k1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md deleted file mode 100644 index 0b20ff68957..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify.md +++ /dev/null @@ -1,28 +0,0 @@ -# ecdsa\_secp256r1\_verify() - -```ts -ecdsa_secp256r1_verify( - hashed_msg, - public_key_x_bytes, - public_key_y_bytes, - signature): boolean -``` - -Verifies a ECDSA signature over the secp256r1 curve. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `hashed_msg` | `Uint8Array` | | -| `public_key_x_bytes` | `Uint8Array` | | -| `public_key_y_bytes` | `Uint8Array` | | -| `signature` | `Uint8Array` | | - -## Returns - -`boolean` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/xor.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/xor.md deleted file mode 100644 index 8d762b895d3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/functions/xor.md +++ /dev/null @@ -1,22 +0,0 @@ -# xor() - -```ts -xor(lhs, rhs): string -``` - -Performs a bitwise XOR operation between `lhs` and `rhs` - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `lhs` | `string` | | -| `rhs` | `string` | | - -## Returns - -`string` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/index.md deleted file mode 100644 index 4de7a696991..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/index.md +++ /dev/null @@ -1,47 +0,0 @@ -# noir_js - -## Exports - -### Classes - -| Class | Description | -| :------ | :------ | -| [Noir](classes/Noir.md) | - | - -### Type Aliases - -| Type alias | Description | -| :------ | :------ | -| [ErrorWithPayload](type-aliases/ErrorWithPayload.md) | - | -| [ForeignCallHandler](type-aliases/ForeignCallHandler.md) | A callback which performs an foreign call and returns the response. | -| [ForeignCallInput](type-aliases/ForeignCallInput.md) | - | -| [ForeignCallOutput](type-aliases/ForeignCallOutput.md) | - | -| [WitnessMap](type-aliases/WitnessMap.md) | - | - -### Functions - -| Function | Description | -| :------ | :------ | -| [and](functions/and.md) | Performs a bitwise AND operation between `lhs` and `rhs` | -| [blake2s256](functions/blake2s256.md) | Calculates the Blake2s256 hash of the input bytes | -| [ecdsa\_secp256k1\_verify](functions/ecdsa_secp256k1_verify.md) | Verifies a ECDSA signature over the secp256k1 curve. | -| [ecdsa\_secp256r1\_verify](functions/ecdsa_secp256r1_verify.md) | Verifies a ECDSA signature over the secp256r1 curve. | -| [xor](functions/xor.md) | Performs a bitwise XOR operation between `lhs` and `rhs` | - -## References - -### CompiledCircuit - -Renames and re-exports [InputMap](index.md#inputmap) - -## Variables - -### InputMap - -```ts -InputMap: any; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md deleted file mode 100644 index e8c2f4aef3d..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ErrorWithPayload.md +++ /dev/null @@ -1,15 +0,0 @@ -# ErrorWithPayload - -```ts -type ErrorWithPayload: ExecutionError & object; -``` - -## Type declaration - -| Member | Type | Description | -| :------ | :------ | :------ | -| `decodedAssertionPayload` | `any` | - | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md deleted file mode 100644 index 812b8b16481..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallHandler.md +++ /dev/null @@ -1,24 +0,0 @@ -# ForeignCallHandler - -```ts -type ForeignCallHandler: (name, inputs) => Promise; -``` - -A callback which performs an foreign call and returns the response. - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | The identifier for the type of foreign call being performed. | -| `inputs` | [`ForeignCallInput`](ForeignCallInput.md)[] | An array of hex encoded inputs to the foreign call. | - -## Returns - -`Promise`\<[`ForeignCallOutput`](ForeignCallOutput.md)[]\> - -outputs - An array of hex encoded outputs containing the results of the foreign call. - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md deleted file mode 100644 index dd95809186a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallInput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallInput - -```ts -type ForeignCallInput: string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md deleted file mode 100644 index b71fb78a946..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/ForeignCallOutput.md +++ /dev/null @@ -1,9 +0,0 @@ -# ForeignCallOutput - -```ts -type ForeignCallOutput: string | string[]; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md deleted file mode 100644 index 258c46f9d0c..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/type-aliases/WitnessMap.md +++ /dev/null @@ -1,9 +0,0 @@ -# WitnessMap - -```ts -type WitnessMap: Map; -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs deleted file mode 100644 index 4796b5abaa8..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_js/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"category","label":"Classes","items":[{"type":"doc","id":"reference/NoirJS/noir_js/classes/Noir","label":"Noir"}]},{"type":"category","label":"Type Aliases","items":[{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ErrorWithPayload","label":"ErrorWithPayload"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallHandler","label":"ForeignCallHandler"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallInput","label":"ForeignCallInput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/ForeignCallOutput","label":"ForeignCallOutput"},{"type":"doc","id":"reference/NoirJS/noir_js/type-aliases/WitnessMap","label":"WitnessMap"}]},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_js/functions/and","label":"and"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/blake2s256","label":"blake2s256"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256k1_verify","label":"ecdsa_secp256k1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/ecdsa_secp256r1_verify","label":"ecdsa_secp256r1_verify"},{"type":"doc","id":"reference/NoirJS/noir_js/functions/xor","label":"xor"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/.nojekyll b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/.nojekyll deleted file mode 100644 index e2ac6616add..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/.nojekyll +++ /dev/null @@ -1 +0,0 @@ -TypeDoc added this file to prevent GitHub Pages from using Jekyll. You can turn off this behavior by setting the `githubPages` option to false. \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/compile.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/compile.md deleted file mode 100644 index 6faf763b37f..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/compile.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile() - -```ts -compile( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ProgramCompilationArtifacts`](../index.md#programcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_program(fm); -``` - -```typescript -// Browser - -import { compile_program, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_program(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/compile_contract.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/compile_contract.md deleted file mode 100644 index 7d0b39a43ef..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/compile_contract.md +++ /dev/null @@ -1,51 +0,0 @@ -# compile\_contract() - -```ts -compile_contract( - fileManager, - projectPath?, - logFn?, -debugLogFn?): Promise -``` - -Compiles a Noir project - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `fileManager` | `FileManager` | The file manager to use | -| `projectPath`? | `string` | The path to the project inside the file manager. Defaults to the root of the file manager | -| `logFn`? | `LogFn` | A logging function. If not provided, console.log will be used | -| `debugLogFn`? | `LogFn` | A debug logging function. If not provided, logFn will be used | - -## Returns - -`Promise`\<[`ContractCompilationArtifacts`](../index.md#contractcompilationartifacts)\> - -## Example - -```typescript -// Node.js - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager(myProjectPath); -const myCompiledCode = await compile_contract(fm); -``` - -```typescript -// Browser - -import { compile_contract, createFileManager } from '@noir-lang/noir_wasm'; - -const fm = createFileManager('/'); -for (const path of files) { - await fm.writeFile(path, await getFileAsStream(path)); -} -const myCompiledCode = await compile_contract(fm); -``` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/createFileManager.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/createFileManager.md deleted file mode 100644 index 7e65c1d69c7..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/createFileManager.md +++ /dev/null @@ -1,21 +0,0 @@ -# createFileManager() - -```ts -createFileManager(dataDir): FileManager -``` - -Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `dataDir` | `string` | root of the file system | - -## Returns - -`FileManager` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md deleted file mode 100644 index fcea9275341..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/functions/inflateDebugSymbols.md +++ /dev/null @@ -1,21 +0,0 @@ -# inflateDebugSymbols() - -```ts -inflateDebugSymbols(debugSymbols): any -``` - -Decompresses and decodes the debug symbols - -## Parameters - -| Parameter | Type | Description | -| :------ | :------ | :------ | -| `debugSymbols` | `string` | The base64 encoded debug symbols | - -## Returns - -`any` - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/index.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/index.md deleted file mode 100644 index b6e0f9d1bc0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/index.md +++ /dev/null @@ -1,49 +0,0 @@ -# noir_wasm - -## Exports - -### Functions - -| Function | Description | -| :------ | :------ | -| [compile](functions/compile.md) | Compiles a Noir project | -| [compile\_contract](functions/compile_contract.md) | Compiles a Noir project | -| [createFileManager](functions/createFileManager.md) | Creates a new FileManager instance based on fs in node and memfs in the browser (via webpack alias) | -| [inflateDebugSymbols](functions/inflateDebugSymbols.md) | Decompresses and decodes the debug symbols | - -## References - -### compile\_program - -Renames and re-exports [compile](functions/compile.md) - -## Interfaces - -### ContractCompilationArtifacts - -The compilation artifacts of a given contract. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `contract` | `ContractArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -### ProgramCompilationArtifacts - -The compilation artifacts of a given program. - -#### Properties - -| Property | Type | Description | -| :------ | :------ | :------ | -| `name` | `string` | not part of the compilation output, injected later | -| `program` | `ProgramArtifact` | The compiled contract. | -| `warnings` | `unknown`[] | Compilation warnings. | - -*** - -Generated using [typedoc-plugin-markdown](https://www.npmjs.com/package/typedoc-plugin-markdown) and [TypeDoc](https://typedoc.org/) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs deleted file mode 100644 index e0870710349..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/NoirJS/noir_wasm/typedoc-sidebar.cjs +++ /dev/null @@ -1,4 +0,0 @@ -// @ts-check -/** @type {import('@docusaurus/plugin-content-docs').SidebarsConfig} */ -const typedocSidebar = { items: [{"type":"doc","id":"reference/NoirJS/noir_wasm/index","label":"API"},{"type":"category","label":"Functions","items":[{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile","label":"compile"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/compile_contract","label":"compile_contract"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/createFileManager","label":"createFileManager"},{"type":"doc","id":"reference/NoirJS/noir_wasm/functions/inflateDebugSymbols","label":"inflateDebugSymbols"}]}]}; -module.exports = typedocSidebar.items; \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/_category_.json deleted file mode 100644 index 5b6a20a609a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/_category_.json +++ /dev/null @@ -1,5 +0,0 @@ -{ - "position": 4, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/_category_.json deleted file mode 100644 index 27869205ad3..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/_category_.json +++ /dev/null @@ -1,6 +0,0 @@ -{ - "label": "Debugger", - "position": 1, - "collapsible": true, - "collapsed": true -} diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_known_limitations.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_known_limitations.md deleted file mode 100644 index 936d416ac4b..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_known_limitations.md +++ /dev/null @@ -1,59 +0,0 @@ ---- -title: Known limitations -description: - An overview of known limitations of the current version of the Noir debugger -keywords: - [ - Nargo, - Noir Debugger, - VS Code, - ] -sidebar_position: 2 ---- - -# Debugger Known Limitations - -There are currently some limits to what the debugger can observe. - -## Mutable references - -The debugger is currently blind to any state mutated via a mutable reference. For example, in: - -``` -let mut x = 1; -let y = &mut x; -*y = 2; -``` - -The update on `x` will not be observed by the debugger. That means, when running `vars` from the debugger REPL, or inspecting the _local variables_ pane in the VS Code debugger, `x` will appear with value 1 despite having executed `*y = 2;`. - -## Variables of type function or mutable references are opaque - -When inspecting variables, any variable of type `Function` or `MutableReference` will render its value as `<>` or `<>`. - -## Debugger instrumentation affects resulting ACIR - -In order to make the state of local variables observable, the debugger compiles Noir circuits interleaving foreign calls that track any mutations to them. While this works (except in the cases described above) and doesn't introduce any behavior changes, it does as a side effect produce bigger bytecode. In particular, when running the command `opcodes` on the REPL debugger, you will notice Unconstrained VM blocks that look like this: - -``` -... -5 BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [], q_c: 2 }), Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(2))], q_c: 0 })] - | outputs=[] - 5.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 5.1 | Mov { destination: RegisterIndex(3), source: RegisterIndex(1) } - 5.2 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 5.3 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 5.4 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 5.5 | Mov { destination: RegisterIndex(3), source: RegisterIndex(3) } - 5.6 | Call { location: 8 } - 5.7 | Stop - 5.8 | ForeignCall { function: "__debug_var_assign", destinations: [], inputs: [RegisterIndex(RegisterIndex(2)), RegisterIndex(RegisterIndex(3))] } -... -``` - -If you are interested in debugging/inspecting compiled ACIR without these synthetic changes, you can invoke the REPL debugger with the `--skip-instrumentation` flag or launch the VS Code debugger with the `skipConfiguration` property set to true in its launch configuration. You can find more details about those in the [Debugger REPL reference](debugger_repl.md) and the [VS Code Debugger reference](debugger_vscode.md). - -:::note -Skipping debugger instrumentation means you won't be able to inspect values of local variables. -::: - diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_repl.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_repl.md deleted file mode 100644 index 46e2011304e..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_repl.md +++ /dev/null @@ -1,360 +0,0 @@ ---- -title: REPL Debugger -description: - Noir Debugger REPL options and commands. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - REPL, - ] -sidebar_position: 1 ---- - -## Running the REPL debugger - -`nargo debug [OPTIONS] [WITNESS_NAME]` - -Runs the Noir REPL debugger. If a `WITNESS_NAME` is provided the debugger writes the resulting execution witness to a `WITNESS_NAME` file. - -### Options - -| Option | Description | -| --------------------- | ------------------------------------------------------------ | -| `-p, --prover-name ` | The name of the toml file which contains the inputs for the prover [default: Prover]| -| `--package ` | The name of the package to debug | -| `--print-acir` | Display the ACIR for compiled circuit | -| `--deny-warnings` | Treat all warnings as errors | -| `--silence-warnings` | Suppress warnings | -| `-h, --help` | Print help | - -None of these options are required. - -:::note -Since the debugger starts by compiling the target package, all Noir compiler options are also available. Check out the [compiler reference](../nargo_commands.md#nargo-compile) to learn more about the compiler options. -::: - -## REPL commands - -Once the debugger is running, it accepts the following commands. - -#### `help` (h) - -Displays the menu of available commands. - -``` -> help -Available commands: - - opcodes display ACIR opcodes - into step into to the next opcode - next step until a new source location is reached - out step until a new source location is reached - and the current stack frame is finished - break LOCATION:OpcodeLocation add a breakpoint at an opcode location - over step until a new source location is reached - without diving into function calls - restart restart the debugging session - delete LOCATION:OpcodeLocation delete breakpoint at an opcode location - witness show witness map - witness index:u32 display a single witness from the witness map - witness index:u32 value:String update a witness with the given value - memset index:usize value:String update a memory cell with the given - value - continue continue execution until the end of the - program - vars show variable values available at this point - in execution - stacktrace display the current stack trace - memory show memory (valid when executing unconstrained code) value - step step to the next ACIR opcode - -Other commands: - - help Show this help message - quit Quit repl - -``` - -### Stepping through programs - -#### `next` (n) - -Step until the next Noir source code location. While other commands, such as [`into`](#into-i) and [`step`](#step-s), allow for finer grained control of the program's execution at the opcode level, `next` is source code centric. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `next` here would cause the debugger to jump to the definition of `deep_entry_point` (if available). - -If you want to step over `deep_entry_point` and go straight to line 8, use [the `over` command](#over) instead. - -#### `over` - -Step until the next source code location, without diving into function calls. For example: - -``` -3 ... -4 fn main(x: u32) { -5 assert(entry_point(x) == 2); -6 swap_entry_point(x, x + 1); -7 -> assert(deep_entry_point(x) == 4); -8 multiple_values_entry_point(x); -9 } -``` - - -Using `over` here would cause the debugger to execute until line 8 (`multiple_values_entry_point(x);`). - -If you want to step into `deep_entry_point` instead, use [the `next` command](#next-n). - -#### `out` - -Step until the end of the current function call. For example: - -``` - 3 ... - 4 fn main(x: u32) { - 5 assert(entry_point(x) == 2); - 6 swap_entry_point(x, x + 1); - 7 -> assert(deep_entry_point(x) == 4); - 8 multiple_values_entry_point(x); - 9 } - 10 - 11 unconstrained fn returns_multiple_values(x: u32) -> (u32, u32, u32, u32) { - 12 ... - ... - 55 - 56 unconstrained fn deep_entry_point(x: u32) -> u32 { - 57 -> level_1(x + 1) - 58 } - -``` - -Running `out` here will resume execution until line 8. - -#### `step` (s) - -Skips to the next ACIR code. A compiled Noir program is a sequence of ACIR opcodes. However, an unconstrained VM opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `step` command at this point would result in the debugger stopping at ACIR opcode 2, `EXPR`, skipping unconstrained computation steps. - -Use [the `into` command](#into-i) instead if you want to follow unconstrained computation step by step. - -#### `into` (i) - -Steps into the next opcode. A compiled Noir program is a sequence of ACIR opcodes. However, a BRILLIG opcode denotes the start of an unconstrained code block, to be executed by the unconstrained VM. For example (redacted for brevity): - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -The `->` here shows the debugger paused at an ACIR opcode: `BRILLIG`, at index 1, which denotes an unconstrained code block is about to start. - -Using the `into` command at this point would result in the debugger stopping at opcode 1.0, `Mov ...`, allowing the debugger user to follow unconstrained computation step by step. - -Use [the `step` command](#step-s) instead if you want to skip to the next ACIR code directly. - -#### `continue` (c) - -Continues execution until the next breakpoint, or the end of the program. - -#### `restart` (res) - -Interrupts execution, and restarts a new debugging session from scratch. - -#### `opcodes` (o) - -Display the program's ACIR opcode sequence. For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -### Breakpoints - -#### `break [Opcode]` (or shorthand `b [Opcode]`) - -Sets a breakpoint on the specified opcode index. To get a list of the program opcode numbers, see [the `opcode` command](#opcodes-o). For example: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -In this example, issuing a `break 1.2` command adds break on opcode 1.2, as denoted by the `*` character: - -``` -0 BLACKBOX::RANGE [(_0, num_bits: 32)] [ ] -1 -> BRILLIG inputs=[Single(Expression { mul_terms: [], linear_combinations: [(1, Witness(0))], q_c: 0 })] outputs=[Simple(Witness(1))] - 1.0 | Mov { destination: RegisterIndex(2), source: RegisterIndex(0) } - 1.1 | Const { destination: RegisterIndex(0), value: Value { inner: 0 } } - 1.2 | * Const { destination: RegisterIndex(1), value: Value { inner: 0 } } - 1.3 | Mov { destination: RegisterIndex(2), source: RegisterIndex(2) } - 1.4 | Call { location: 7 } - ... - 1.43 | Return -2 EXPR [ (1, _1) -2 ] -``` - -Running [the `continue` command](#continue-c) at this point would cause the debugger to execute the program until opcode 1.2. - -#### `delete [Opcode]` (or shorthand `d [Opcode]`) - -Deletes a breakpoint at an opcode location. Usage is analogous to [the `break` command](#). - -### Variable inspection - -#### vars - -Show variable values available at this point in execution. - -:::note -The ability to inspect variable values from the debugger depends on compilation to be run in a special debug instrumentation mode. This instrumentation weaves variable tracing code with the original source code. - -So variable value inspection comes at the expense of making the resulting ACIR bytecode bigger and harder to understand and optimize. - -If you find this compromise unacceptable, you can run the debugger with the flag `--skip-debug-instrumentation`. This will compile your circuit without any additional debug information, so the resulting ACIR bytecode will be identical to the one produced by standard Noir compilation. However, if you opt for this, the `vars` command will not be available while debugging. -::: - - -### Stacktrace - -#### `stacktrace` - -Displays the current stack trace. - - -### Witness map - -#### `witness` (w) - -Show witness map. For example: - -``` -_0 = 0 -_1 = 2 -_2 = 1 -``` - -#### `witness [Witness Index]` - -Display a single witness from the witness map. For example: - -``` -> witness 1 -_1 = 2 -``` - -#### `witness [Witness Index] [New value]` - -Overwrite the given index with a new value. For example: - -``` -> witness 1 3 -_1 = 3 -``` - - -### Unconstrained VM memory - -#### `memory` - -Show unconstrained VM memory state. For example: - -``` -> memory -At opcode 1.13: Store { destination_pointer: RegisterIndex(0), source: RegisterIndex(3) } -... -> registers -0 = 0 -1 = 10 -2 = 0 -3 = 1 -4 = 1 -5 = 2³² -6 = 1 -> into -At opcode 1.14: Const { destination: RegisterIndex(5), value: Value { inner: 1 } } -... -> memory -0 = 1 -> -``` - -In the example above: we start with clean memory, then step through a `Store` opcode which stores the value of register 3 (1) into the memory address stored in register 0 (0). Thus now `memory` shows memory address 0 contains value 1. - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: - -#### `memset [Memory address] [New value]` - -Update a memory cell with the given value. For example: - -``` -> memory -0 = 1 -> memset 0 2 -> memory -0 = 2 -> memset 1 4 -> memory -0 = 2 -1 = 4 -> -``` - -:::note -This command is only functional while the debugger is executing unconstrained code. -::: \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_vscode.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_vscode.md deleted file mode 100644 index c027332b3b0..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/debugger/debugger_vscode.md +++ /dev/null @@ -1,82 +0,0 @@ ---- -title: VS Code Debugger -description: - VS Code Debugger configuration and features. -keywords: - [ - Nargo, - Noir CLI, - Noir Debugger, - VS Code, - IDE, - ] -sidebar_position: 0 ---- - -# VS Code Noir Debugger Reference - -The Noir debugger enabled by the vscode-noir extension ships with default settings such that the most common scenario should run without any additional configuration steps. - -These defaults can nevertheless be overridden by defining a launch configuration file. This page provides a reference for the properties you can override via a launch configuration file, as well as documenting the Nargo `dap` command, which is a dependency of the VS Code Noir debugger. - - -## Creating and editing launch configuration files - -To create a launch configuration file from VS Code, open the _debug pane_, and click on _create a launch.json file_. - -![Creating a launch configuration file](@site/static/img/debugger/ref1-create-launch.png) - -A `launch.json` file will be created, populated with basic defaults. - -### Noir Debugger launch.json properties - -#### projectFolder - -_String, optional._ - -Absolute path to the Nargo project to debug. By default, it is dynamically determined by looking for the nearest `Nargo.toml` file to the active file at the moment of launching the debugger. - -#### proverName - -_String, optional._ - -Name of the prover input to use. Defaults to `Prover`, which looks for a file named `Prover.toml` at the `projectFolder`. - -#### generateAcir - -_Boolean, optional._ - -If true, generate ACIR opcodes instead of unconstrained opcodes which will be closer to release binaries but less convenient for debugging. Defaults to `false`. - -#### skipInstrumentation - -_Boolean, optional._ - -Skips variables debugging instrumentation of code, making debugging less convenient but the resulting binary smaller and closer to production. Defaults to `false`. - -:::note -Skipping instrumentation causes the debugger to be unable to inspect local variables. -::: - -## `nargo dap [OPTIONS]` - -When run without any option flags, it starts the Nargo Debug Adapter Protocol server, which acts as the debugging backend for the VS Code Noir Debugger. - -All option flags are related to preflight checks. The Debug Adapter Protocol specifies how errors are to be informed from a running DAP server, but it doesn't specify mechanisms to communicate server initialization errors between the DAP server and its client IDE. - -Thus `nargo dap` ships with a _preflight check_ mode. If flag `--preflight-check` and the rest of the `--preflight-*` flags are provided, Nargo will run the same initialization routine except it will not start the DAP server. - -`vscode-noir` will then run `nargo dap` in preflight check mode first before a debugging session starts. If the preflight check ends in error, vscode-noir will present stderr and stdout output from this process through its own Output pane in VS Code. This makes it possible for users to diagnose what pieces of configuration might be wrong or missing in case of initialization errors. - -If the preflight check succeeds, `vscode-noir` proceeds to start the DAP server normally but running `nargo dap` without any additional flags. - -### Options - -| Option | Description | -| --------------------------------------------------------- | --------------------------------------------------------------------------------------------------------- | -| `--preflight-check` | If present, dap runs in preflight check mode. | -| `--preflight-project-folder ` | Absolute path to the project to debug for preflight check. | -| `--preflight-prover-name ` | Name of prover file to use for preflight check | -| `--preflight-generate-acir` | Optional. If present, compile in ACIR mode while running preflight check. | -| `--preflight-skip-instrumentation` | Optional. If present, compile without introducing debug instrumentation while running preflight check. | -| `-h, --help` | Print help. | diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/noir_codegen.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/noir_codegen.md deleted file mode 100644 index e4c362f9610..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/reference/noir_codegen.md +++ /dev/null @@ -1,116 +0,0 @@ ---- -title: Noir Codegen for TypeScript -description: Learn how to use Noir codegen to generate TypeScript bindings -keywords: [Nargo, Noir, compile, TypeScript] -sidebar_position: 3 ---- - -When using TypeScript, it is extra work to interpret Noir program outputs in a type-safe way. Third party libraries may exist for popular Noir programs, but they are either hard to find or unmaintained. - -Now you can generate TypeScript bindings for your Noir programs in two steps: - -1. Exporting Noir functions using `nargo export` -2. Using the TypeScript module `noir_codegen` to generate TypeScript binding - -**Note:** you can only export functions from a Noir *library* (not binary or contract program types). - -## Installation - -### Your TypeScript project - -If you don't already have a TypeScript project you can add the module with `yarn` (or `npm`), then initialize it: - -```bash -yarn add typescript -D -npx tsc --init -``` - -### Add TypeScript module - `noir_codegen` - -The following command will add the module to your project's devDependencies: - -```bash -yarn add @noir-lang/noir_codegen -D -``` - -### Nargo library - -Make sure you have Nargo, v0.25.0 or greater, installed. If you don't, follow the [installation guide](../getting_started/noir_installation.md). - -If you're in a new project, make a `circuits` folder and create a new Noir library: - -```bash -mkdir circuits && cd circuits -nargo new --lib myNoirLib -``` - -## Usage - -### Export ABI of specified functions - -First go to the `.nr` files in your Noir library, and add the `#[export]` macro to each function that you want to use in TypeScript. - -```rust -#[export] -fn your_function(... -``` - -From your Noir library (where `Nargo.toml` is), run the following command: - -```bash -nargo export -``` - -You will now have an `export` directory with a .json file per exported function. - -You can also specify the directory of Noir programs using `--program-dir`, for example: - -```bash -nargo export --program-dir=./circuits/myNoirLib -``` - -### Generate TypeScript bindings from exported functions - -To use the `noir-codegen` package we added to the TypeScript project: - -```bash -yarn noir-codegen ./export/your_function.json -``` - -This creates an `exports` directory with an `index.ts` file containing all exported functions. - -**Note:** adding `--out-dir` allows you to specify an output dir for your TypeScript bindings to go. Eg: - -```bash -yarn noir-codegen ./export/*.json --out-dir ./path/to/output/dir -``` - -## Example .nr function to .ts output - -Consider a Noir library with this function: - -```rust -#[export] -fn not_equal(x: Field, y: Field) -> bool { - x != y -} -``` - -After the export and codegen steps, you should have an `index.ts` like: - -```typescript -export type Field = string; - - -export const is_equal_circuit: CompiledCircuit = -{"abi":{"parameters":[{"name":"x","type":{"kind":"field"},"visibility":"private"},{"name":"y","type":{"kind":"field"},"visibility":"private"}],"return_type":{"abi_type":{"kind":"boolean"},"visibility":"private"}},"bytecode":"H4sIAAAAAAAA/7WUMQ7DIAxFQ0Krrr2JjSGYLVcpKrn/CaqqDQN12WK+hPBgmWd/wEyHbF1SS923uhOs3pfoChI+wKXMAXzIKyNj4PB0TFTYc0w5RUjoqeAeEu1wqK0F54RGkWvW44LPzExnlkbMEs4JNZmN8PxS42uHv82T8a3Jeyn2Ks+VLPcO558HmyLMCDOXAXXtpPt4R/Rt9T36ss6dS9HGPx/eG17nGegKBQAA"}; - -export async function is_equal(x: Field, y: Field, foreignCallHandler?: ForeignCallHandler): Promise { - const program = new Noir(is_equal_circuit); - const args: InputMap = { x, y }; - const { returnValue } = await program.execute(args, foreignCallHandler); - return returnValue as boolean; -} -``` - -Now the `is_equal()` function and relevant types are readily available for use in TypeScript. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/debugger.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/debugger.md deleted file mode 100644 index 200b5fc423a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/debugger.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: Debugger -description: Learn about the Noir Debugger, in its REPL or VS Code versions. -keywords: [Nargo, VSCode, Visual Studio Code, REPL, Debugger] -sidebar_position: 2 ---- - -# Noir Debugger - -There are currently two ways of debugging Noir programs: - -1. From VS Code, via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). -2. Via the REPL debugger, which ships with Nargo. - -In order to use either version of the debugger, you will need to install recent enough versions of Noir, [Nargo](../getting_started/noir_installation.md) and vscode-noir: - -- Noir & Nargo ≥0.28.0 -- Noir's VS Code extension ≥0.0.11 - -:::info -At the moment, the debugger supports debugging binary projects, but not contracts. -::: - -We cover the VS Code Noir debugger more in depth in [its VS Code debugger how-to guide](../how_to/debugger/debugging_with_vs_code.md) and [the reference](../reference/debugger/debugger_vscode.md). - -The REPL debugger is discussed at length in [the REPL debugger how-to guide](../how_to/debugger/debugging_with_the_repl.md) and [the reference](../reference/debugger/debugger_repl.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/language_server.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/language_server.md deleted file mode 100644 index 81e0356ef8a..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/language_server.md +++ /dev/null @@ -1,43 +0,0 @@ ---- -title: Language Server -description: Learn about the Noir Language Server, how to install the components, and configuration that may be required. -keywords: [Nargo, Language Server, LSP, VSCode, Visual Studio Code] -sidebar_position: 0 ---- - -This section helps you install and configure the Noir Language Server. - -The Language Server Protocol (LSP) has two components, the [Server](#language-server) and the [Client](#language-client). Below we describe each in the context of Noir. - -## Language Server - -The Server component is provided by the Nargo command line tool that you installed at the beginning of this guide. -As long as Nargo is installed and you've used it to run other commands in this guide, it should be good to go! - -If you'd like to verify that the `nargo lsp` command is available, you can run `nargo --help` and look for `lsp` in the list of commands. If you see it, you're using a version of Noir with LSP support. - -## Language Client - -The Client component is usually an editor plugin that launches the Server. It communicates LSP messages between the editor and the Server. For example, when you save a file, the Client will alert the Server, so it can try to compile the project and report any errors. - -Currently, Noir provides a Language Client for Visual Studio Code via the [vscode-noir](https://github.com/noir-lang/vscode-noir) extension. You can install it via the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=noir-lang.vscode-noir). - -> **Note:** Noir's Language Server Protocol support currently assumes users' VSCode workspace root to be the same as users' Noir project root (i.e. where Nargo.toml lies). -> -> If LSP features seem to be missing / malfunctioning, make sure you are opening your Noir project directly (instead of as a sub-folder) in your VSCode instance. - -When your language server is running correctly and the VSCode plugin is installed, you should see handy codelens buttons for compilation, measuring circuit size, execution, and tests: - -![Compile and Execute](@site/static/img/codelens_compile_execute.png) -![Run test](@site/static/img/codelens_run_test.png) - -You should also see your tests in the `testing` panel: - -![Testing panel](@site/static/img/codelens_testing_panel.png) - -### Configuration - -- **Noir: Enable LSP** - If checked, the extension will launch the Language Server via `nargo lsp` and communicate with it. -- **Noir: Nargo Flags** - Additional flags may be specified if you require them to be added when the extension calls `nargo lsp`. -- **Noir: Nargo Path** - An absolute path to a Nargo binary with the `lsp` command. This may be useful if Nargo is not within the `PATH` of your editor. -- **Noir > Trace: Server** - Setting this to `"messages"` or `"verbose"` will log LSP messages between the Client and Server. Useful for debugging. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/testing.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/testing.md deleted file mode 100644 index 866677da567..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tooling/testing.md +++ /dev/null @@ -1,79 +0,0 @@ ---- -title: Testing in Noir -description: Learn how to use Nargo to test your Noir program in a quick and easy way -keywords: [Nargo, testing, Noir, compile, test] -sidebar_position: 1 ---- - -You can test your Noir programs using Noir circuits. - -Nargo will automatically compile and run any functions which have the decorator `#[test]` on them if -you run `nargo test`. - -For example if you have a program like: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test] -fn test_add() { - assert(add(2,2) == 4); - assert(add(0,1) == 1); - assert(add(1,0) == 1); -} -``` - -Running `nargo test` will test that the `test_add` function can be executed while satisfying all -the constraints which allows you to test that add returns the expected values. Test functions can't -have any arguments currently. - -### Test fail - -You can write tests that are expected to fail by using the decorator `#[test(should_fail)]`. For example: - -```rust -fn add(x: u64, y: u64) -> u64 { - x + y -} -#[test(should_fail)] -fn test_add() { - assert(add(2,2) == 5); -} -``` - -You can be more specific and make it fail with a specific reason by using `should_fail_with = ""`: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "What is the airspeed velocity of an unladen swallow")] -fn test_bridgekeeper() { - main(32); -} -``` - -The string given to `should_fail_with` doesn't need to exactly match the failure reason, it just needs to be a substring of it: - -```rust -fn main(african_swallow_avg_speed : Field) { - assert(african_swallow_avg_speed == 65, "What is the airspeed velocity of an unladen swallow"); -} - -#[test] -fn test_king_arthur() { - main(65); -} - -#[test(should_fail_with = "airspeed velocity")] -fn test_bridgekeeper() { - main(32); -} -``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tutorials/noirjs_app.md deleted file mode 100644 index 6e69ea0bbed..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/tutorials/noirjs_app.md +++ /dev/null @@ -1,366 +0,0 @@ ---- -title: Building a web app with NoirJS -description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. -keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] -sidebar_position: 0 -pagination_next: noir/concepts/data_types/index ---- - -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! - -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/quick_start.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` - -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! - -### Nargo - -Run: - -```bash -nargo new circuit -``` - -And... That's about it. Your program is ready to be compiled and run. - -To compile, let's `cd` into the `circuit` folder to enter our project, and call: - -```bash -nargo compile -``` - -This compiles our circuit into `json` format and add it to a new `target` folder. - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` - -::: - -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/quick_start.md) guide. However, we want our app to run on the browser, so we need Vite. - -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. - -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". - -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: - -```bash -cd vite-project -``` - -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; - -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); -``` - -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: - -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev -``` - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... -``` - -::: - -#### Some cleanup - -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. - -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) - -## HTML - -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: - -```html - - - - - - -

Noir app

-
- - -
-
-

Logs

-

Proof

-
- - -``` - -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). - -Start by pasting in this boilerplate code: - -```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} - -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } -}); -``` - -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 - -:::info - -At this point in the tutorial, your folder structure should look like this: - -```tree -. -└── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html -``` - -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - -::: - -## Some NoirJS - -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: - -```ts -import circuit from '../circuit/target/circuit.json'; -``` - -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: - -```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -import { Noir } from '@noir-lang/noir_js'; -``` - -And instantiate them inside our try-catch block: - -```ts -// try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); -// } -``` - -:::note - -For the remainder of the tutorial, everything will be happening inside the `try` block - -::: - -## Our app - -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: - -```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; -``` - -Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: - -```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); -const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); -``` - -You're probably eager to see stuff happening, so go and run your app now! - -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. - -![Getting Started 0](@site/static/img/noir_getting_started_1.png) - -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! - -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. - -## Verifying - -Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: - -```js -display('logs', 'Verifying proof... ⌛'); -const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); -``` - -You have successfully generated a client-side Noir web app! - -![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) - -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. - -## UltraHonk Backend - -Barretenberg has recently exposed a new UltraHonk backend. We can use UltraHonk in NoirJS after version 0.33.0. Everything will be the same as the tutorial above, except that the class we need to import will change: - -```js -import { UltraHonkBackend, UltraHonkVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -``` - -The backend will then be instantiated as such: - -```js -const backend = new UltraHonkBackend(circuit); -``` - -Then all the commands to prove and verify your circuit will be same. - -The only feature currently unsupported with UltraHonk are [recursive proofs](../explainers/explainer-recursion.md). diff --git a/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/how_to/how-to-solidity-verifier.md deleted file mode 100644 index 2cc0f8e57ce..00000000000 --- a/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/how_to/how-to-solidity-verifier.md +++ /dev/null @@ -1,259 +0,0 @@ ---- -title: Generate a Solidity Verifier -description: - Learn how to run the verifier as a smart contract on the blockchain. Compile a Solidity verifier - contract for your Noir program and deploy it on any EVM blockchain acting as a verifier smart - contract. Read more to find out -keywords: - [ - solidity verifier, - smart contract, - blockchain, - compiler, - plonk_vk.sol, - EVM blockchain, - verifying Noir programs, - proving backend, - Barretenberg, - ] -sidebar_position: 0 -pagination_next: tutorials/noirjs_app ---- - -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. - -This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. - -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: - -- You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network -- You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/quick_start.md) with Nargo and the example Hello Noir circuit -- You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. - -## Rundown - -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: - -1. How to generate a solidity smart contract -2. How to compile the smart contract in the RemixIDE -3. How to deploy it to a testnet - -## Step 1 - Generate a contract - -This is by far the most straightforward step. Just run: - -```sh -nargo compile -``` - -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: - -```sh -# Here we pass the path to the newly generated Noir artifact. -bb write_vk -b ./target/.json -bb contract -``` - -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. - -You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/quick_start.md#proving-backend). - -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. - -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: - -## Step 2 - Compiling - -We will mostly skip the details of RemixIDE, as the UI can change from version to version. For now, we can just open -Remix and create a blank workspace. - -![Create Workspace](@site/static/img/how-tos/solidity_verifier_1.png) - -We will create a new file to contain the contract Nargo generated, and copy-paste its content. - -:::warning - -You'll likely see a warning advising you to not trust pasted code. While it is an important warning, it is irrelevant in the context of this guide and can be ignored. We will not be deploying anywhere near a mainnet. - -::: - -To compile our the verifier, we can navigate to the compilation tab: - -![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) - -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: - -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) - -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: - -![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) - -## Step 3 - Deploying - -At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. - -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": - -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) - -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. - -:::note - -Why "UltraVerifier"? - -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. - -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: - -## Step 4 - Verifying - -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: - -```solidity -function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) -``` - -When using the default example in the [Hello Noir](../getting_started/quick_start.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. - -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/quick_start.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the - -```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') - -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS - -echo "Proof:" -echo "0x$HEX_PROOF" -``` - -Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. - -A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): - -```solidity -function castVote(bytes calldata proof, uint proposalId, uint vote, bytes32 nullifierHash) public returns (bool) { - // ... - bytes32[] memory publicInputs = new bytes32[](4); - publicInputs[0] = merkleRoot; - publicInputs[1] = bytes32(proposalId); - publicInputs[2] = bytes32(vote); - publicInputs[3] = nullifierHash; - require(verifier.verify(proof, publicInputs), "Invalid proof"); -``` - -:::info[Return Values] - -A circuit doesn't have the concept of a return value. Return values are just syntactic sugar in Noir. - -Under the hood, the return value is passed as an input to the circuit and is checked at the end of the circuit program. - -For example, if you have Noir program like this: - -```rust -fn main( - // Public inputs - pubkey_x: pub Field, - pubkey_y: pub Field, - // Private inputs - priv_key: Field, -) -> pub Field -``` - -the `verify` function will expect the public inputs array (second function parameter) to be of length 3, the two inputs and the return value. - -Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. - -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. - -::: - -:::tip[Structs] - -You can pass structs to the verifier contract. They will be flattened so that the array of inputs is 1-dimensional array. - -For example, consider the following program: - -```rust -struct Type1 { - val1: Field, - val2: Field, -} - -struct Nested { - t1: Type1, - is_true: bool, -} - -fn main(x: pub Field, nested: pub Nested, y: pub Field) { - //... -} -``` - -The order of these inputs would be flattened to: `[x, nested.t1.val1, nested.t1.val2, nested.is_true, y]` - -::: - -The other function you can call is our entrypoint `verify` function, as defined above. - -:::tip - -It's worth noticing that the `verify` function is actually a `view` function. A `view` function does not alter the blockchain state, so it doesn't need to be distributed (i.e. it will run only on the executing node), and therefore doesn't cost any gas. - -This can be particularly useful in some situations. If Alice generated a proof and wants Bob to verify its correctness, Bob doesn't need to run Nargo, NoirJS, or any Noir specific infrastructure. He can simply make a call to the blockchain with the proof and verify it is correct without paying any gas. - -It would be incorrect to say that a Noir proof verification costs any gas at all. However, most of the time the result of `verify` is used to modify state (for example, to update a balance, a game state, etc). In that case the whole network needs to execute it, which does incur gas costs (calldata and execution, but not storage). - -::: - -## A Note on EVM chains - -Noir proof verification requires the ecMul, ecAdd and ecPairing precompiles. Not all EVM chains support EC Pairings, notably some of the ZK-EVMs. This means that you won't be able to use the verifier contract in all of them. You can find an incomplete list of which EVM chains support these precompiles [here](https://www.evmdiff.com/features?feature=precompiles). - -For example, chains like `zkSync ERA` and `Polygon zkEVM` do not currently support these precompiles, so proof verification via Solidity verifier contracts won't work. Here's a quick list of EVM chains that have been tested and are known to work: - -- Optimism -- Arbitrum -- Polygon PoS -- Scroll -- Celo -- BSC -- Blast L2 -- Avalanche C-Chain -- Mode -- Linea -- Moonbeam - -If you test any other chains, please open a PR on this page to update the list. See [this doc](https://github.com/noir-lang/noir-starter/tree/main/with-foundry#testing-on-chain) for more info about testing verifier contracts on different EVM chains. - -## What's next - -Now that you know how to call a Noir Solidity Verifier on a smart contract using Remix, you should be comfortable with using it with some programmatic frameworks, such as [hardhat](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat) and [foundry](https://github.com/noir-lang/noir-starter/tree/main/with-foundry). - -You can find other tools, examples, boilerplates and libraries in the [awesome-noir](https://github.com/noir-lang/awesome-noir) repository. - -You should also be ready to write and deploy your first NoirJS app and start generating proofs on websites, phones, and NodeJS environments! Head on to the [NoirJS tutorial](../tutorials/noirjs_app.md) to learn how to do that. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/how_to/how-to-solidity-verifier.mdx similarity index 64% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-solidity-verifier.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/how_to/how-to-solidity-verifier.mdx index 2cc0f8e57ce..da36b60920d 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/how_to/how-to-solidity-verifier.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/how_to/how-to-solidity-verifier.mdx @@ -20,24 +20,37 @@ sidebar_position: 0 pagination_next: tutorials/noirjs_app --- -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +Noir is universal. The witness and the compiled program can be fed into a proving backend such as Aztec's [Barretenberg](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg), which can then generate a verifier contract for deployment on blockchains. This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: +Although not strictly in the domain of Noir itself, this guide shows how to generate a Solidity Verifier with Barretenberg and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: +- You will be using Barretenberg as your proving backend +- You will be using an EVM blockchain to verify your proof - You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network - You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/quick_start.md) with Nargo and the example Hello Noir circuit - You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. ## Rundown -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: +Generating a Solidity Verifier with Barretenberg contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: 1. How to generate a solidity smart contract 2. How to compile the smart contract in the RemixIDE 3. How to deploy it to a testnet +:::info[Which proving system to use?] + +Barretenberg currently provides two provers: `UltraPlonk` and `UltraHonk`. In a nutshell, `UltraHonk` is faster and uses less RAM, but its verifier contract is much more expensive. `UltraPlonk` is optimized for on-chain verification, but proving is more expensive. + +In any case, we provide instructions for both. Choose your poison ☠️ + +::: + ## Step 1 - Generate a contract This is by far the most straightforward step. Just run: @@ -46,25 +59,31 @@ This is by far the most straightforward step. Just run: nargo compile ``` -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: +This will compile your source code into a Noir build artifact to be stored in the `./target` directory. From here on, it's Barretenberg's work. You can generate the smart contract using the commands: + + + + +```sh +bb write_vk_ultra_keccak_honk -b ./target/.json +bb contract_ultra_honk +``` + + + ```sh -# Here we pass the path to the newly generated Noir artifact. bb write_vk -b ./target/.json bb contract ``` -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. + + -You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/quick_start.md#proving-backend). +replacing `` with the name of your Noir project. A `Verifier.sol` contract is now in the target folder and can be deployed to any EVM blockchain acting as a verifier smart contract. -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. +You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/quick_start.md#proving-backend). -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: ## Step 2 - Compiling @@ -85,17 +104,12 @@ To compile our the verifier, we can navigate to the compilation tab: ![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: +Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely tell you the contract is too big to deploy on mainnet, or complain about a stack too deep: -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) +![Contract code too big](@site/static/img/how-tos/solidity_verifier_6.png) +![Stack too deep](@site/static/img/how-tos/solidity_verifier_8.png) -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: +To avoid this, you can just use some optimization. Open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. ![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) @@ -103,54 +117,81 @@ This time we will see a warning about an unused function parameter. This is expe At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": +Looking closely, we will notice that our "Solidity Verifier" is composed on multiple contracts working together. Remix will take care of the dependencies for us so we can simply deploy the Verifier contract by selecting it and hitting "deploy": -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) + + -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. +![Deploying HonkVerifier](@site/static/img/how-tos/solidity_verifier_7.png) -:::note + + -Why "UltraVerifier"? +![Deploying PlonkVerifier](@site/static/img/how-tos/solidity_verifier_9.png) -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. + + -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: +A contract will show up in the "Deployed Contracts" section. ## Step 4 - Verifying -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: +To verify a proof using the Solidity verifier contract, we call the `verify` function: ```solidity function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) ``` -When using the default example in the [Hello Noir](../getting_started/quick_start.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. +First generate a proof with `bb`. We need a `Prover.toml` file for our inputs. Run: -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/quick_start.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the +```bash +nargo check +``` + +This will generate a `Prover.toml` you can fill with the values you want to prove. We can now execute the circuit with `nargo` and then use the proving backend to prove: + + + + +```bash +nargo execute +bb prove_ultra_keccak_honk -b ./target/.json -w ./target/ -o ./target/proof +``` + +:::tip[Public inputs] +Barretenberg attaches the public inputs to the proof, which in this case it's not very useful. If you're up for some JS, `bb.js` has [a method for it](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/ts/src/proof/index.ts), but in the CLI you can use this ugly snippet: + +```bash +cat ./target/proof | od -An -v -t x1 | tr -d $' \n' | sed 's/^.\{8\}//' | (read hex; echo "${hex:0:192}${hex:256}") +``` + +Beautiful. This assumes a circuit with one public input (32 bytes, for Barretenberg). For more inputs, you can just increment `hex:256` with 32 more bytes for each public input. + +::: + + + ```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') +nargo execute +bb prove -b ./target/.json -w ./target/ -o ./target/proof +``` + -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS +:::tip[Public inputs] +Barretenberg attaches the public inputs to the proof, which in this case it's not very useful. If you're up for some JS, `bb.js` has [a method for it](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/ts/src/proof/index.ts), but in the CLI you can use this ugly snippet: -echo "Proof:" -echo "0x$HEX_PROOF" +```bash +tail -c +33 ./target/proof | od -An -v -t x1 | tr -d $' \n' ``` +Beautiful. This assumes a circuit with one public input (32 bytes, for Barretenberg). For more inputs, you can just add 32 more bytes for each public input to the `tail` command. + +::: + + + + Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): @@ -188,7 +229,7 @@ the `verify` function will expect the public inputs array (second function param Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. +In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return]`. ::: diff --git a/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/tutorials/noirjs_app.md index 6e69ea0bbed..8967ee005ce 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/tutorials/noirjs_app.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.0/tutorials/noirjs_app.md @@ -1,186 +1,91 @@ --- -title: Building a web app with NoirJS +title: Building a web app with Noir and Barretenberg description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] sidebar_position: 0 pagination_next: noir/concepts/data_types/index --- -NoirJS is a set of packages meant to work both in a browser and a server environment. In this tutorial, we will build a simple web app using them. From here, you should get an idea on how to proceed with your own Noir projects! +NoirJS is a Typescript package meant to work both in a browser and a server environment. -You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). - -## Setup - -:::note - -Feel free to use whatever versions, just keep in mind that Nargo and the NoirJS packages are meant to be in sync. For example, Nargo 0.31.x matches `noir_js@0.31.x`, etc. - -In this guide, we will be pinned to 0.31.0. - -::: - -Before we start, we want to make sure we have Node, Nargo and the Barretenberg proving system (`bb`) installed. - -We start by opening a terminal and executing `node --version`. If we don't get an output like `v20.10.0`, that means node is not installed. Let's do that by following the handy [nvm guide](https://github.com/nvm-sh/nvm?tab=readme-ov-file#install--update-script). - -As for `Nargo`, we can follow the [Nargo guide](../getting_started/quick_start.md) to install it. If you're lazy, just paste this on a terminal and run `noirup`: - -```sh -curl -L https://raw.githubusercontent.com/noir-lang/noirup/main/install | bash -``` +In this tutorial, we will combine NoirJS with Aztec's Barretenberg backend to build a simple web app. From here, you should get an idea on how to proceed with your own Noir projects! -Follow the instructions on [this page](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg/cpp/src/barretenberg/bb#installation) to install `bb`. -Version 0.41.0 is compatible with `nargo` version 0.31.0, which you can install with `bbup -v 0.41.0` once `bbup` is installed. - -Easy enough. Onwards! - -## Our project - -ZK is a powerful technology. An app that doesn't reveal one of the inputs to _anyone_ is almost unbelievable, yet Noir makes it as easy as a single line of code. - -In fact, it's so simple that it comes nicely packaged in `nargo`. Let's do that! +You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). -### Nargo +## Dependencies -Run: +Before we start, we want to make sure we have Node installed. For convenience (and speed), we can just install [Bun](https://bun.sh) as our package manager, and Node will work out-of-the-box: ```bash -nargo new circuit +curl -fsSL https://bun.sh/install | bash ``` -And... That's about it. Your program is ready to be compiled and run. +Let's go barebones. Doing the bare minimum is not only simple, but also allows you to easily adapt it to almost any frontend framework. -To compile, let's `cd` into the `circuit` folder to enter our project, and call: +Barebones means we can immediately start with the dependencies even on an empty folder 😈: ```bash -nargo compile +bun i @noir-lang/noir_wasm@1.0.0-beta.0 @noir-lang/noir_js@1.0.0-beta.0 @aztec/bb.js@0.63.1 ``` -This compiles our circuit into `json` format and add it to a new `target` folder. +Wait, what are these dependencies? -:::info +- `noir_wasm` is the `wasm` version of the Noir compiler. Although most developers prefer to use `nargo` for compiling, there's nothing wrong with `noir_wasm`. We like `noir_wasm`. +- `noir_js` is the main Noir package. It will execute our program, and generate the witness that will be sent to the backend. +- `bb.js` is the Typescript interface for Aztec's Barretenberg proving backend. It also uses the `wasm` version in order to run on the browser. -At this point in the tutorial, your folder structure should look like this: +:::info -```tree -. -└── circuit <---- our working directory - ├── Nargo.toml - ├── src - │ └── main.nr - └── target - └── circuit.json -``` +In this guide, we will install versions pinned to 1.0.0-beta.0. These work with Barretenberg version 0.63.1, so we are using that one version too. Feel free to try with older or later versions, though! ::: -### Node and Vite - -If you want to explore Nargo, feel free to go on a side-quest now and follow the steps in the -[getting started](../getting_started/quick_start.md) guide. However, we want our app to run on the browser, so we need Vite. +## Setting up our Noir program -Vite is a powerful tool to generate static websites. While it provides all kinds of features, let's just go barebones with some good old vanilla JS. +ZK is a powerful technology. An app that reveals computational correctness but doesn't reveal some of its inputs is almost unbelievable, yet Noir makes it as easy as a single line of code. -To do this this, go back to the previous folder (`cd ..`) and create a new vite project by running `npm create vite` and choosing "Vanilla" and "Javascript". +:::tip -A wild `vite-project` directory should now appear in your root folder! Let's not waste any time and dive right in: +It's not just you. We also enjoy syntax highlighting. [Check out the Language Server](../tooling/language_server.md) -```bash -cd vite-project -``` +::: -### Setting Up Vite and Configuring the Project - -Before we proceed with any coding, let's get our environment tailored for Noir. We'll start by laying down the foundations with a `vite.config.js` file. This little piece of configuration is our secret sauce for making sure everything meshes well with the NoirJS libraries and other special setups we might need, like handling WebAssembly modules. Here’s how you get that going: - -#### Creating the vite.config.js - -In your freshly minted `vite-project` folder, create a new file named `vite.config.js` and open it in your code editor. Paste the following to set the stage: - -```javascript -import { defineConfig } from 'vite'; -import copy from 'rollup-plugin-copy'; -import fs from 'fs'; -import path from 'path'; - -const wasmContentTypePlugin = { - name: 'wasm-content-type-plugin', - configureServer(server) { - server.middlewares.use(async (req, res, next) => { - if (req.url.endsWith('.wasm')) { - res.setHeader('Content-Type', 'application/wasm'); - const newPath = req.url.replace('deps', 'dist'); - const targetPath = path.join(__dirname, newPath); - const wasmContent = fs.readFileSync(targetPath); - return res.end(wasmContent); - } - next(); - }); - }, -}; +All you need is a `main.nr` and a `Nargo.toml` file. You can follow the [noirup](../getting_started/noir_installation.md) installation and just run `noirup -v 1.0.0-beta.0`, or just create them by hand: -export default defineConfig(({ command }) => { - if (command === 'serve') { - return { - build: { - target: 'esnext', - rollupOptions: { - external: ['@aztec/bb.js'] - } - }, - optimizeDeps: { - esbuildOptions: { - target: 'esnext' - } - }, - plugins: [ - copy({ - targets: [{ src: 'node_modules/**/*.wasm', dest: 'node_modules/.vite/dist' }], - copySync: true, - hook: 'buildStart', - }), - command === 'serve' ? wasmContentTypePlugin : [], - ], - }; - } - - return {}; -}); +```bash +mkdir -p circuit/src +touch circuit/src/main.nr circuit/Nargo.toml ``` -#### Install Dependencies - -Now that our stage is set, install the necessary NoirJS packages along with our other dependencies: +To make our program interesting, let's give it a real use-case scenario: Bob wants to prove he is older than 18, without disclosing his age. Open `main.nr` and write: -```bash -npm install && npm install @noir-lang/backend_barretenberg@0.31.0 @noir-lang/noir_js@0.31.0 -npm install rollup-plugin-copy --save-dev +```rust +fn main(age: u8) { + assert(age >= 18); +} ``` -:::info - -At this point in the tutorial, your folder structure should look like this: +This program accepts a private input called age, and simply proves this number is higher than 18. But to run this code, we need to give the compiler a `Nargo.toml` with at least a name and a type: -```tree -. -└── circuit - └── ...etc... -└── vite-project <---- our working directory - └── ...etc... +```toml +[package] +name = "circuit" +type = "bin" ``` -::: +This is all that we need to get started with Noir. -#### Some cleanup +![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) -`npx create vite` is amazing but it creates a bunch of files we don't really need for our simple example. Actually, let's just delete everything except for `vite.config.js`, `index.html`, `main.js` and `package.json`. I feel lighter already. +## Setting up our app -![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) +Remember when apps only had one `html` and one `js` file? Well, that's enough for Noir webapps. Let's create them: -## HTML +```bash +touch index.html index.js +``` -Our app won't run like this, of course. We need some working HTML, at least. Let's open our broken-hearted `index.html` and replace everything with this code snippet: +And add something useful to our HTML file: ```html @@ -200,11 +105,11 @@ Our app won't run like this, of course. We need some working HTML, at least. Let - +

Noir app

- - + +

Logs

@@ -214,32 +119,26 @@ Our app won't run like this, of course. We need some working HTML, at least. Let ``` -It _could_ be a beautiful UI... Depending on which universe you live in. - -## Some good old vanilla Javascript - -Our love for Noir needs undivided attention, so let's just open `main.js` and delete everything (this is where the romantic scenery becomes a bit creepy). +It _could_ be a beautiful UI... Depending on which universe you live in. In any case, we're using some scary CSS to make two boxes that will show cool things on the screen. -Start by pasting in this boilerplate code: +As for the JS, real madmen could just `console.log` everything, but let's say we want to see things happening (the true initial purpose of JS... right?). Here's some boilerplate for that. Just paste it in `index.js`: ```js -function display(container, msg) { - const c = document.getElementById(container); - const p = document.createElement('p'); - p.textContent = msg; - c.appendChild(p); -} +const show = (id, content) => { + const container = document.getElementById(id); + container.appendChild(document.createTextNode(content)); + container.appendChild(document.createElement("br")); +}; -document.getElementById('submitGuess').addEventListener('click', async () => { - try { - // here's where love happens - } catch (err) { - display('logs', 'Oh 💔 Wrong guess'); - } +document.getElementById("submit").addEventListener("click", async () => { + try { + // noir goes here + } catch { + show("logs", "Oh 💔"); + } }); -``` -The display function doesn't do much. We're simply manipulating our website to see stuff happening. For example, if the proof fails, it will simply log a broken heart 😢 +``` :::info @@ -248,30 +147,56 @@ At this point in the tutorial, your folder structure should look like this: ```tree . └── circuit - └── ...same as above -└── vite-project - ├── vite.config.js - ├── main.js - ├── package.json - └── index.html + └── src + └── main.nr + Nargo.toml + index.js + package.json + index.html + ...etc ``` -You'll see other files and folders showing up (like `package-lock.json`, `node_modules`) but you shouldn't have to care about those. - ::: -## Some NoirJS +## Compile compile compile -We're starting with the good stuff now. If you've compiled the circuit as described above, you should have a `json` file we want to import at the very top of our `main.js` file: +Finally we're up for something cool. But before we can execute a Noir program, we need to compile it into ACIR: an abstract representation. Here's where `noir_wasm` comes in. -```ts -import circuit from '../circuit/target/circuit.json'; +`noir_wasm` expects a filesystem so it can resolve dependencies. While we could use the `public` folder, let's just import those using the nice `?url` syntax provided by vite. At the top of the file: + +```js +import { compile, createFileManager } from "@noir-lang/noir_wasm" + +import main from "./circuit/src/main.nr?url"; +import nargoToml from "./circuit/Nargo.toml?url"; ``` -[Noir is backend-agnostic](../index.mdx#whats-new-about-noir). We write Noir, but we also need a proving backend. That's why we need to import and instantiate the two dependencies we installed above: `BarretenbergBackend` and `Noir`. Let's import them right below: +Compiling on the browser is common enough that `createFileManager` already gives us a nice in-memory filesystem we can use. So all we need to compile is fetching these files, writing them to our filesystem, and compile. Add this function: ```js -import { BarretenbergBackend, BarretenbergVerifier as Verifier } from '@noir-lang/backend_barretenberg'; +export async function getCircuit() { + const fm = createFileManager("/"); + const { body } = await fetch(main); + const { body: nargoTomlBody } = await fetch(nargoToml); + + fm.writeFile("./src/main.nr", body); + fm.writeFile("./Nargo.toml", nargoTomlBody); + return await compile(fm); +} +``` + +:::tip + +As you can imagine, with `node` it's all conveniently easier since you get native access to `fs`... + +::: + +## Some more JS + +We're starting with the good stuff now. We want to execute our circuit to get the witness, and then feed that witness to Barretenberg. Luckily, both packages are quite easy to work with. Let's import them at the top of the file: + +```js +import { UltraHonkBackend } from '@aztec/bb.js'; import { Noir } from '@noir-lang/noir_js'; ``` @@ -279,88 +204,103 @@ And instantiate them inside our try-catch block: ```ts // try { -const backend = new BarretenbergBackend(circuit); -const noir = new Noir(circuit); +const { program } = await getCircuit(); +const noir = new Noir(program); +const backend = new UltraHonkBackend(program.bytecode); // } ``` -:::note +:::warning -For the remainder of the tutorial, everything will be happening inside the `try` block +WASMs are not always easy to work with. In our case, `vite` likes serving them with the wrong MIME type. There are different fixes but we found the easiest one is just YOLO instantiating the WASMs manually. Paste this at the top of the file, just below the other imports, and it will work just fine: + +```js +import initNoirC from "@noir-lang/noirc_abi"; +import initACVM from "@noir-lang/acvm_js"; +import acvm from "@noir-lang/acvm_js/web/acvm_js_bg.wasm?url"; +import noirc from "@noir-lang/noirc_abi/web/noirc_abi_wasm_bg.wasm?url"; +await Promise.all([initACVM(fetch(acvm)), initNoirC(fetch(noirc))]); +``` ::: -## Our app +## Executing and proving -Now for the app itself. We're capturing whatever is in the input when people press the submit button. Just add this: +Now for the app itself. We're capturing whatever is in the input when people press the submit button. Inside our `try` block, let's just grab that input and get its value. Noir will gladly execute it, and give us a witness: ```js -const x = parseInt(document.getElementById('guessInput').value); -const input = { x, y: 2 }; +const age = document.getElementById("age").value; +show("logs", "Generating witness... ⏳"); +const { witness } = await noir.execute({ age }); +show("logs", "Generated witness... ✅"); + ``` +:::note + +For the remainder of the tutorial, everything will be happening inside the `try` block + +::: + Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: ```js -await setup(); // let's squeeze our wasm inits here - -display('logs', 'Generating proof... ⌛'); -const { witness } = await noir.execute(input); +show("logs", "Generating proof... ⏳"); const proof = await backend.generateProof(witness); -display('logs', 'Generating proof... ✅'); -display('results', proof.proof); +show("logs", "Generated proof... ✅"); +show("results", proof.proof); ``` -You're probably eager to see stuff happening, so go and run your app now! +Our program is technically **done** . You're probably eager to see stuff happening! To serve this in a convenient way, we can use a bundler like `vite` by creating a `vite.config.js` file: + +```bash +touch vite.config.js +``` -From your terminal, run `npm run dev`. If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. +`vite` helps us with a little catch: `bb.js` in particular uses top-level awaits which aren't supported everywhere. So we can add this to the `vite.config.js` to make the bundler optimize them: -![Getting Started 0](@site/static/img/noir_getting_started_1.png) +```js +export default { optimizeDeps: { esbuildOptions: { target: "esnext" } } }; +``` + +This should be enough for vite. We don't even need to install it, just run: + +```bash +bunx vite +``` -Now, our circuit says `fn main(x: Field, y: pub Field)`. This means only the `y` value is public, and it's hardcoded above: `input = { x, y: 2 }`. In other words, you won't need to send your secret`x` to the verifier! +If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. -By inputting any number other than 2 in the input box and clicking "submit", you should get a valid proof. Otherwise the proof won't even generate correctly. By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. +![Noir Webapp UI](@site/static/img/tutorials/noirjs_webapp/webapp1.png) + +Now, our circuit requires a private input `fn main(age: u8)`, and fails if it is less than 18. Let's see if it works. Submit any number above 18 (as long as it fits in 8 bits) and you should get a valid proof. Otherwise the proof won't even generate correctly. + +By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. ## Verifying Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: ```js -display('logs', 'Verifying proof... ⌛'); +show('logs', 'Verifying proof... ⌛'); const isValid = await backend.verifyProof(proof); - -// or to cache and use the verification key: -// const verificationKey = await backend.getVerificationKey(); -// const verifier = new Verifier(); -// const isValid = await verifier.verifyProof(proof, verificationKey); - -if (isValid) display('logs', 'Verifying proof... ✅'); +show("logs", `Proof is ${isValid ? "valid" : "invalid"}... ✅`); ``` You have successfully generated a client-side Noir web app! ![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) -## Further Reading - -You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. - -You should also check out the more advanced examples in the [noir-examples repo](https://github.com/noir-lang/noir-examples), where you'll find reference usage for some cool apps. +## Next steps -## UltraHonk Backend +At this point, you have a working ZK app that works on the browser. Actually, it works on a mobile phone too! -Barretenberg has recently exposed a new UltraHonk backend. We can use UltraHonk in NoirJS after version 0.33.0. Everything will be the same as the tutorial above, except that the class we need to import will change: +If you want to continue learning by doing, here are some challenges for you: -```js -import { UltraHonkBackend, UltraHonkVerifier as Verifier } from '@noir-lang/backend_barretenberg'; -``` - -The backend will then be instantiated as such: - -```js -const backend = new UltraHonkBackend(circuit); -``` +- Install [nargo](https://noir-lang.org/docs/getting_started/noir_installation) and write [Noir tests](../tooling/testing) +- Change the circuit to accept a [public input](../noir/concepts/data_types/#private--public-types) as the cutoff age. It could be different depending on the purpose, for example! +- Enjoy Noir's Rust-like syntax and write a struct `Country` that implements a trait `MinAge` with a method `get_min_age`. Then, make a struct `Person` have an `u8` as its age and a country of type `Country`. You can pass a `person` in JS just like a JSON object `person: { age, country: { min_age: 18 }}` -Then all the commands to prove and verify your circuit will be same. +The world is your stage, just have fun with ZK! You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. -The only feature currently unsupported with UltraHonk are [recursive proofs](../explainers/explainer-recursion.md). +Check out other starters, tools, or just cool projects in the [awesome noir repository](https://github.com/noir-lang/awesome-noir). diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/cspell.json b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/explainers/cspell.json similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.33.0/explainers/cspell.json rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/explainers/cspell.json diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/explainers/explainer-oracle.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/explainers/explainer-oracle.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/explainers/explainer-oracle.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/explainers/explainer-oracle.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/explainers/explainer-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/explainers/explainer-recursion.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/explainers/explainer-recursion.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/explainers/explainer-recursion.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/explainer-writing-noir.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/explainers/explainer-writing-noir.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/explainers/explainer-writing-noir.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/explainers/explainer-writing-noir.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/noir_installation.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/noir_installation.md similarity index 98% rename from noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/noir_installation.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/noir_installation.md index a5c7e649278..05f036d4f6d 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/noir_installation.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/noir_installation.md @@ -93,7 +93,7 @@ step 2: Follow the [Noirup instructions](#installing-noirup). ## Setting up shell completions -Once `nargo` is installed, you can [set up shell completions for it](setting_up_shell_completions). +Once `nargo` is installed, you can [set up shell completions for it](setting_up_shell_completions.md). ## Uninstalling Nargo diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/getting_started/project_breakdown.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/project_breakdown.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/getting_started/project_breakdown.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/project_breakdown.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/quick_start.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/quick_start.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/quick_start.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/quick_start.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/setting_up_shell_completions.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/setting_up_shell_completions.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.38.0/getting_started/setting_up_shell_completions.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/getting_started/setting_up_shell_completions.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/_category_.json similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/_category_.json rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/_category_.json diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/debugger/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/debugger/_category_.json similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/debugger/_category_.json rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/debugger/_category_.json diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/debugger/debugging_with_the_repl.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/debugger/debugging_with_the_repl.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/debugger/debugging_with_the_repl.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/debugger/debugging_with_the_repl.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/debugger/debugging_with_vs_code.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/debugger/debugging_with_vs_code.md similarity index 96% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/debugger/debugging_with_vs_code.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/debugger/debugging_with_vs_code.md index a5858c1a5eb..8bda93324f5 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/debugger/debugging_with_vs_code.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/debugger/debugging_with_vs_code.md @@ -1,7 +1,7 @@ --- title: Using the VS Code Debugger description: - Step by step guide on how to debug your Noir circuits with the VS Code Debugger configuration and features. + Step-by-step guide on how to debug your Noir circuits with the VS Code Debugger configuration and features. keywords: [ Nargo, @@ -65,4 +65,4 @@ We just need to click the to the right of the line number 18. Once the breakpoin Now we are debugging the `keccak256` function, notice the _Call Stack pane_ at the lower right. This lets us inspect the current call stack of our process. -That covers most of the current debugger functionalities. Check out [the reference](../../reference/debugger/debugger_vscode.md) for more details on how to configure the debugger. \ No newline at end of file +That covers most of the current debugger functionalities. Check out [the reference](../../reference/debugger/debugger_vscode.md) for more details on how to configure the debugger. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/how-to-oracles.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/how-to-oracles.md similarity index 99% rename from noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/how-to-oracles.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/how-to-oracles.md index 4763b7788d6..0bb8743e361 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/how-to-oracles.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/how-to-oracles.md @@ -30,7 +30,7 @@ This guide has 3 major steps: An oracle is defined in a Noir program by defining two methods: -- An unconstrained method - This tells the compiler that it is executing an [unconstrained functions](../noir/concepts//unconstrained.md). +- An unconstrained method - This tells the compiler that it is executing an [unconstrained function](../noir/concepts//unconstrained.md). - A decorated oracle method - This tells the compiler that this method is an RPC call. An example of an oracle that returns a `Field` would be: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/how-to-recursion.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/how-to-recursion.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.39.0/how_to/how-to-recursion.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/how-to-recursion.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-solidity-verifier.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/how-to-solidity-verifier.mdx similarity index 64% rename from noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-solidity-verifier.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/how-to-solidity-verifier.mdx index 2cc0f8e57ce..da36b60920d 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/how_to/how-to-solidity-verifier.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/how-to-solidity-verifier.mdx @@ -20,24 +20,37 @@ sidebar_position: 0 pagination_next: tutorials/noirjs_app --- -Noir has the ability to generate a verifier contract in Solidity, which can be deployed in many EVM-compatible blockchains such as Ethereum. +import Tabs from '@theme/Tabs'; +import TabItem from '@theme/TabItem'; + +Noir is universal. The witness and the compiled program can be fed into a proving backend such as Aztec's [Barretenberg](https://github.com/AztecProtocol/aztec-packages/tree/master/barretenberg), which can then generate a verifier contract for deployment on blockchains. This allows for a powerful feature set, as one can make use of the conciseness and the privacy provided by Noir in an immutable ledger. Applications can range from simple P2P guessing games, to complex private DeFi interactions. -This guide shows you how to generate a Solidity Verifier and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: +Although not strictly in the domain of Noir itself, this guide shows how to generate a Solidity Verifier with Barretenberg and deploy it on the [Remix IDE](https://remix.ethereum.org/). It is assumed that: +- You will be using Barretenberg as your proving backend +- You will be using an EVM blockchain to verify your proof - You are comfortable with the Solidity programming language and understand how contracts are deployed on the Ethereum network - You have Noir installed and you have a Noir program. If you don't, [get started](../getting_started/quick_start.md) with Nargo and the example Hello Noir circuit - You are comfortable navigating RemixIDE. If you aren't or you need a refresher, you can find some video tutorials [here](https://www.youtube.com/channel/UCjTUPyFEr2xDGN6Cg8nKDaA) that could help you. ## Rundown -Generating a Solidity Verifier contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: +Generating a Solidity Verifier with Barretenberg contract is actually a one-command process. However, compiling it and deploying it can have some caveats. Here's the rundown of this guide: 1. How to generate a solidity smart contract 2. How to compile the smart contract in the RemixIDE 3. How to deploy it to a testnet +:::info[Which proving system to use?] + +Barretenberg currently provides two provers: `UltraPlonk` and `UltraHonk`. In a nutshell, `UltraHonk` is faster and uses less RAM, but its verifier contract is much more expensive. `UltraPlonk` is optimized for on-chain verification, but proving is more expensive. + +In any case, we provide instructions for both. Choose your poison ☠️ + +::: + ## Step 1 - Generate a contract This is by far the most straightforward step. Just run: @@ -46,25 +59,31 @@ This is by far the most straightforward step. Just run: nargo compile ``` -This will compile your source code into a Noir build artifact to be stored in the `./target` directory, you can then generate the smart contract using the commands: +This will compile your source code into a Noir build artifact to be stored in the `./target` directory. From here on, it's Barretenberg's work. You can generate the smart contract using the commands: + + + + +```sh +bb write_vk_ultra_keccak_honk -b ./target/.json +bb contract_ultra_honk +``` + + + ```sh -# Here we pass the path to the newly generated Noir artifact. bb write_vk -b ./target/.json bb contract ``` -replacing `` with the name of your Noir project. A new `contract` folder would then be generated in your project directory, containing the Solidity -file `contract.sol`. It can be deployed to any EVM blockchain acting as a verifier smart contract. + + -You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/quick_start.md#proving-backend). +replacing `` with the name of your Noir project. A `Verifier.sol` contract is now in the target folder and can be deployed to any EVM blockchain acting as a verifier smart contract. -:::info - -It is possible to generate verifier contracts of Noir programs for other smart contract platforms as long as the proving backend supplies an implementation. +You can find more information about `bb` and the default Noir proving backend on [this page](../getting_started/quick_start.md#proving-backend). -Barretenberg, the default proving backend for Nargo, supports generation of verifier contracts, for the time being these are only in Solidity. -::: ## Step 2 - Compiling @@ -85,17 +104,12 @@ To compile our the verifier, we can navigate to the compilation tab: ![Compilation Tab](@site/static/img/how-tos/solidity_verifier_2.png) -Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely generate a "Stack too deep" error: +Remix should automatically match a suitable compiler version. However, hitting the "Compile" button will most likely tell you the contract is too big to deploy on mainnet, or complain about a stack too deep: -![Stack too deep](@site/static/img/how-tos/solidity_verifier_3.png) +![Contract code too big](@site/static/img/how-tos/solidity_verifier_6.png) +![Stack too deep](@site/static/img/how-tos/solidity_verifier_8.png) -This is due to the verify function needing to put many variables on the stack, but enabling the optimizer resolves the issue. To do this, let's open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. - -:::info - -This time we will see a warning about an unused function parameter. This is expected, as the `verify` function doesn't use the `_proof` parameter inside a solidity block, it is loaded from calldata and used in assembly. - -::: +To avoid this, you can just use some optimization. Open the "Advanced Configurations" tab and enable optimization. The default 200 runs will suffice. ![Compilation success](@site/static/img/how-tos/solidity_verifier_4.png) @@ -103,54 +117,81 @@ This time we will see a warning about an unused function parameter. This is expe At this point we should have a compiled contract ready to deploy. If we navigate to the deploy section in Remix, we will see many different environments we can deploy to. The steps to deploy on each environment would be out-of-scope for this guide, so we will just use the default Remix VM. -Looking closely, we will notice that our "Solidity Verifier" is actually three contracts working together: - -- An `UltraVerificationKey` library which simply stores the verification key for our circuit. -- An abstract contract `BaseUltraVerifier` containing most of the verifying logic. -- A main `UltraVerifier` contract that inherits from the Base and uses the Key contract. - -Remix will take care of the dependencies for us so we can simply deploy the UltraVerifier contract by selecting it and hitting "deploy": +Looking closely, we will notice that our "Solidity Verifier" is composed on multiple contracts working together. Remix will take care of the dependencies for us so we can simply deploy the Verifier contract by selecting it and hitting "deploy": -![Deploying UltraVerifier](@site/static/img/how-tos/solidity_verifier_5.png) + + -A contract will show up in the "Deployed Contracts" section, where we can retrieve the Verification Key Hash. This is particularly useful for double-checking that the deployer contract is the correct one. +![Deploying HonkVerifier](@site/static/img/how-tos/solidity_verifier_7.png) -:::note + + -Why "UltraVerifier"? +![Deploying PlonkVerifier](@site/static/img/how-tos/solidity_verifier_9.png) -To be precise, the Noir compiler (`nargo`) doesn't generate the verifier contract directly. It compiles the Noir code into an intermediate language (ACIR), which is then executed by the backend. So it is the backend that returns the verifier smart contract, not Noir. + + -In this case, the Barretenberg Backend uses the UltraPlonk proving system, hence the "UltraVerifier" name. - -::: +A contract will show up in the "Deployed Contracts" section. ## Step 4 - Verifying -To verify a proof using the Solidity verifier contract, we call the `verify` function in this extended contract: +To verify a proof using the Solidity verifier contract, we call the `verify` function: ```solidity function verify(bytes calldata _proof, bytes32[] calldata _publicInputs) external view returns (bool) ``` -When using the default example in the [Hello Noir](../getting_started/quick_start.md) guide, the easiest way to confirm that the verifier contract is doing its job is by calling the `verify` function via remix with the required parameters. Note that the public inputs must be passed in separately to the rest of the proof so we must split the proof as returned from `bb`. +First generate a proof with `bb`. We need a `Prover.toml` file for our inputs. Run: -First generate a proof with `bb` at the location `./proof` using the steps in [get started](../getting_started/quick_start.md), this proof is in a binary format but we want to convert it into a hex string to pass into Remix, this can be done with the +```bash +nargo check +``` + +This will generate a `Prover.toml` you can fill with the values you want to prove. We can now execute the circuit with `nargo` and then use the proving backend to prove: + + + + +```bash +nargo execute +bb prove_ultra_keccak_honk -b ./target/.json -w ./target/ -o ./target/proof +``` + +:::tip[Public inputs] +Barretenberg attaches the public inputs to the proof, which in this case it's not very useful. If you're up for some JS, `bb.js` has [a method for it](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/ts/src/proof/index.ts), but in the CLI you can use this ugly snippet: + +```bash +cat ./target/proof | od -An -v -t x1 | tr -d $' \n' | sed 's/^.\{8\}//' | (read hex; echo "${hex:0:192}${hex:256}") +``` + +Beautiful. This assumes a circuit with one public input (32 bytes, for Barretenberg). For more inputs, you can just increment `hex:256` with 32 more bytes for each public input. + +::: + + + ```bash -# This value must be changed to match the number of public inputs (including return values!) in your program. -NUM_PUBLIC_INPUTS=1 -PUBLIC_INPUT_BYTES=32*NUM_PUBLIC_INPUTS -HEX_PUBLIC_INPUTS=$(head -c $PUBLIC_INPUT_BYTES ./proof | od -An -v -t x1 | tr -d $' \n') -HEX_PROOF=$(tail -c +$(($PUBLIC_INPUT_BYTES + 1)) ./proof | od -An -v -t x1 | tr -d $' \n') +nargo execute +bb prove -b ./target/.json -w ./target/ -o ./target/proof +``` + -echo "Public inputs:" -echo $HEX_PUBLIC_INPUTS +:::tip[Public inputs] +Barretenberg attaches the public inputs to the proof, which in this case it's not very useful. If you're up for some JS, `bb.js` has [a method for it](https://github.com/AztecProtocol/aztec-packages/blob/master/barretenberg/ts/src/proof/index.ts), but in the CLI you can use this ugly snippet: -echo "Proof:" -echo "0x$HEX_PROOF" +```bash +tail -c +33 ./target/proof | od -An -v -t x1 | tr -d $' \n' ``` +Beautiful. This assumes a circuit with one public input (32 bytes, for Barretenberg). For more inputs, you can just add 32 more bytes for each public input to the `tail` command. + +::: + + + + Remix expects that the public inputs will be split into an array of `bytes32` values so `HEX_PUBLIC_INPUTS` needs to be split up into 32 byte chunks which are prefixed with `0x` accordingly. A programmatic example of how the `verify` function is called can be seen in the example zk voting application [here](https://github.com/noir-lang/noir-examples/blob/33e598c257e2402ea3a6b68dd4c5ad492bce1b0a/foundry-voting/src/zkVote.sol#L35): @@ -188,7 +229,7 @@ the `verify` function will expect the public inputs array (second function param Passing only two inputs will result in an error such as `PUBLIC_INPUT_COUNT_INVALID(3, 2)`. -In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return`. +In this case, the inputs parameter to `verify` would be an array ordered as `[pubkey_x, pubkey_y, return]`. ::: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/merkle-proof.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/merkle-proof.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/merkle-proof.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/merkle-proof.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/using-devcontainers.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/using-devcontainers.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/how_to/using-devcontainers.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/how_to/using-devcontainers.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/index.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/index.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/index.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/index.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/migration_notes.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/migration_notes.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/migration_notes.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/migration_notes.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/_category_.json similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/_category_.json rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/_category_.json diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/assert.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/assert.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/assert.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/assert.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/comments.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/comments.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/comments.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/comments.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/comptime.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/comptime.md similarity index 76% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/comptime.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/comptime.md index 2ceb030c7e1..353d6dbc681 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/comptime.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/comptime.md @@ -41,7 +41,7 @@ Note that while in a `comptime` context, any runtime variables _local to the cur Evaluation rules of `comptime` follows the normal unconstrained evaluation rules for other Noir code. There are a few things to note though: - Certain built-in functions may not be available, although more may be added over time. -- Evaluation order of global items is currently unspecified. For example, given the following two functions we can't guarantee +- Evaluation order of `comptime {}` blocks within global items is currently unspecified. For example, given the following two functions we can't guarantee which `println` will execute first. The ordering of the two printouts will be arbitrary, but should be stable across multiple compilations with the same `nargo` version as long as the program is also unchanged. ```rust @@ -56,11 +56,14 @@ fn two() { - Since evaluation order is unspecified, care should be taken when using mutable globals so that they do not rely on a particular ordering. For example, using globals to generate unique ids should be fine but relying on certain ids always being produced (especially after edits to the program) should be avoided. -- Although most ordering of globals is unspecified, two are: +- Although the ordering of comptime code is usually unspecified, there are cases where it is: - Dependencies of a crate will always be evaluated before the dependent crate. - - Any annotations on a function will be run before the function itself is resolved. This is to allow the annotation to modify the function if necessary. Note that if the + - Any attributes on a function will be run before the function body is resolved. This is to allow the attribute to modify the function if necessary. Note that if the function itself was called at compile-time previously, it will already be resolved and cannot be modified. To prevent accidentally calling functions you wish to modify - at compile-time, it may be helpful to sort your `comptime` annotation functions into a different crate along with any dependencies they require. + at compile-time, it may be helpful to sort your `comptime` annotation functions into a different submodule crate along with any dependencies they require. + - Unlike raw `comptime {}` blocks, attributes on top-level items in the program do have a set evaluation order. Attributes within a module are evaluated top-down, and attributes + in different modules are evaluated submodule-first. Sibling modules to the same parent module are evaluated in order of the module declarations (`mod foo; mod bar;`) in their + parent module. ### Lowering @@ -89,7 +92,7 @@ fn main() { } ``` -Not all types of values can be lowered. For example, `Type`s and `TypeDefinition`s (among other types) cannot be lowered at all. +Not all types of values can be lowered. For example, references, `Type`s, and `TypeDefinition`s (among other types) cannot be lowered at all. ```rust fn main() { @@ -100,6 +103,19 @@ fn main() { comptime fn get_type() -> Type { ... } ``` +Values of certain types may also change type when they are lowered. For example, a comptime format string will already be +formatted, and thus lowers into a runtime string instead: + +```rust +fn main() { + let foo = comptime { + let i = 2; + f"i = {i}" + }; + assert_eq(foo, "i = 2"); +} +``` + --- ## (Quasi) Quote @@ -143,6 +159,21 @@ comptime fn quote_one() -> Quoted { For those familiar with quoting from other languages (primarily lisps), Noir's `quote` is actually a _quasiquote_. This means we can escape the quoting by using the unquote operator to splice values in the middle of quoted code. +In addition to curly braces, you can also use square braces for the quote operator: + +```rust +comptime { + let q1 = quote { 1 }; + let q2 = quote [ 2 ]; + assert_eq(q1, q2); + + // Square braces can be used to quote mismatched curly braces if needed + let _ = quote[}]; +} +``` + +--- + ## Unquote The unquote operator `$` is usable within a `quote` expression. @@ -171,7 +202,7 @@ If it is an expression (even a parenthesized one), it will do nothing. Most like Unquoting can also be avoided by escaping the `$` with a backslash: -``` +```rust comptime { let x = quote { 1 + 2 }; @@ -180,26 +211,83 @@ comptime { } ``` +### Combining Tokens + +Note that `Quoted` is internally a series of separate tokens, and that all unquoting does is combine these token vectors. +This means that code which appears to append like a string actually appends like a vector internally: + +```rust +comptime { + let x = 3; + let q = quote { foo$x }; // This is [foo, 3], not [foo3] + + // Spaces are ignored in general, they're never part of a token + assert_eq(q, quote { foo 3 }); +} +``` + +If you do want string semantics, you can use format strings then convert back to a `Quoted` value with `.quoted_contents()`. +Note that formatting a quoted value with multiple tokens will always insert a space between each token. If this is +undesired, you'll need to only operate on quoted values containing a single token. To do this, you can iterate +over each token of a larger quoted value with `.tokens()`: + +```rust title="concatenate-example" showLineNumbers +comptime fn concatenate(q1: Quoted, q2: Quoted) -> Quoted { + assert(q1.tokens().len() <= 1); + assert(q2.tokens().len() <= 1); + + f"{q1}{q2}".quoted_contents() + } + + // The CtString type is also useful for a compile-time string of unbounded size + // so that you can append to it in a loop. + comptime fn double_spaced(q: Quoted) -> CtString { + let mut result = "".as_ctstring(); + + for token in q.tokens() { + if result != "".as_ctstring() { + result = result.append_str(" "); + } + result = result.append_fmtstr(f"{token}"); + } + + result + } + + #[test] + fn concatenate_test() { + comptime { + let result = concatenate(quote {foo}, quote {bar}); + assert_eq(result, quote {foobar}); + + let result = double_spaced(quote {foo bar 3}).as_quoted_str!(); + assert_eq(result, "foo bar 3"); + } + } +``` +> Source code: noir_stdlib/src/meta/mod.nr#L255-L288 + + --- -## Annotations +## Attributes -Annotations provide a way to run a `comptime` function on an item in the program. -When you use an annotation, the function with the same name will be called with that item as an argument: +Attributes provide a way to run a `comptime` function on an item in the program. +When you use an attribute, the function with the same name will be called with that item as an argument: ```rust -#[my_struct_annotation] +#[my_struct_attribute] struct Foo {} -comptime fn my_struct_annotation(s: StructDefinition) { - println("Called my_struct_annotation!"); +comptime fn my_struct_attribute(s: StructDefinition) { + println("Called my_struct_attribute!"); } -#[my_function_annotation] +#[my_function_attribute] fn foo() {} -comptime fn my_function_annotation(f: FunctionDefinition) { - println("Called my_function_annotation!"); +comptime fn my_function_attribute(f: FunctionDefinition) { + println("Called my_function_attribute!"); } ``` @@ -236,8 +324,8 @@ trait FieldCount { ### Calling annotations with additional arguments -Arguments may optionally be given to annotations. -When this is done, these additional arguments are passed to the annotation function after the item argument. +Arguments may optionally be given to attributes. +When this is done, these additional arguments are passed to the attribute function after the item argument. ```rust title="annotation-arguments-example" showLineNumbers #[assert_field_is_type(quote { i32 }.as_type())] @@ -254,7 +342,7 @@ When this is done, these additional arguments are passed to the annotation funct > Source code: noir_stdlib/src/meta/mod.nr#L166-L177 -We can also take any number of arguments by adding the `varargs` annotation: +We can also take any number of arguments by adding the `varargs` attribute: ```rust title="annotation-varargs-example" showLineNumbers #[assert_three_args(1, 2, 3)] @@ -270,6 +358,38 @@ We can also take any number of arguments by adding the `varargs` annotation: > Source code: noir_stdlib/src/meta/mod.nr#L179-L189 +### Attribute Evaluation Order + +Unlike the evaluation order of stray `comptime {}` blocks within functions, attributes have a well-defined evaluation +order. Within a module, attributes are evaluated top to bottom. Between modules, attributes in child modules are evaluated +first. Attributes in sibling modules are resolved following the `mod foo; mod bar;` declaration order within their parent +modules. + +```rust +mod foo; // attributes in foo are run first +mod bar; // followed by attributes in bar + +// followed by any attributes in the parent module +#[derive(Eq)] +struct Baz {} +``` + +Note that because of this evaluation order, you may get an error trying to derive a trait for a struct whose fields +have not yet had the trait derived already: + +```rust +// Error! `Bar` field of `Foo` does not (yet) implement Eq! +#[derive(Eq)] +struct Foo { + bar: Bar +} + +#[derive(Eq)] +struct Bar {} +``` + +In this case, the issue can be resolved by rearranging the structs. + --- ## Comptime API @@ -358,7 +478,7 @@ pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted let mut result = quote {}; for trait_to_derive in traits { - let handler = unsafe { HANDLERS.get(trait_to_derive) }; + let handler = HANDLERS.get(trait_to_derive); assert(handler.is_some(), f"No derive function registered for `{trait_to_derive}`"); let trait_impl = handler.unwrap()(s); diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/control_flow.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/control_flow.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/control_flow.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/control_flow.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_bus.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_bus.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_bus.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_bus.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/_category_.json similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/getting_started/_category_.json rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/_category_.json diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/arrays.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/arrays.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/arrays.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/arrays.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/booleans.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/booleans.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/booleans.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/booleans.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/fields.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/fields.md similarity index 92% rename from noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/fields.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/fields.md index b9b56f7ecc3..6739d6500dd 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.38.0/noir/concepts/data_types/fields.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/fields.md @@ -56,7 +56,7 @@ fn test_to_le_bits() { assert_eq(bits, [0, 1, 0, 0, 0, 0, 0, 0]); } ``` -> Source code: noir_stdlib/src/field/mod.nr#L276-L282 +> Source code: noir_stdlib/src/field/mod.nr#L277-L283 @@ -79,7 +79,7 @@ fn test_to_be_bits() { assert_eq(bits, [0, 0, 0, 0, 0, 0, 1, 0]); } ``` -> Source code: noir_stdlib/src/field/mod.nr#L267-L273 +> Source code: noir_stdlib/src/field/mod.nr#L268-L274 @@ -103,7 +103,7 @@ fn test_to_le_bytes() { assert_eq(Field::from_le_bytes::<8>(bytes), field); } ``` -> Source code: noir_stdlib/src/field/mod.nr#L295-L302 +> Source code: noir_stdlib/src/field/mod.nr#L296-L303 ### to_be_bytes @@ -126,7 +126,7 @@ fn test_to_be_bytes() { assert_eq(Field::from_be_bytes::<8>(bytes), field); } ``` -> Source code: noir_stdlib/src/field/mod.nr#L285-L292 +> Source code: noir_stdlib/src/field/mod.nr#L286-L293 @@ -157,7 +157,7 @@ fn test_to_le_radix() { assert_eq(Field::from_le_bytes::<8>(bytes), field); } ``` -> Source code: noir_stdlib/src/field/mod.nr#L315-L322 +> Source code: noir_stdlib/src/field/mod.nr#L316-L323 @@ -187,7 +187,7 @@ fn test_to_be_radix() { assert_eq(Field::from_be_bytes::<8>(bytes), field); } ``` -> Source code: noir_stdlib/src/field/mod.nr#L305-L312 +> Source code: noir_stdlib/src/field/mod.nr#L306-L313 @@ -224,7 +224,7 @@ example: ```rust fn main() { let field = 2 - field.assert_max_bit_size(32); + field.assert_max_bit_size::<32>(); } ``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/function_types.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/function_types.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/function_types.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/function_types.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/index.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/index.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/index.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/index.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/integers.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/integers.md similarity index 93% rename from noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/integers.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/integers.md index a1d59bf3166..41a823646dd 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/concepts/data_types/integers.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/integers.md @@ -47,6 +47,17 @@ fn main() { The bit size determines the maximum and minimum range of value the integer type can store. For example, an `i8` variable can store a value in the range of -128 to 127 (i.e. $\\-2^{7}\\$ to $\\2^{7}-1\\$). + +```rust +fn main(x: i16, y: i16) { + // modulo + let c = x % y; + let c = x % -13; +} +``` + +Modulo operation is defined for negative integers thanks to integer division, so that the equality `x = (x/y)*y + (x%y)` holds. + ## 128 bits Unsigned Integers The built-in structure `U128` allows you to use 128-bit unsigned integers almost like a native integer type. However, there are some differences to keep in mind: @@ -68,7 +79,7 @@ fn main() { You can construct a U128 from its limbs: ```rust fn main(x: u64, y: u64) { - let x = U128::from_u64s_be(x,y); + let z = U128::from_u64s_be(x,y); assert(z.hi == x as Field); assert(z.lo == y as Field); } diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/references.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/references.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/references.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/references.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/slices.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/slices.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/data_types/slices.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/slices.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/strings.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/strings.md similarity index 74% rename from noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/strings.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/strings.md index 1fdee42425e..b2257e8bdbb 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/data_types/strings.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/strings.md @@ -77,3 +77,38 @@ let s = r#"Simon says "hello world""#; // Any number of hashes may be used (>= 1) as long as the string also terminates with the same number of hashes let s = r#####"One "#, Two "##, Three "###, Four "####, Five will end the string."#####; ``` + +## Format strings + +A format string begins with the letter `f` and allows inserting the value of local and global variables in it. + +Example: + +```rust +let four = 2 + 2; +let s = f"Two plus two is: {four}"; +println(s); +``` + +The output of the above program is: + +```text +Two plus two is: 4 +``` + +To insert the value of a local or global variable, put it inside `{...}` in the string. + +If you need to write the `{` or `}` characters, use `{{` and `}}` respectively: + +```rust +let four = 2 + 2; + +// Prints "This is not expanded: {four}" +println(f"This is not expanded: {{four}}"); +``` + +More complex expressions are not allowed inside `{...}`: + +```rust +let s = f"Two plus two is: {2 + 2}" // Error: invalid format string +``` \ No newline at end of file diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/structs.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/structs.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/data_types/structs.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/structs.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/tuples.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/tuples.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/tuples.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/data_types/tuples.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/functions.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/functions.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/functions.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/functions.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/generics.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/generics.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.35.0/noir/concepts/generics.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/generics.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/globals.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/globals.md similarity index 85% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/globals.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/globals.md index 6b8314399a2..c64b6c53746 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/concepts/globals.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/globals.md @@ -10,12 +10,12 @@ sidebar_position: 8 ## Globals -Noir supports global variables. The global's type can be inferred by the compiler entirely: +Noir supports global variables. The global's type must be specified by the user: ```rust -global N = 5; // Same as `global N: Field = 5` +global N: Field = 5; -global TUPLE = (3, 2); +global TUPLE: (Field, Field) = (3, 2); fn main() { assert(N == 5); @@ -28,7 +28,7 @@ fn main() { Globals can be defined as any expression, so long as they don't depend on themselves - otherwise there would be a dependency cycle! For example: ```rust -global T = foo(T); // dependency error +global T: u32 = foo(T); // dependency error ``` ::: @@ -47,7 +47,7 @@ fn main(y : [Field; N]) { A global from another module can be imported or referenced externally like any other name: ```rust -global N = 20; +global N: Field = 20; fn main() { assert(my_submodule::N != N); @@ -62,7 +62,7 @@ When a global is used, Noir replaces the name with its definition on each occurr This means globals defined using function calls will repeat the call each time they're used: ```rust -global RESULT = foo(); +global RESULT: [Field; 100] = foo(); fn foo() -> [Field; 100] { ... } ``` @@ -78,5 +78,5 @@ to make the global public or `pub(crate)` to make it public to just its crate: ```rust // This global is now public -pub global N = 5; -``` \ No newline at end of file +pub global N: u32 = 5; +``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/lambdas.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/lambdas.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/lambdas.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/lambdas.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/mutability.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/mutability.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/mutability.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/mutability.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/ops.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/ops.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/ops.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/ops.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/oracles.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/oracles.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/oracles.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/oracles.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/shadowing.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/shadowing.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/shadowing.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/shadowing.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/traits.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/traits.md similarity index 87% rename from noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/traits.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/traits.md index 9da00a77587..b6c0a886eb0 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/concepts/traits.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/traits.md @@ -490,12 +490,95 @@ trait CompSciStudent: Programmer + Student { } ``` +### Trait Aliases + +Similar to the proposed Rust feature for [trait aliases](https://github.com/rust-lang/rust/blob/4d215e2426d52ca8d1af166d5f6b5e172afbff67/src/doc/unstable-book/src/language-features/trait-alias.md), +Noir supports aliasing one or more traits and using those aliases wherever +traits would normally be used. + +```rust +trait Foo { + fn foo(self) -> Self; +} + +trait Bar { + fn bar(self) -> Self; +} + +// Equivalent to: +// trait Baz: Foo + Bar {} +// +// impl Baz for T where T: Foo + Bar {} +trait Baz = Foo + Bar; + +// We can use `Baz` to refer to `Foo + Bar` +fn baz(x: T) -> T where T: Baz { + x.foo().bar() +} +``` + +#### Generic Trait Aliases + +Trait aliases can also be generic by placing the generic arguments after the +trait name. These generics are in scope of every item within the trait alias. + +```rust +trait Foo { + fn foo(self) -> Self; +} + +trait Bar { + fn bar(self) -> T; +} + +// Equivalent to: +// trait Baz: Foo + Bar {} +// +// impl Baz for U where U: Foo + Bar {} +trait Baz = Foo + Bar; +``` + +#### Trait Alias Where Clauses + +Trait aliases support where clauses to add trait constraints to any of their +generic arguments, e.g. ensuring `T: Baz` for a trait alias `Qux`. + +```rust +trait Foo { + fn foo(self) -> Self; +} + +trait Bar { + fn bar(self) -> T; +} + +trait Baz { + fn baz(self) -> bool; +} + +// Equivalent to: +// trait Qux: Foo + Bar where T: Baz {} +// +// impl Qux for U where +// U: Foo + Bar, +// T: Baz, +// {} +trait Qux = Foo + Bar where T: Baz; +``` + +Note that while trait aliases support where clauses, +the equivalent traits can fail due to [#6467](https://github.com/noir-lang/noir/issues/6467) + ### Visibility -By default, like functions, traits are private to the module they exist in. You can use `pub` -to make the trait public or `pub(crate)` to make it public to just its crate: +By default, like functions, traits and trait aliases are private to the module +they exist in. You can use `pub` to make the trait public or `pub(crate)` to make +it public to just its crate: ```rust // This trait is now public pub trait Trait {} -``` \ No newline at end of file + +// This trait alias is now public +pub trait Baz = Foo + Bar; +``` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/unconstrained.md similarity index 95% rename from noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/unconstrained.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/unconstrained.md index b5221b8d2dd..8e07d719f4b 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/concepts/unconstrained.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/concepts/unconstrained.md @@ -67,6 +67,7 @@ We can then run `u72_to_u8` as unconstrained brillig code in order to calculate ```rust fn main(num: u72) -> pub [u8; 8] { let out = unsafe { + //@safety: 'out' is properly constrained below in 'assert(num == reconstructed_num);' u72_to_u8(num) }; @@ -96,6 +97,7 @@ This ends up taking off another ~250 gates from our circuit! We've ended up with Note that in order to invoke unconstrained functions we need to wrap them in an `unsafe` block, to make it clear that the call is unconstrained. +Furthermore, a warning is emitted unless the `unsafe` block starts with a `//@safety: ...` comment explaining why it is fine to call the unconstrained function. Generally we want to use brillig whenever there's something that's easy to verify but hard to compute within the circuit. For example, if you wanted to calculate a square root of a number it'll be a much better idea to calculate this in brillig and then assert that if you square the result you get back your number. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/_category_.json similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/_category_.json rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/_category_.json diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/crates_and_packages.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/crates_and_packages.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/crates_and_packages.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/crates_and_packages.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/dependencies.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/dependencies.md similarity index 94% rename from noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/dependencies.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/dependencies.md index 24e02de08fe..22186b22598 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.33.0/noir/modules_packages_crates/dependencies.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/dependencies.md @@ -81,12 +81,10 @@ use std::hash::sha256; use std::scalar_mul::fixed_base_embedded_curve; ``` -Lastly, as demonstrated in the -[elliptic curve example](../standard_library/cryptographic_primitives/ec_primitives.md#examples), you -can import multiple items in the same line by enclosing them in curly braces: +Lastly, You can import multiple items in the same line by enclosing them in curly braces: ```rust -use std::ec::tecurve::affine::{Curve, Point}; +use std::hash::{keccak256, sha256}; ``` We don't have a way to consume libraries from inside a [workspace](./workspaces.md) as external dependencies right now. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/modules.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/modules.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/modules_packages_crates/modules.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/modules.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/workspaces.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/workspaces.md similarity index 73% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/workspaces.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/workspaces.md index 513497f12bf..2fbf10aec52 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/modules_packages_crates/workspaces.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/modules_packages_crates/workspaces.md @@ -33,10 +33,14 @@ members = ["crates/a", "crates/b"] default-member = "crates/a" ``` -`members` indicates which packages are included in the workspace. As such, all member packages of a workspace will be processed when the `--workspace` flag is used with various commands or if a `default-member` is not specified. +`members` indicates which packages are included in the workspace. As such, all member packages of a workspace will be processed when the `--workspace` flag is used with various commands or if a `default-member` is not specified. The `--package` option can be used to limit +the scope of some commands to a specific member of the workspace; otherwise these commands run on the package nearest on the path to the +current directory where `nargo` was invoked. `default-member` indicates which package various commands process by default. Libraries can be defined in a workspace. Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. Inside a workspace, these are consumed as `{ path = "../to_lib" }` dependencies in Nargo.toml. + +Please note that nesting regular packages is not supported: certain commands work on the workspace level and will use the topmost Nargo.toml file they can find on the path; unless this is a workspace configuration with `members`, the command might run on some unintended package. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/_category_.json similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/_category_.json rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/_category_.json diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/bigint.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/bigint.md similarity index 96% rename from noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/bigint.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/bigint.md index 05c3011634f..2743739e8b1 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/bigint.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/bigint.md @@ -58,7 +58,7 @@ fn big_int_example(x: u8, y: u8) { println(d[0]); } ``` -> Source code: test_programs/execution_success/bigint/src/main.nr#L74-L82 +> Source code: test_programs/execution_success/bigint/src/main.nr#L75-L83 ## Methods diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/black_box_fns.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/black_box_fns.md similarity index 96% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/black_box_fns.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/black_box_fns.md index d6079ab182c..e9392b20a92 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/black_box_fns.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/black_box_fns.md @@ -14,7 +14,6 @@ Here is a list of the current black box functions: - [AES128](./cryptographic_primitives/ciphers.mdx#aes128) - [SHA256](./cryptographic_primitives/hashes.mdx#sha256) -- [Schnorr signature verification](./cryptographic_primitives/schnorr.mdx) - [Blake2s](./cryptographic_primitives/hashes.mdx#blake2s) - [Blake3](./cryptographic_primitives/hashes.mdx#blake3) - [Pedersen Hash](./cryptographic_primitives/hashes.mdx#pedersen_hash) diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/bn254.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/bn254.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/bn254.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/bn254.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/boundedvec.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/boundedvec.md similarity index 82% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/boundedvec.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/boundedvec.md index 509b214bf3a..dad25f26717 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/containers/boundedvec.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/boundedvec.md @@ -380,6 +380,66 @@ Example: let bounded_vec: BoundedVec = BoundedVec::from_array([1, 2, 3]) ``` +### from_parts + +```rust +pub fn from_parts(mut array: [T; MaxLen], len: u32) -> Self +``` + +Creates a new BoundedVec from the given array and length. +The given length must be less than or equal to the length of the array. + +This function will zero out any elements at or past index `len` of `array`. +This incurs an extra runtime cost of O(MaxLen). If you are sure your array is +zeroed after that index, you can use `from_parts_unchecked` to remove the extra loop. + +Example: + +```rust title="from-parts" showLineNumbers +let vec: BoundedVec = BoundedVec::from_parts([1, 2, 3, 0], 3); + assert_eq(vec.len(), 3); + + // Any elements past the given length are zeroed out, so these + // two vectors will be completely equal + let vec1: BoundedVec = BoundedVec::from_parts([1, 2, 3, 1], 3); + let vec2: BoundedVec = BoundedVec::from_parts([1, 2, 3, 2], 3); + assert_eq(vec1, vec2); +``` +> Source code: noir_stdlib/src/collections/bounded_vec.nr#L663-L672 + + +### from_parts_unchecked + +```rust +pub fn from_parts_unchecked(array: [T; MaxLen], len: u32) -> Self +``` + +Creates a new BoundedVec from the given array and length. +The given length must be less than or equal to the length of the array. + +This function is unsafe because it expects all elements past the `len` index +of `array` to be zeroed, but does not check for this internally. Use `from_parts` +for a safe version of this function which does zero out any indices past the +given length. Invalidating this assumption can notably cause `BoundedVec::eq` +to give incorrect results since it will check even elements past `len`. + +Example: + +```rust title="from-parts-unchecked" showLineNumbers +let vec: BoundedVec = BoundedVec::from_parts_unchecked([1, 2, 3, 0], 3); + assert_eq(vec.len(), 3); + + // invalid use! + let vec1: BoundedVec = BoundedVec::from_parts_unchecked([1, 2, 3, 1], 3); + let vec2: BoundedVec = BoundedVec::from_parts_unchecked([1, 2, 3, 2], 3); + + // both vectors have length 3 so we'd expect them to be equal, but this + // fails because elements past the length are still checked in eq + assert(vec1 != vec2); +``` +> Source code: noir_stdlib/src/collections/bounded_vec.nr#L677-L688 + + ### map ```rust @@ -394,7 +454,7 @@ Example: let vec: BoundedVec = BoundedVec::from_array([1, 2, 3, 4]); let result = vec.map(|value| value * 2); ``` -> Source code: noir_stdlib/src/collections/bounded_vec.nr#L495-L498 +> Source code: noir_stdlib/src/collections/bounded_vec.nr#L551-L554 ### any diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/hashmap.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/hashmap.md similarity index 82% rename from noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/hashmap.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/hashmap.md index 395cc312705..810baad16ba 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/containers/hashmap.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/hashmap.md @@ -45,7 +45,7 @@ where /// ``` fn default() -> Self { ``` -> Source code: noir_stdlib/src/collections/map.nr#L681-L696 +> Source code: noir_stdlib/src/collections/map.nr#L688-L703 Creates a fresh, empty HashMap. @@ -61,7 +61,7 @@ Example: let hashmap: HashMap> = HashMap::default(); assert(hashmap.is_empty()); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L207-L210 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L208-L211 Because `HashMap` has so many generic arguments that are likely to be the same throughout @@ -70,7 +70,7 @@ your program, it may be helpful to create a type alias: ```rust title="type_alias" showLineNumbers type MyMap = HashMap>; ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L201-L203 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L202-L204 ### with_hasher @@ -95,7 +95,7 @@ let my_hasher: BuildHasherDefault = Default::default(); HashMap::with_hasher(my_hasher); assert(hashmap.is_empty()); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L211-L216 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L212-L217 ### get @@ -108,7 +108,7 @@ pub fn get(self, key: K) -> Option H: Hasher, { ``` -> Source code: noir_stdlib/src/collections/map.nr#L465-L472 +> Source code: noir_stdlib/src/collections/map.nr#L472-L479 Retrieves a value from the hashmap, returning `Option::none()` if it was not found. @@ -124,7 +124,7 @@ fn get_example(map: HashMap } } ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L296-L304 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L297-L305 ### insert @@ -137,7 +137,7 @@ pub fn insert(&mut self, key: K, value: V) H: Hasher, { ``` -> Source code: noir_stdlib/src/collections/map.nr#L507-L514 +> Source code: noir_stdlib/src/collections/map.nr#L514-L521 Inserts a new key-value pair into the map. If the key was already in the map, its @@ -150,7 +150,7 @@ let mut map: HashMap> = Has map.insert(12, 42); assert(map.len() == 1); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L217-L221 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L218-L222 ### remove @@ -163,7 +163,7 @@ pub fn remove(&mut self, key: K) H: Hasher, { ``` -> Source code: noir_stdlib/src/collections/map.nr#L563-L570 +> Source code: noir_stdlib/src/collections/map.nr#L570-L577 Removes the given key-value pair from the map. If the key was not already present @@ -179,7 +179,7 @@ map.remove(12); map.remove(12); assert(map.is_empty()); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L224-L231 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L225-L232 ### is_empty @@ -203,7 +203,7 @@ assert(map.is_empty()); map.remove(1); assert(map.is_empty()); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L232-L240 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L233-L241 ### len @@ -211,7 +211,7 @@ assert(map.is_empty()); ```rust title="len" showLineNumbers pub fn len(self) -> u32 { ``` -> Source code: noir_stdlib/src/collections/map.nr#L424-L426 +> Source code: noir_stdlib/src/collections/map.nr#L431-L433 Returns the current length of this hash map. @@ -234,7 +234,7 @@ Example: map.remove(1); assert(map.len() == 2); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L241-L256 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L242-L257 ### capacity @@ -242,7 +242,7 @@ Example: ```rust title="capacity" showLineNumbers pub fn capacity(_self: Self) -> u32 { ``` -> Source code: noir_stdlib/src/collections/map.nr#L446-L448 +> Source code: noir_stdlib/src/collections/map.nr#L453-L455 Returns the maximum capacity of this hashmap. This is always equal to the capacity @@ -262,7 +262,7 @@ let empty_map: HashMap> = assert(empty_map.len() == 0); assert(empty_map.capacity() == 42); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L257-L262 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L258-L263 ### clear @@ -282,7 +282,7 @@ assert(!map.is_empty()); map.clear(); assert(map.is_empty()); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L263-L267 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L264-L268 ### contains_key @@ -311,7 +311,7 @@ if map.contains_key(7) { println("No value for key 7!"); } ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L268-L275 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L269-L276 ### entries @@ -340,7 +340,7 @@ let entries = map.entries(); } } ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L307-L318 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L308-L319 ### keys @@ -348,7 +348,7 @@ let entries = map.entries(); ```rust title="keys" showLineNumbers pub fn keys(self) -> BoundedVec { ``` -> Source code: noir_stdlib/src/collections/map.nr#L227-L229 +> Source code: noir_stdlib/src/collections/map.nr#L230-L232 Returns a vector of each key present in the hashmap. @@ -368,7 +368,7 @@ let keys = map.keys(); } } ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L319-L329 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L320-L330 ### values @@ -376,7 +376,7 @@ let keys = map.keys(); ```rust title="values" showLineNumbers pub fn values(self) -> BoundedVec { ``` -> Source code: noir_stdlib/src/collections/map.nr#L262-L264 +> Source code: noir_stdlib/src/collections/map.nr#L267-L269 Returns a vector of each value present in the hashmap. @@ -395,7 +395,7 @@ let values = map.values(); } } ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L330-L339 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L331-L340 ### iter_mut @@ -408,7 +408,7 @@ pub fn iter_mut(&mut self, f: fn(K, V) -> (K, V)) H: Hasher, { ``` -> Source code: noir_stdlib/src/collections/map.nr#L297-L304 +> Source code: noir_stdlib/src/collections/map.nr#L304-L311 Iterates through each key-value pair of the HashMap, setting each key-value pair to the @@ -427,7 +427,7 @@ Example: // Add 1 to each key in the map, and double the value associated with that key. map.iter_mut(|k, v| (k + 1, v * 2)); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L343-L346 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L344-L347 ### iter_keys_mut @@ -440,7 +440,7 @@ pub fn iter_keys_mut(&mut self, f: fn(K) -> K) H: Hasher, { ``` -> Source code: noir_stdlib/src/collections/map.nr#L335-L342 +> Source code: noir_stdlib/src/collections/map.nr#L342-L349 Iterates through the HashMap, mutating each key to the result returned from @@ -459,7 +459,7 @@ Example: // Double each key, leaving the value associated with that key untouched map.iter_keys_mut(|k| k * 2); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L347-L350 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L348-L351 ### iter_values_mut @@ -467,7 +467,7 @@ Example: ```rust title="iter_values_mut" showLineNumbers pub fn iter_values_mut(&mut self, f: fn(V) -> V) { ``` -> Source code: noir_stdlib/src/collections/map.nr#L367-L369 +> Source code: noir_stdlib/src/collections/map.nr#L374-L376 Iterates through the HashMap, applying the given function to each value and mutating the @@ -480,7 +480,7 @@ Example: // Halve each value map.iter_values_mut(|v| v / 2); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L351-L354 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L352-L355 ### retain @@ -488,7 +488,7 @@ Example: ```rust title="retain" showLineNumbers pub fn retain(&mut self, f: fn(K, V) -> bool) { ``` -> Source code: noir_stdlib/src/collections/map.nr#L388-L390 +> Source code: noir_stdlib/src/collections/map.nr#L395-L397 Retains only the key-value pairs for which the given function returns true. @@ -499,7 +499,7 @@ Example: ```rust title="retain_example" showLineNumbers map.retain(|k, v| (k != 0) & (v != 0)); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L279-L281 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L280-L282 ## Trait Implementations @@ -522,7 +522,7 @@ where /// ``` fn default() -> Self { ``` -> Source code: noir_stdlib/src/collections/map.nr#L681-L696 +> Source code: noir_stdlib/src/collections/map.nr#L688-L703 Constructs an empty HashMap. @@ -533,7 +533,7 @@ Example: let hashmap: HashMap> = HashMap::default(); assert(hashmap.is_empty()); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L207-L210 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L208-L211 ### eq @@ -564,7 +564,7 @@ where /// ``` fn eq(self, other: HashMap) -> bool { ``` -> Source code: noir_stdlib/src/collections/map.nr#L629-L654 +> Source code: noir_stdlib/src/collections/map.nr#L636-L661 Checks if two HashMaps are equal. @@ -583,5 +583,5 @@ let mut map1: HashMap> = Hash assert(map1 == map2); ``` -> Source code: test_programs/execution_success/hashmap/src/main.nr#L282-L293 +> Source code: test_programs/execution_success/hashmap/src/main.nr#L283-L294 diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/index.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/index.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/index.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/index.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/vec.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/vec.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/containers/vec.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/containers/vec.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/_category_.json b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/_category_.json similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/concepts/data_types/_category_.json rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/_category_.json diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/ciphers.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/ciphers.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/ciphers.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/ciphers.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/ecdsa_sig_verification.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/embedded_curve_ops.mdx diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/hashes.mdx b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/hashes.mdx similarity index 98% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/hashes.mdx rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/hashes.mdx index 541a1971561..e010d6d85c2 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/cryptographic_primitives/hashes.mdx +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/hashes.mdx @@ -136,7 +136,7 @@ Given an array of bytes (`u8`), returns the resulting keccak hash as an array of ```rust title="keccak256" showLineNumbers pub fn keccak256(input: [u8; N], message_size: u32) -> [u8; 32] ``` -> Source code: noir_stdlib/src/hash/mod.nr#L116-L118 +> Source code: noir_stdlib/src/hash/mod.nr#L119-L121 example: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/index.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/index.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/cryptographic_primitives/index.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/cryptographic_primitives/index.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/fmtstr.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/fmtstr.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/fmtstr.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/fmtstr.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/is_unconstrained.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/is_unconstrained.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/is_unconstrained.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/is_unconstrained.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/logging.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/logging.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/logging.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/logging.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/mem.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/mem.md similarity index 64% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/mem.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/mem.md index 95d36ac2a72..1e9102b32dc 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/mem.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/mem.md @@ -42,7 +42,7 @@ fn checked_transmute(value: T) -> U Transmutes a value of one type into the same value but with a new type `U`. This function is safe to use since both types are asserted to be equal later during compilation after the concrete values for generic types become known. -This function is useful for cases where the compiler may fails a type check that is expected to pass where +This function is useful for cases where the compiler may fail a type check that is expected to pass where a user knows the two types to be equal. For example, when using arithmetic generics there are cases the compiler does not see as equal, such as `[Field; N*(A + B)]` and `[Field; N*A + N*B]`, which users may know to be equal. In these cases, `checked_transmute` can be used to cast the value to the desired type while also preserving safety @@ -50,3 +50,33 @@ by checking this equality once `N`, `A`, `B` are fully resolved. Note that since this safety check is performed after type checking rather than during, no error is issued if the function containing `checked_transmute` is never called. + +# `std::mem::array_refcount` + +```rust +fn array_refcount(array: [T; N]) -> u32 {} +``` + +Returns the internal reference count of an array value in unconstrained code. + +Arrays only have reference count in unconstrained code - using this anywhere +else will return zero. + +This function is mostly intended for debugging compiler optimizations but can also be used +to find where array copies may be happening in unconstrained code by placing it before array +mutations. + +# `std::mem::slice_refcount` + +```rust +fn slice_refcount(slice: [T]) -> u32 {} +``` + +Returns the internal reference count of a slice value in unconstrained code. + +Slices only have reference count in unconstrained code - using this anywhere +else will return zero. + +This function is mostly intended for debugging compiler optimizations but can also be used +to find where slice copies may be happening in unconstrained code by placing it before slice +mutations. diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/merkle_trees.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/merkle_trees.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/noir/standard_library/merkle_trees.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/merkle_trees.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/ctstring.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/ctstring.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/ctstring.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/ctstring.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/expr.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/expr.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/expr.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/expr.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/function_def.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/function_def.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/function_def.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/function_def.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/index.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/index.md similarity index 96% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/index.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/index.md index 14544c07442..132f4c0216e 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/index.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/index.md @@ -178,7 +178,7 @@ way to write your derive handler. The arguments are as follows: - `for_each_field`: An operation to be performed on each field. E.g. `|name| quote { (self.$name == other.$name) }`. - `join_fields_with`: A separator to join each result of `for_each_field` with. E.g. `quote { & }`. You can also use an empty `quote {}` for no separator. -- `body`: The result of the field operations are passed into this function for any final processing. +- `body`: The result of the field operations is passed into this function for any final processing. This is the place to insert any setup/teardown code the trait requires. If the trait doesn't require any such code, you can return the body as-is: `|body| body`. @@ -199,7 +199,7 @@ comptime fn derive_hash(s: StructDefinition) -> Quoted { ) } ``` -> Source code: noir_stdlib/src/hash/mod.nr#L137-L151 +> Source code: noir_stdlib/src/hash/mod.nr#L140-L154 Example deriving `Ord`: diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/module.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/module.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/module.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/module.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/op.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/op.md similarity index 97% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/op.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/op.md index 03ea49ad8ec..b53208d6507 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/op.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/op.md @@ -218,10 +218,10 @@ pub fn is_shift_right(self) -> bool { #### is_shift_left -```rust title="is_shift_right" showLineNumbers -pub fn is_shift_right(self) -> bool { +```rust title="is_shift_left" showLineNumbers +pub fn is_shift_left(self) -> bool { ``` -> Source code: noir_stdlib/src/meta/op.nr#L164-L166 +> Source code: noir_stdlib/src/meta/op.nr#L170-L172 `true` if this operator is `<<` diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/quoted.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/quoted.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/quoted.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/quoted.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/struct_def.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/struct_def.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/struct_def.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/struct_def.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/trait_constraint.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/trait_constraint.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.34.0/noir/standard_library/meta/trait_constraint.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/trait_constraint.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/trait_def.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/trait_def.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/trait_def.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/trait_def.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/trait_impl.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/trait_impl.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/noir/standard_library/meta/trait_impl.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/trait_impl.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/typ.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/typ.md similarity index 99% rename from noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/typ.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/typ.md index 90222c222f5..3a85a739a4c 100644 --- a/noir/noir-repo/docs/versioned_docs/version-v0.39.0/noir/standard_library/meta/typ.md +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/noir/standard_library/meta/typ.md @@ -180,7 +180,7 @@ pub comptime fn get_trait_impl(self, constraint: TraitConstraint) -> Option` — The name of the package to check -* `--workspace` — Check all packages in the workspace +* `--package ` — The name of the package to run the command on. By default run on the first one found moving up along the ancestors of the current directory +* `--workspace` — Run on all packages in the workspace Possible values: `true`, `false` @@ -108,6 +108,10 @@ Checks the constraint system for errors Possible values: `true`, `false` +* `--skip-brillig-constraints-check` — Flag to turn off the compiler check for missing Brillig call constrains. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code + + Possible values: `true`, `false` + @@ -123,6 +127,11 @@ Format the Noir files in a workspace Possible values: `true`, `false` +* `--package ` — The name of the package to run the command on. By default run on the first one found moving up along the ancestors of the current directory +* `--workspace` — Run on all packages in the workspace + + Possible values: `true`, `false` + @@ -134,8 +143,8 @@ Compile the program and its secret execution trace into ACIR format ###### **Options:** -* `--package ` — The name of the package to compile -* `--workspace` — Compile all packages in the workspace +* `--package ` — The name of the package to run the command on. By default run on the first one found moving up along the ancestors of the current directory +* `--workspace` — Run on all packages in the workspace Possible values: `true`, `false` @@ -167,6 +176,10 @@ Compile the program and its secret execution trace into ACIR format Possible values: `true`, `false` +* `--skip-brillig-constraints-check` — Flag to turn off the compiler check for missing Brillig call constrains. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code + + Possible values: `true`, `false` + @@ -239,8 +252,8 @@ Defaults to the name of the package being executed. * `-p`, `--prover-name ` — The name of the toml file which contains the inputs for the prover Default value: `Prover` -* `--package ` — The name of the package to execute -* `--workspace` — Execute all packages in the workspace +* `--package ` — The name of the package to run the command on. By default run on the first one found moving up along the ancestors of the current directory +* `--workspace` — Run on all packages in the workspace Possible values: `true`, `false` @@ -272,6 +285,10 @@ Defaults to the name of the package being executed. Possible values: `true`, `false` +* `--skip-brillig-constraints-check` — Flag to turn off the compiler check for missing Brillig call constrains. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code + + Possible values: `true`, `false` + * `--oracle-resolver ` — JSON RPC url to solve oracle calls @@ -320,6 +337,10 @@ Executes a circuit in debug mode Possible values: `true`, `false` +* `--skip-brillig-constraints-check` — Flag to turn off the compiler check for missing Brillig call constrains. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code + + Possible values: `true`, `false` + * `--acir-mode` — Force ACIR output (disabling instrumentation) Possible values: `true`, `false` @@ -351,8 +372,8 @@ Run the tests for this program Possible values: `true`, `false` -* `--package ` — The name of the package to test -* `--workspace` — Test all packages in the workspace +* `--package ` — The name of the package to run the command on. By default run on the first one found moving up along the ancestors of the current directory +* `--workspace` — Run on all packages in the workspace Possible values: `true`, `false` @@ -384,7 +405,28 @@ Run the tests for this program Possible values: `true`, `false` +* `--skip-brillig-constraints-check` — Flag to turn off the compiler check for missing Brillig call constrains. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code + + Possible values: `true`, `false` + * `--oracle-resolver ` — JSON RPC url to solve oracle calls +* `--test-threads ` — Number of threads used for running tests in parallel + + Default value: `4` +* `--format ` — Configure formatting of output + + Possible values: + - `pretty`: + Print verbose output + - `terse`: + Display one character per test + - `json`: + Output a JSON Lines document + +* `-q`, `--quiet` — Display one character per test instead of one line + + Possible values: `true`, `false` + @@ -398,8 +440,8 @@ Current information provided per circuit: 1. The number of ACIR opcodes 2. Count ###### **Options:** -* `--package ` — The name of the package to detail -* `--workspace` — Detail all packages in the workspace +* `--package ` — The name of the package to run the command on. By default run on the first one found moving up along the ancestors of the current directory +* `--workspace` — Run on all packages in the workspace Possible values: `true`, `false` @@ -438,6 +480,10 @@ Current information provided per circuit: 1. The number of ACIR opcodes 2. Count Possible values: `true`, `false` +* `--skip-brillig-constraints-check` — Flag to turn off the compiler check for missing Brillig call constrains. Warning: This can improve compilation speed but can also lead to correctness errors. This check should always be run on production code + + Possible values: `true`, `false` + diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/noir_codegen.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/reference/noir_codegen.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/reference/noir_codegen.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/reference/noir_codegen.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.37.0/tooling/debugger.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tooling/debugger.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.37.0/tooling/debugger.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tooling/debugger.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/tooling/language_server.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tooling/language_server.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/tooling/language_server.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tooling/language_server.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v0.32.0/tooling/testing.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tooling/testing.md similarity index 100% rename from noir/noir-repo/docs/versioned_docs/version-v0.32.0/tooling/testing.md rename to noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tooling/testing.md diff --git a/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tutorials/noirjs_app.md b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tutorials/noirjs_app.md new file mode 100644 index 00000000000..8967ee005ce --- /dev/null +++ b/noir/noir-repo/docs/versioned_docs/version-v1.0.0-beta.1/tutorials/noirjs_app.md @@ -0,0 +1,306 @@ +--- +title: Building a web app with Noir and Barretenberg +description: Learn how to setup a new app that uses Noir to generate and verify zero-knowledge SNARK proofs in a typescript or javascript environment. +keywords: [how to, guide, javascript, typescript, noir, barretenberg, zero-knowledge, proofs, app] +sidebar_position: 0 +pagination_next: noir/concepts/data_types/index +--- + +NoirJS is a Typescript package meant to work both in a browser and a server environment. + +In this tutorial, we will combine NoirJS with Aztec's Barretenberg backend to build a simple web app. From here, you should get an idea on how to proceed with your own Noir projects! + +You can find the complete app code for this guide [here](https://github.com/noir-lang/tiny-noirjs-app). + +## Dependencies + +Before we start, we want to make sure we have Node installed. For convenience (and speed), we can just install [Bun](https://bun.sh) as our package manager, and Node will work out-of-the-box: + +```bash +curl -fsSL https://bun.sh/install | bash +``` + +Let's go barebones. Doing the bare minimum is not only simple, but also allows you to easily adapt it to almost any frontend framework. + +Barebones means we can immediately start with the dependencies even on an empty folder 😈: + +```bash +bun i @noir-lang/noir_wasm@1.0.0-beta.0 @noir-lang/noir_js@1.0.0-beta.0 @aztec/bb.js@0.63.1 +``` + +Wait, what are these dependencies? + +- `noir_wasm` is the `wasm` version of the Noir compiler. Although most developers prefer to use `nargo` for compiling, there's nothing wrong with `noir_wasm`. We like `noir_wasm`. +- `noir_js` is the main Noir package. It will execute our program, and generate the witness that will be sent to the backend. +- `bb.js` is the Typescript interface for Aztec's Barretenberg proving backend. It also uses the `wasm` version in order to run on the browser. + +:::info + +In this guide, we will install versions pinned to 1.0.0-beta.0. These work with Barretenberg version 0.63.1, so we are using that one version too. Feel free to try with older or later versions, though! + +::: + +## Setting up our Noir program + +ZK is a powerful technology. An app that reveals computational correctness but doesn't reveal some of its inputs is almost unbelievable, yet Noir makes it as easy as a single line of code. + +:::tip + +It's not just you. We also enjoy syntax highlighting. [Check out the Language Server](../tooling/language_server.md) + +::: + +All you need is a `main.nr` and a `Nargo.toml` file. You can follow the [noirup](../getting_started/noir_installation.md) installation and just run `noirup -v 1.0.0-beta.0`, or just create them by hand: + +```bash +mkdir -p circuit/src +touch circuit/src/main.nr circuit/Nargo.toml +``` + +To make our program interesting, let's give it a real use-case scenario: Bob wants to prove he is older than 18, without disclosing his age. Open `main.nr` and write: + +```rust +fn main(age: u8) { + assert(age >= 18); +} +``` + +This program accepts a private input called age, and simply proves this number is higher than 18. But to run this code, we need to give the compiler a `Nargo.toml` with at least a name and a type: + +```toml +[package] +name = "circuit" +type = "bin" +``` + +This is all that we need to get started with Noir. + +![my heart is ready for you, noir.js](@site/static/img/memes/titanic.jpeg) + +## Setting up our app + +Remember when apps only had one `html` and one `js` file? Well, that's enough for Noir webapps. Let's create them: + +```bash +touch index.html index.js +``` + +And add something useful to our HTML file: + +```html + + + + + + +

Noir app

+
+ + +
+
+

Logs

+

Proof

+
+ + +``` + +It _could_ be a beautiful UI... Depending on which universe you live in. In any case, we're using some scary CSS to make two boxes that will show cool things on the screen. + +As for the JS, real madmen could just `console.log` everything, but let's say we want to see things happening (the true initial purpose of JS... right?). Here's some boilerplate for that. Just paste it in `index.js`: + +```js +const show = (id, content) => { + const container = document.getElementById(id); + container.appendChild(document.createTextNode(content)); + container.appendChild(document.createElement("br")); +}; + +document.getElementById("submit").addEventListener("click", async () => { + try { + // noir goes here + } catch { + show("logs", "Oh 💔"); + } +}); + +``` + +:::info + +At this point in the tutorial, your folder structure should look like this: + +```tree +. +└── circuit + └── src + └── main.nr + Nargo.toml + index.js + package.json + index.html + ...etc +``` + +::: + +## Compile compile compile + +Finally we're up for something cool. But before we can execute a Noir program, we need to compile it into ACIR: an abstract representation. Here's where `noir_wasm` comes in. + +`noir_wasm` expects a filesystem so it can resolve dependencies. While we could use the `public` folder, let's just import those using the nice `?url` syntax provided by vite. At the top of the file: + +```js +import { compile, createFileManager } from "@noir-lang/noir_wasm" + +import main from "./circuit/src/main.nr?url"; +import nargoToml from "./circuit/Nargo.toml?url"; +``` + +Compiling on the browser is common enough that `createFileManager` already gives us a nice in-memory filesystem we can use. So all we need to compile is fetching these files, writing them to our filesystem, and compile. Add this function: + +```js +export async function getCircuit() { + const fm = createFileManager("/"); + const { body } = await fetch(main); + const { body: nargoTomlBody } = await fetch(nargoToml); + + fm.writeFile("./src/main.nr", body); + fm.writeFile("./Nargo.toml", nargoTomlBody); + return await compile(fm); +} +``` + +:::tip + +As you can imagine, with `node` it's all conveniently easier since you get native access to `fs`... + +::: + +## Some more JS + +We're starting with the good stuff now. We want to execute our circuit to get the witness, and then feed that witness to Barretenberg. Luckily, both packages are quite easy to work with. Let's import them at the top of the file: + +```js +import { UltraHonkBackend } from '@aztec/bb.js'; +import { Noir } from '@noir-lang/noir_js'; +``` + +And instantiate them inside our try-catch block: + +```ts +// try { +const { program } = await getCircuit(); +const noir = new Noir(program); +const backend = new UltraHonkBackend(program.bytecode); +// } +``` + +:::warning + +WASMs are not always easy to work with. In our case, `vite` likes serving them with the wrong MIME type. There are different fixes but we found the easiest one is just YOLO instantiating the WASMs manually. Paste this at the top of the file, just below the other imports, and it will work just fine: + +```js +import initNoirC from "@noir-lang/noirc_abi"; +import initACVM from "@noir-lang/acvm_js"; +import acvm from "@noir-lang/acvm_js/web/acvm_js_bg.wasm?url"; +import noirc from "@noir-lang/noirc_abi/web/noirc_abi_wasm_bg.wasm?url"; +await Promise.all([initACVM(fetch(acvm)), initNoirC(fetch(noirc))]); +``` + +::: + +## Executing and proving + +Now for the app itself. We're capturing whatever is in the input when people press the submit button. Inside our `try` block, let's just grab that input and get its value. Noir will gladly execute it, and give us a witness: + +```js +const age = document.getElementById("age").value; +show("logs", "Generating witness... ⏳"); +const { witness } = await noir.execute({ age }); +show("logs", "Generated witness... ✅"); + +``` + +:::note + +For the remainder of the tutorial, everything will be happening inside the `try` block + +::: + +Now we're ready to prove stuff! Let's feed some inputs to our circuit and calculate the proof: + +```js +show("logs", "Generating proof... ⏳"); +const proof = await backend.generateProof(witness); +show("logs", "Generated proof... ✅"); +show("results", proof.proof); +``` + +Our program is technically **done** . You're probably eager to see stuff happening! To serve this in a convenient way, we can use a bundler like `vite` by creating a `vite.config.js` file: + +```bash +touch vite.config.js +``` + +`vite` helps us with a little catch: `bb.js` in particular uses top-level awaits which aren't supported everywhere. So we can add this to the `vite.config.js` to make the bundler optimize them: + +```js +export default { optimizeDeps: { esbuildOptions: { target: "esnext" } } }; +``` + +This should be enough for vite. We don't even need to install it, just run: + +```bash +bunx vite +``` + +If it doesn't open a browser for you, just visit `localhost:5173`. You should now see the worst UI ever, with an ugly input. + +![Noir Webapp UI](@site/static/img/tutorials/noirjs_webapp/webapp1.png) + +Now, our circuit requires a private input `fn main(age: u8)`, and fails if it is less than 18. Let's see if it works. Submit any number above 18 (as long as it fits in 8 bits) and you should get a valid proof. Otherwise the proof won't even generate correctly. + +By the way, if you're human, you shouldn't be able to understand anything on the "proof" box. That's OK. We like you, human ❤️. + +## Verifying + +Time to celebrate, yes! But we shouldn't trust machines so blindly. Let's add these lines to see our proof being verified: + +```js +show('logs', 'Verifying proof... ⌛'); +const isValid = await backend.verifyProof(proof); +show("logs", `Proof is ${isValid ? "valid" : "invalid"}... ✅`); +``` + +You have successfully generated a client-side Noir web app! + +![coded app without math knowledge](@site/static/img/memes/flextape.jpeg) + +## Next steps + +At this point, you have a working ZK app that works on the browser. Actually, it works on a mobile phone too! + +If you want to continue learning by doing, here are some challenges for you: + +- Install [nargo](https://noir-lang.org/docs/getting_started/noir_installation) and write [Noir tests](../tooling/testing) +- Change the circuit to accept a [public input](../noir/concepts/data_types/#private--public-types) as the cutoff age. It could be different depending on the purpose, for example! +- Enjoy Noir's Rust-like syntax and write a struct `Country` that implements a trait `MinAge` with a method `get_min_age`. Then, make a struct `Person` have an `u8` as its age and a country of type `Country`. You can pass a `person` in JS just like a JSON object `person: { age, country: { min_age: 18 }}` + +The world is your stage, just have fun with ZK! You can see how noirjs is used in a full stack Next.js hardhat application in the [noir-starter repo here](https://github.com/noir-lang/noir-starter/tree/main/vite-hardhat). The example shows how to calculate a proof in the browser and verify it with a deployed Solidity verifier contract from noirjs. + +Check out other starters, tools, or just cool projects in the [awesome noir repository](https://github.com/noir-lang/awesome-noir). diff --git a/noir/noir-repo/docs/versioned_sidebars/version-v0.33.0-sidebars.json b/noir/noir-repo/docs/versioned_sidebars/version-v0.33.0-sidebars.json deleted file mode 100644 index b9ad026f69f..00000000000 --- a/noir/noir-repo/docs/versioned_sidebars/version-v0.33.0-sidebars.json +++ /dev/null @@ -1,93 +0,0 @@ -{ - "sidebar": [ - { - "type": "doc", - "id": "index" - }, - { - "type": "category", - "label": "Getting Started", - "items": [ - { - "type": "autogenerated", - "dirName": "getting_started" - } - ] - }, - { - "type": "category", - "label": "The Noir Language", - "items": [ - { - "type": "autogenerated", - "dirName": "noir" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "category", - "label": "How To Guides", - "items": [ - { - "type": "autogenerated", - "dirName": "how_to" - } - ] - }, - { - "type": "category", - "label": "Explainers", - "items": [ - { - "type": "autogenerated", - "dirName": "explainers" - } - ] - }, - { - "type": "category", - "label": "Tutorials", - "items": [ - { - "type": "autogenerated", - "dirName": "tutorials" - } - ] - }, - { - "type": "category", - "label": "Reference", - "items": [ - { - "type": "autogenerated", - "dirName": "reference" - } - ] - }, - { - "type": "category", - "label": "Tooling", - "items": [ - { - "type": "autogenerated", - "dirName": "tooling" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "doc", - "id": "migration_notes", - "label": "Migration notes" - } - ] -} diff --git a/noir/noir-repo/docs/versioned_sidebars/version-v0.34.0-sidebars.json b/noir/noir-repo/docs/versioned_sidebars/version-v0.34.0-sidebars.json deleted file mode 100644 index b9ad026f69f..00000000000 --- a/noir/noir-repo/docs/versioned_sidebars/version-v0.34.0-sidebars.json +++ /dev/null @@ -1,93 +0,0 @@ -{ - "sidebar": [ - { - "type": "doc", - "id": "index" - }, - { - "type": "category", - "label": "Getting Started", - "items": [ - { - "type": "autogenerated", - "dirName": "getting_started" - } - ] - }, - { - "type": "category", - "label": "The Noir Language", - "items": [ - { - "type": "autogenerated", - "dirName": "noir" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "category", - "label": "How To Guides", - "items": [ - { - "type": "autogenerated", - "dirName": "how_to" - } - ] - }, - { - "type": "category", - "label": "Explainers", - "items": [ - { - "type": "autogenerated", - "dirName": "explainers" - } - ] - }, - { - "type": "category", - "label": "Tutorials", - "items": [ - { - "type": "autogenerated", - "dirName": "tutorials" - } - ] - }, - { - "type": "category", - "label": "Reference", - "items": [ - { - "type": "autogenerated", - "dirName": "reference" - } - ] - }, - { - "type": "category", - "label": "Tooling", - "items": [ - { - "type": "autogenerated", - "dirName": "tooling" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "doc", - "id": "migration_notes", - "label": "Migration notes" - } - ] -} diff --git a/noir/noir-repo/docs/versioned_sidebars/version-v0.35.0-sidebars.json b/noir/noir-repo/docs/versioned_sidebars/version-v0.35.0-sidebars.json deleted file mode 100644 index b9ad026f69f..00000000000 --- a/noir/noir-repo/docs/versioned_sidebars/version-v0.35.0-sidebars.json +++ /dev/null @@ -1,93 +0,0 @@ -{ - "sidebar": [ - { - "type": "doc", - "id": "index" - }, - { - "type": "category", - "label": "Getting Started", - "items": [ - { - "type": "autogenerated", - "dirName": "getting_started" - } - ] - }, - { - "type": "category", - "label": "The Noir Language", - "items": [ - { - "type": "autogenerated", - "dirName": "noir" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "category", - "label": "How To Guides", - "items": [ - { - "type": "autogenerated", - "dirName": "how_to" - } - ] - }, - { - "type": "category", - "label": "Explainers", - "items": [ - { - "type": "autogenerated", - "dirName": "explainers" - } - ] - }, - { - "type": "category", - "label": "Tutorials", - "items": [ - { - "type": "autogenerated", - "dirName": "tutorials" - } - ] - }, - { - "type": "category", - "label": "Reference", - "items": [ - { - "type": "autogenerated", - "dirName": "reference" - } - ] - }, - { - "type": "category", - "label": "Tooling", - "items": [ - { - "type": "autogenerated", - "dirName": "tooling" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "doc", - "id": "migration_notes", - "label": "Migration notes" - } - ] -} diff --git a/noir/noir-repo/docs/versioned_sidebars/version-v0.37.0-sidebars.json b/noir/noir-repo/docs/versioned_sidebars/version-v0.37.0-sidebars.json deleted file mode 100644 index b9ad026f69f..00000000000 --- a/noir/noir-repo/docs/versioned_sidebars/version-v0.37.0-sidebars.json +++ /dev/null @@ -1,93 +0,0 @@ -{ - "sidebar": [ - { - "type": "doc", - "id": "index" - }, - { - "type": "category", - "label": "Getting Started", - "items": [ - { - "type": "autogenerated", - "dirName": "getting_started" - } - ] - }, - { - "type": "category", - "label": "The Noir Language", - "items": [ - { - "type": "autogenerated", - "dirName": "noir" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "category", - "label": "How To Guides", - "items": [ - { - "type": "autogenerated", - "dirName": "how_to" - } - ] - }, - { - "type": "category", - "label": "Explainers", - "items": [ - { - "type": "autogenerated", - "dirName": "explainers" - } - ] - }, - { - "type": "category", - "label": "Tutorials", - "items": [ - { - "type": "autogenerated", - "dirName": "tutorials" - } - ] - }, - { - "type": "category", - "label": "Reference", - "items": [ - { - "type": "autogenerated", - "dirName": "reference" - } - ] - }, - { - "type": "category", - "label": "Tooling", - "items": [ - { - "type": "autogenerated", - "dirName": "tooling" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "doc", - "id": "migration_notes", - "label": "Migration notes" - } - ] -} diff --git a/noir/noir-repo/docs/versioned_sidebars/version-v0.38.0-sidebars.json b/noir/noir-repo/docs/versioned_sidebars/version-v0.38.0-sidebars.json deleted file mode 100644 index b9ad026f69f..00000000000 --- a/noir/noir-repo/docs/versioned_sidebars/version-v0.38.0-sidebars.json +++ /dev/null @@ -1,93 +0,0 @@ -{ - "sidebar": [ - { - "type": "doc", - "id": "index" - }, - { - "type": "category", - "label": "Getting Started", - "items": [ - { - "type": "autogenerated", - "dirName": "getting_started" - } - ] - }, - { - "type": "category", - "label": "The Noir Language", - "items": [ - { - "type": "autogenerated", - "dirName": "noir" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "category", - "label": "How To Guides", - "items": [ - { - "type": "autogenerated", - "dirName": "how_to" - } - ] - }, - { - "type": "category", - "label": "Explainers", - "items": [ - { - "type": "autogenerated", - "dirName": "explainers" - } - ] - }, - { - "type": "category", - "label": "Tutorials", - "items": [ - { - "type": "autogenerated", - "dirName": "tutorials" - } - ] - }, - { - "type": "category", - "label": "Reference", - "items": [ - { - "type": "autogenerated", - "dirName": "reference" - } - ] - }, - { - "type": "category", - "label": "Tooling", - "items": [ - { - "type": "autogenerated", - "dirName": "tooling" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "doc", - "id": "migration_notes", - "label": "Migration notes" - } - ] -} diff --git a/noir/noir-repo/docs/versioned_sidebars/version-v0.39.0-sidebars.json b/noir/noir-repo/docs/versioned_sidebars/version-v0.39.0-sidebars.json deleted file mode 100644 index b9ad026f69f..00000000000 --- a/noir/noir-repo/docs/versioned_sidebars/version-v0.39.0-sidebars.json +++ /dev/null @@ -1,93 +0,0 @@ -{ - "sidebar": [ - { - "type": "doc", - "id": "index" - }, - { - "type": "category", - "label": "Getting Started", - "items": [ - { - "type": "autogenerated", - "dirName": "getting_started" - } - ] - }, - { - "type": "category", - "label": "The Noir Language", - "items": [ - { - "type": "autogenerated", - "dirName": "noir" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "category", - "label": "How To Guides", - "items": [ - { - "type": "autogenerated", - "dirName": "how_to" - } - ] - }, - { - "type": "category", - "label": "Explainers", - "items": [ - { - "type": "autogenerated", - "dirName": "explainers" - } - ] - }, - { - "type": "category", - "label": "Tutorials", - "items": [ - { - "type": "autogenerated", - "dirName": "tutorials" - } - ] - }, - { - "type": "category", - "label": "Reference", - "items": [ - { - "type": "autogenerated", - "dirName": "reference" - } - ] - }, - { - "type": "category", - "label": "Tooling", - "items": [ - { - "type": "autogenerated", - "dirName": "tooling" - } - ] - }, - { - "type": "html", - "value": "
", - "defaultStyle": true - }, - { - "type": "doc", - "id": "migration_notes", - "label": "Migration notes" - } - ] -} diff --git a/noir/noir-repo/docs/versioned_sidebars/version-v0.32.0-sidebars.json b/noir/noir-repo/docs/versioned_sidebars/version-v1.0.0-beta.1-sidebars.json similarity index 100% rename from noir/noir-repo/docs/versioned_sidebars/version-v0.32.0-sidebars.json rename to noir/noir-repo/docs/versioned_sidebars/version-v1.0.0-beta.1-sidebars.json diff --git a/noir/noir-repo/examples/codegen_verifier/codegen_verifier.sh b/noir/noir-repo/examples/codegen_verifier/codegen_verifier.sh index f224f74168f..9d1a6251dde 100755 --- a/noir/noir-repo/examples/codegen_verifier/codegen_verifier.sh +++ b/noir/noir-repo/examples/codegen_verifier/codegen_verifier.sh @@ -11,7 +11,7 @@ $BACKEND contract -o ./src/contract.sol # We now generate a proof and check whether the verifier contract will verify it. -nargo execute witness +nargo execute --pedantic-solving witness PROOF_PATH=./target/proof $BACKEND prove -b ./target/hello_world.json -w ./target/witness.gz -o $PROOF_PATH diff --git a/noir/noir-repo/examples/prove_and_verify/prove_and_verify.sh b/noir/noir-repo/examples/prove_and_verify/prove_and_verify.sh index df3ec3ff97a..411f5258caf 100755 --- a/noir/noir-repo/examples/prove_and_verify/prove_and_verify.sh +++ b/noir/noir-repo/examples/prove_and_verify/prove_and_verify.sh @@ -3,7 +3,7 @@ set -eu BACKEND=${BACKEND:-bb} -nargo execute witness +nargo execute --pedantic-solving witness # TODO: `bb` should create `proofs` directory if it doesn't exist. mkdir -p proofs @@ -11,4 +11,4 @@ $BACKEND prove -b ./target/hello_world.json -w ./target/witness.gz # TODO: backend should automatically generate vk if necessary. $BACKEND write_vk -b ./target/hello_world.json -$BACKEND verify -k ./target/vk -p ./proofs/proof \ No newline at end of file +$BACKEND verify -k ./target/vk -p ./proofs/proof diff --git a/noir/noir-repo/noir_stdlib/src/aes128.nr b/noir/noir-repo/noir_stdlib/src/aes128.nr index 7b0876b86f3..d781812eee9 100644 --- a/noir/noir-repo/noir_stdlib/src/aes128.nr +++ b/noir/noir-repo/noir_stdlib/src/aes128.nr @@ -1,4 +1,8 @@ #[foreign(aes128_encrypt)] // docs:start:aes128 -pub fn aes128_encrypt(input: [u8; N], iv: [u8; 16], key: [u8; 16]) -> [u8] {} +pub fn aes128_encrypt( + input: [u8; N], + iv: [u8; 16], + key: [u8; 16], +) -> [u8; N + 16 - N % 16] {} // docs:end:aes128 diff --git a/noir/noir-repo/noir_stdlib/src/array/check_shuffle.nr b/noir/noir-repo/noir_stdlib/src/array/check_shuffle.nr index 2e8c227feee..327f0559810 100644 --- a/noir/noir-repo/noir_stdlib/src/array/check_shuffle.nr +++ b/noir/noir-repo/noir_stdlib/src/array/check_shuffle.nr @@ -42,6 +42,8 @@ pub(crate) fn check_shuffle(lhs: [T; N], rhs: [T; N]) where T: Eq, { + /// Safety: shuffle_indices is ensured to be a permutation of 0..N, and then + /// shuffle_indices is ensured to map lhs to rhs: assert(lhs[i] == rhs[shuffle_indices[i]]), for all i in 0..N unsafe { let shuffle_indices = __get_shuffle_indices(lhs, rhs); diff --git a/noir/noir-repo/noir_stdlib/src/array/mod.nr b/noir/noir-repo/noir_stdlib/src/array/mod.nr index 8bb425854f2..85cc0580aae 100644 --- a/noir/noir-repo/noir_stdlib/src/array/mod.nr +++ b/noir/noir-repo/noir_stdlib/src/array/mod.nr @@ -157,7 +157,7 @@ where /// } /// ``` pub fn sort(self) -> Self { - self.sort_via(|a: T, b: T| a <= b) + self.sort_via(|a, b| a <= b) } } @@ -184,10 +184,10 @@ where /// } /// ``` pub fn sort_via(self, ordering: fn[Env](T, T) -> bool) -> Self { + /// Safety: `sorted` array is checked to be: + /// a. a permutation of `input`'s elements + /// b. satisfying the predicate `ordering` unsafe { - // Safety: `sorted` array is checked to be: - // a. a permutation of `input`'s elements - // b. satisfying the predicate `ordering` let sorted = quicksort::quicksort(self, ordering); if !is_unconstrained() { diff --git a/noir/noir-repo/noir_stdlib/src/bigint.nr b/noir/noir-repo/noir_stdlib/src/bigint.nr index c94a7a75f25..6ce3ea7d72a 100644 --- a/noir/noir-repo/noir_stdlib/src/bigint.nr +++ b/noir/noir-repo/noir_stdlib/src/bigint.nr @@ -33,17 +33,17 @@ pub struct BigInt { // docs:end:big_int_definition impl BigInt { - #[builtin(bigint_add)] + #[foreign(bigint_add)] fn bigint_add(self, other: BigInt) -> BigInt {} - #[builtin(bigint_sub)] + #[foreign(bigint_sub)] fn bigint_sub(self, other: BigInt) -> BigInt {} - #[builtin(bigint_mul)] + #[foreign(bigint_mul)] fn bigint_mul(self, other: BigInt) -> BigInt {} - #[builtin(bigint_div)] + #[foreign(bigint_div)] fn bigint_div(self, other: BigInt) -> BigInt {} - #[builtin(bigint_from_le_bytes)] + #[foreign(bigint_from_le_bytes)] fn from_le_bytes(bytes: [u8], modulus: [u8]) -> BigInt {} - #[builtin(bigint_to_le_bytes)] + #[foreign(bigint_to_le_bytes)] fn to_le_bytes(self) -> [u8; 32] {} fn check_32_bytes(self: Self, other: BigInt) -> bool { diff --git a/noir/noir-repo/noir_stdlib/src/cmp.nr b/noir/noir-repo/noir_stdlib/src/cmp.nr index ae515150a4d..7f19594c30e 100644 --- a/noir/noir-repo/noir_stdlib/src/cmp.nr +++ b/noir/noir-repo/noir_stdlib/src/cmp.nr @@ -12,7 +12,7 @@ comptime fn derive_eq(s: StructDefinition) -> Quoted { let signature = quote { fn eq(_self: Self, _other: Self) -> bool }; let for_each_field = |name| quote { (_self.$name == _other.$name) }; let body = |fields| { - if s.fields().len() == 0 { + if s.fields_as_written().len() == 0 { quote { true } } else { fields diff --git a/noir/noir-repo/noir_stdlib/src/collections/bounded_vec.nr b/noir/noir-repo/noir_stdlib/src/collections/bounded_vec.nr index 0ad39c518c4..7aed5e6a0e4 100644 --- a/noir/noir-repo/noir_stdlib/src/collections/bounded_vec.nr +++ b/noir/noir-repo/noir_stdlib/src/collections/bounded_vec.nr @@ -620,6 +620,7 @@ mod bounded_vec_tests { mod trait_from { use crate::collections::bounded_vec::BoundedVec; + use crate::convert::From; #[test] fn simple() { diff --git a/noir/noir-repo/noir_stdlib/src/collections/map.nr b/noir/noir-repo/noir_stdlib/src/collections/map.nr index 2b0da1b90ec..bc0b80124db 100644 --- a/noir/noir-repo/noir_stdlib/src/collections/map.nr +++ b/noir/noir-repo/noir_stdlib/src/collections/map.nr @@ -271,7 +271,7 @@ impl HashMap { for slot in self._table { if slot.is_valid() { - let (_, value) = unsafe { slot.key_value_unchecked() }; + let (_, value) = slot.key_value_unchecked(); values.push(value); } } diff --git a/noir/noir-repo/noir_stdlib/src/collections/umap.nr b/noir/noir-repo/noir_stdlib/src/collections/umap.nr index 7aebeb437cf..bcb9759b4db 100644 --- a/noir/noir-repo/noir_stdlib/src/collections/umap.nr +++ b/noir/noir-repo/noir_stdlib/src/collections/umap.nr @@ -113,6 +113,7 @@ impl UHashMap { H: Hasher, { // docs:end:contains_key + /// Safety: unconstrained context unsafe { self.get(key) }.is_some() } @@ -432,6 +433,7 @@ where // Not marked as deleted and has key-value. if equal & slot.is_valid() { let (key, value) = slot.key_value_unchecked(); + /// Safety: unconstrained context let other_value = unsafe { other.get(key) }; if other_value.is_none() { diff --git a/noir/noir-repo/noir_stdlib/src/convert.nr b/noir/noir-repo/noir_stdlib/src/convert.nr index d4bea8b6390..1e5c1484b5f 100644 --- a/noir/noir-repo/noir_stdlib/src/convert.nr +++ b/noir/noir-repo/noir_stdlib/src/convert.nr @@ -117,3 +117,51 @@ impl From for Field { } } // docs:end:from-impls + +/// A generic interface for casting between primitive types, +/// equivalent of using the `as` keyword between values. +/// +/// # Example +/// +/// ``` +/// let x: Field = 1234567890; +/// let y: u8 = x as u8; +/// let z: u8 = x.as_(); +/// assert_eq(y, z); +/// ``` +pub trait AsPrimitive { + /// The equivalent of doing `self as T`. + fn as_(self) -> T; +} + +#[generate_as_primitive_impls] +comptime fn generate_as_primitive_impls(_: FunctionDefinition) -> Quoted { + let types = [ + quote { bool }, + quote { u8 }, + quote { u16 }, + quote { u32 }, + quote { u64 }, + quote { i8 }, + quote { i16 }, + quote { i32 }, + quote { i64 }, + quote { Field }, + ]; + + let mut impls = &[]; + for type1 in types { + for type2 in types { + impls = impls.push_back( + quote { + impl AsPrimitive<$type1> for $type2 { + fn as_(self) -> $type1 { + self as $type1 + } + } + }, + ); + } + } + impls.join(quote {}) +} diff --git a/noir/noir-repo/noir_stdlib/src/field/bn254.nr b/noir/noir-repo/noir_stdlib/src/field/bn254.nr index a7ca7d77373..a298a5d1e38 100644 --- a/noir/noir-repo/noir_stdlib/src/field/bn254.nr +++ b/noir/noir-repo/noir_stdlib/src/field/bn254.nr @@ -38,6 +38,9 @@ unconstrained fn lte_hint(x: Field, y: Field) -> bool { fn assert_gt_limbs(a: (Field, Field), b: (Field, Field)) { let (alo, ahi) = a; let (blo, bhi) = b; + /// Safety: borrow is enforced to be boolean due to its type. + /// if borrow is 0, it asserts that (alo > blo && ahi >= bhi) + /// if borrow is 1, it asserts that (alo <= blo && ahi > bhi) unsafe { let borrow = lte_hint(alo, blo); @@ -54,6 +57,7 @@ pub fn decompose(x: Field) -> (Field, Field) { if is_unconstrained() { compute_decomposition(x) } else { + /// Safety: decomposition is properly checked below unsafe { // Take hints of the decomposition let (xlo, xhi) = decompose_hint(x); @@ -74,7 +78,10 @@ pub fn decompose(x: Field) -> (Field, Field) { pub fn assert_gt(a: Field, b: Field) { if is_unconstrained() { - assert(unsafe { field_less_than(b, a) }); + assert( + /// Safety: already unconstrained + unsafe { field_less_than(b, a) }, + ); } else { // Decompose a and b let a_limbs = decompose(a); @@ -91,13 +98,14 @@ pub fn assert_lt(a: Field, b: Field) { pub fn gt(a: Field, b: Field) -> bool { if is_unconstrained() { + /// Safety: unsafe in unconstrained unsafe { field_less_than(b, a) } } else if a == b { false } else { - // Take a hint of the comparison and verify it + /// Safety: Take a hint of the comparison and verify it unsafe { if field_less_than(a, b) { assert_gt(b, a); diff --git a/noir/noir-repo/noir_stdlib/src/field/mod.nr b/noir/noir-repo/noir_stdlib/src/field/mod.nr index a5c1ff7bb62..d0760447ff1 100644 --- a/noir/noir-repo/noir_stdlib/src/field/mod.nr +++ b/noir/noir-repo/noir_stdlib/src/field/mod.nr @@ -237,6 +237,7 @@ pub fn bytes32_to_field(bytes32: [u8; 32]) -> Field { fn lt_fallback(x: Field, y: Field) -> bool { if is_unconstrained() { + /// Safety: unconstrained context unsafe { field_less_than(x, y) } diff --git a/noir/noir-repo/noir_stdlib/src/hash/mod.nr b/noir/noir-repo/noir_stdlib/src/hash/mod.nr index 5f7017b60b7..104368b50fa 100644 --- a/noir/noir-repo/noir_stdlib/src/hash/mod.nr +++ b/noir/noir-repo/noir_stdlib/src/hash/mod.nr @@ -96,6 +96,7 @@ fn __derive_generators( // does not assert the limbs are 128 bits // does not assert the decomposition does not overflow the EmbeddedCurveScalar fn from_field_unsafe(scalar: Field) -> EmbeddedCurveScalar { + /// Safety: xlo and xhi decomposition is checked below let (xlo, xhi) = unsafe { crate::field::bn254::decompose_hint(scalar) }; // Check that the decomposition is correct assert_eq(scalar, xlo + crate::field::bn254::TWO_POW_128 * xhi); diff --git a/noir/noir-repo/noir_stdlib/src/hash/sha256.nr b/noir/noir-repo/noir_stdlib/src/hash/sha256.nr index b9a2b02c9d9..fce263ce25d 100644 --- a/noir/noir-repo/noir_stdlib/src/hash/sha256.nr +++ b/noir/noir-repo/noir_stdlib/src/hash/sha256.nr @@ -74,6 +74,7 @@ pub fn sha256_var(msg: [u8; N], message_size: u64) -> HASH { let mut msg_byte_ptr = 0; for i in 0..num_blocks { let msg_start = BLOCK_SIZE * i; + /// Safety: the msg_block is checked below in verify_msg_block let (new_msg_block, new_msg_byte_ptr) = unsafe { build_msg_block(msg, message_size, msg_start) }; @@ -104,6 +105,7 @@ pub fn sha256_var(msg: [u8; N], message_size: u64) -> HASH { // or our message cannot be evenly split into blocks. if modulo != 0 { let msg_start = BLOCK_SIZE * num_blocks; + /// Safety: the msg_block is checked below in verify_msg_block let (new_msg_block, new_msg_byte_ptr) = unsafe { build_msg_block(msg, message_size, msg_start) }; @@ -146,6 +148,7 @@ pub fn sha256_var(msg: [u8; N], message_size: u64) -> HASH { msg_byte_ptr = 0; } + /// Safety: the msg_len is checked below in verify_msg_len msg_block = unsafe { attach_len_to_msg_block(msg_block, msg_byte_ptr, message_size) }; if !is_unconstrained() { @@ -798,6 +801,8 @@ mod tests { 101, 115, 46, 48, ]; assert_eq(input.len(), 22); + + /// Safety: testing context let (msg_block, msg_byte_ptr) = unsafe { build_msg_block(input, input.len(), 0) }; assert_eq(msg_byte_ptr, input.len()); assert_eq(msg_block[0], make_item(input[0], input[1], input[2], input[3])); @@ -815,6 +820,7 @@ mod tests { 108, 97, 105, 110, 59, 32, 99, 104, 97, 114, 115, 101, 116, ]; assert_eq(input.len(), 68); + /// Safety: test context let (msg_block, msg_byte_ptr) = unsafe { build_msg_block(input, input.len(), 64) }; assert_eq(msg_byte_ptr, 4); assert_eq(msg_block[0], make_item(input[64], input[65], input[66], input[67])); @@ -828,6 +834,7 @@ mod tests { 1919905082, 1131376244, 1701737517, 1417244773, 978151789, 1697470053, 1920166255, 1849316213, 1651139939, ]; + /// Safety: testing context let msg_block = unsafe { attach_len_to_msg_block(input, 1, 448) }; assert_eq(msg_block[0], ((1 << 7) as u32) * 256 * 256 * 256); assert_eq(msg_block[1], 0); diff --git a/noir/noir-repo/noir_stdlib/src/lib.nr b/noir/noir-repo/noir_stdlib/src/lib.nr index c3a289a8fa2..fb073516d29 100644 --- a/noir/noir-repo/noir_stdlib/src/lib.nr +++ b/noir/noir-repo/noir_stdlib/src/lib.nr @@ -28,6 +28,8 @@ pub mod mem; pub mod panic; pub mod hint; +use convert::AsPrimitive; + // Oracle calls are required to be wrapped in an unconstrained function // Thus, the only argument to the `println` oracle is expected to always be an ident #[oracle(print)] @@ -38,12 +40,14 @@ unconstrained fn print_unconstrained(with_newline: bool, input: T) { } pub fn println(input: T) { + /// Safety: a print statement cannot be constrained unsafe { print_unconstrained(true, input); } } pub fn print(input: T) { + /// Safety: a print statement cannot be constrained unsafe { print_unconstrained(false, input); } @@ -89,29 +93,37 @@ pub fn assert_constant(x: T) {} #[builtin(static_assert)] pub fn static_assert(predicate: bool, message: str) {} -// from_field and as_field are private since they are not valid for every type. -// `as` should be the default for users to cast between primitive types, and in the future -// traits can be used to work with generic types. -#[builtin(from_field)] -fn from_field(x: Field) -> T {} - -#[builtin(as_field)] -fn as_field(x: T) -> Field {} - -pub fn wrapping_add(x: T, y: T) -> T { - crate::from_field(crate::as_field(x) + crate::as_field(y)) +pub fn wrapping_add(x: T, y: T) -> T +where + T: AsPrimitive, + Field: AsPrimitive, +{ + AsPrimitive::as_(x.as_() + y.as_()) } -pub fn wrapping_sub(x: T, y: T) -> T { +pub fn wrapping_sub(x: T, y: T) -> T +where + T: AsPrimitive, + Field: AsPrimitive, +{ //340282366920938463463374607431768211456 is 2^128, it is used to avoid underflow - crate::from_field( - crate::as_field(x) + 340282366920938463463374607431768211456 - crate::as_field(y), - ) + AsPrimitive::as_(x.as_() + 340282366920938463463374607431768211456 - y.as_()) } -pub fn wrapping_mul(x: T, y: T) -> T { - crate::from_field(crate::as_field(x) * crate::as_field(y)) +pub fn wrapping_mul(x: T, y: T) -> T +where + T: AsPrimitive, + Field: AsPrimitive, +{ + AsPrimitive::as_(x.as_() * y.as_()) } #[builtin(as_witness)] pub fn as_witness(x: Field) {} + +mod tests { + #[test(should_fail_with = "custom message")] + fn test_static_assert_custom_message() { + super::static_assert(1 == 2, "custom message"); + } +} diff --git a/noir/noir-repo/noir_stdlib/src/meta/ctstring.nr b/noir/noir-repo/noir_stdlib/src/meta/ctstring.nr index b414b3418d9..e23567ece7d 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/ctstring.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/ctstring.nr @@ -86,6 +86,8 @@ comptime fn ctstring_eq(_first: CtString, _second: CtString) -> bool {} comptime fn ctstring_hash(_string: CtString) -> Field {} mod test { + use super::AsCtString; + #[test] fn as_quoted_str_example() { comptime { diff --git a/noir/noir-repo/noir_stdlib/src/meta/expr.nr b/noir/noir-repo/noir_stdlib/src/meta/expr.nr index bb795a520d3..a1663135c20 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/expr.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/expr.nr @@ -285,33 +285,31 @@ impl Expr { } comptime fn modify_array(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_array().map(|exprs: [Expr]| { + expr.as_array().map(|exprs| { let exprs = modify_expressions(exprs, f); new_array(exprs) }) } comptime fn modify_assert(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_assert().map(|expr: (Expr, Option)| { - let (predicate, msg) = expr; + expr.as_assert().map(|(predicate, msg)| { let predicate = predicate.modify(f); - let msg = msg.map(|msg: Expr| msg.modify(f)); + let msg = msg.map(|msg| msg.modify(f)); new_assert(predicate, msg) }) } comptime fn modify_assert_eq(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_assert_eq().map(|expr: (Expr, Expr, Option)| { - let (lhs, rhs, msg) = expr; + expr.as_assert_eq().map(|(lhs, rhs, msg)| { let lhs = lhs.modify(f); let rhs = rhs.modify(f); - let msg = msg.map(|msg: Expr| msg.modify(f)); + let msg = msg.map(|msg| msg.modify(f)); new_assert_eq(lhs, rhs, msg) }) } comptime fn modify_assign(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_assign().map(|expr: (Expr, Expr)| { + expr.as_assign().map(|expr| { let (lhs, rhs) = expr; let lhs = lhs.modify(f); let rhs = rhs.modify(f); @@ -320,8 +318,7 @@ comptime fn modify_assign(expr: Expr, f: fn[Env](Expr) -> Option) -> } comptime fn modify_binary_op(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_binary_op().map(|expr: (Expr, BinaryOp, Expr)| { - let (lhs, op, rhs) = expr; + expr.as_binary_op().map(|(lhs, op, rhs)| { let lhs = lhs.modify(f); let rhs = rhs.modify(f); new_binary_op(lhs, op, rhs) @@ -329,34 +326,29 @@ comptime fn modify_binary_op(expr: Expr, f: fn[Env](Expr) -> Option) } comptime fn modify_block(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_block().map(|exprs: [Expr]| { + expr.as_block().map(|exprs| { let exprs = modify_expressions(exprs, f); new_block(exprs) }) } comptime fn modify_cast(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_cast().map(|expr: (Expr, UnresolvedType)| { - let (expr, typ) = expr; + expr.as_cast().map(|(expr, typ)| { let expr = expr.modify(f); new_cast(expr, typ) }) } comptime fn modify_comptime(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_comptime().map(|exprs: [Expr]| { - let exprs = exprs.map(|expr: Expr| expr.modify(f)); + expr.as_comptime().map(|exprs| { + let exprs = exprs.map(|expr| expr.modify(f)); new_comptime(exprs) }) } comptime fn modify_constructor(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_constructor().map(|expr: (UnresolvedType, [(Quoted, Expr)])| { - let (typ, fields) = expr; - let fields = fields.map(|field: (Quoted, Expr)| { - let (name, value) = field; - (name, value.modify(f)) - }); + expr.as_constructor().map(|(typ, fields)| { + let fields = fields.map(|(name, value)| (name, value.modify(f))); new_constructor(typ, fields) }) } @@ -365,27 +357,24 @@ comptime fn modify_function_call( expr: Expr, f: fn[Env](Expr) -> Option, ) -> Option { - expr.as_function_call().map(|expr: (Expr, [Expr])| { - let (function, arguments) = expr; + expr.as_function_call().map(|(function, arguments)| { let function = function.modify(f); - let arguments = arguments.map(|arg: Expr| arg.modify(f)); + let arguments = arguments.map(|arg| arg.modify(f)); new_function_call(function, arguments) }) } comptime fn modify_if(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_if().map(|expr: (Expr, Expr, Option)| { - let (condition, consequence, alternative) = expr; + expr.as_if().map(|(condition, consequence, alternative)| { let condition = condition.modify(f); let consequence = consequence.modify(f); - let alternative = alternative.map(|alternative: Expr| alternative.modify(f)); + let alternative = alternative.map(|alternative| alternative.modify(f)); new_if(condition, consequence, alternative) }) } comptime fn modify_index(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_index().map(|expr: (Expr, Expr)| { - let (object, index) = expr; + expr.as_index().map(|(object, index)| { let object = object.modify(f); let index = index.modify(f); new_index(object, index) @@ -393,8 +382,7 @@ comptime fn modify_index(expr: Expr, f: fn[Env](Expr) -> Option) -> O } comptime fn modify_for(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_for().map(|expr: (Quoted, Expr, Expr)| { - let (identifier, array, body) = expr; + expr.as_for().map(|(identifier, array, body)| { let array = array.modify(f); let body = body.modify(f); new_for(identifier, array, body) @@ -402,8 +390,7 @@ comptime fn modify_for(expr: Expr, f: fn[Env](Expr) -> Option) -> Opt } comptime fn modify_for_range(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_for_range().map(|expr: (Quoted, Expr, Expr, Expr)| { - let (identifier, from, to, body) = expr; + expr.as_for_range().map(|(identifier, from, to, body)| { let from = from.modify(f); let to = to.modify(f); let body = body.modify(f); @@ -412,18 +399,15 @@ comptime fn modify_for_range(expr: Expr, f: fn[Env](Expr) -> Option) } comptime fn modify_lambda(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_lambda().map(|expr: ([(Expr, Option)], Option, Expr)| { - let (params, return_type, body) = expr; - let params = - params.map(|param: (Expr, Option)| (param.0.modify(f), param.1)); + expr.as_lambda().map(|(params, return_type, body)| { + let params = params.map(|(name, typ)| (name.modify(f), typ)); let body = body.modify(f); new_lambda(params, return_type, body) }) } comptime fn modify_let(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_let().map(|expr: (Expr, Option, Expr)| { - let (pattern, typ, expr) = expr; + expr.as_let().map(|(pattern, typ, expr)| { let pattern = pattern.modify(f); let expr = expr.modify(f); new_let(pattern, typ, expr) @@ -434,18 +418,16 @@ comptime fn modify_member_access( expr: Expr, f: fn[Env](Expr) -> Option, ) -> Option { - expr.as_member_access().map(|expr: (Expr, Quoted)| { - let (object, name) = expr; + expr.as_member_access().map(|(object, name)| { let object = object.modify(f); new_member_access(object, name) }) } comptime fn modify_method_call(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_method_call().map(|expr: (Expr, Quoted, [UnresolvedType], [Expr])| { - let (object, name, generics, arguments) = expr; + expr.as_method_call().map(|(object, name, generics, arguments)| { let object = object.modify(f); - let arguments = arguments.map(|arg: Expr| arg.modify(f)); + let arguments = arguments.map(|arg| arg.modify(f)); new_method_call(object, name, generics, arguments) }) } @@ -454,8 +436,7 @@ comptime fn modify_repeated_element_array( expr: Expr, f: fn[Env](Expr) -> Option, ) -> Option { - expr.as_repeated_element_array().map(|expr: (Expr, Expr)| { - let (expr, length) = expr; + expr.as_repeated_element_array().map(|(expr, length)| { let expr = expr.modify(f); let length = length.modify(f); new_repeated_element_array(expr, length) @@ -466,8 +447,7 @@ comptime fn modify_repeated_element_slice( expr: Expr, f: fn[Env](Expr) -> Option, ) -> Option { - expr.as_repeated_element_slice().map(|expr: (Expr, Expr)| { - let (expr, length) = expr; + expr.as_repeated_element_slice().map(|(expr, length)| { let expr = expr.modify(f); let length = length.modify(f); new_repeated_element_slice(expr, length) @@ -475,36 +455,35 @@ comptime fn modify_repeated_element_slice( } comptime fn modify_slice(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_slice().map(|exprs: [Expr]| { + expr.as_slice().map(|exprs| { let exprs = modify_expressions(exprs, f); new_slice(exprs) }) } comptime fn modify_tuple(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_tuple().map(|exprs: [Expr]| { + expr.as_tuple().map(|exprs| { let exprs = modify_expressions(exprs, f); new_tuple(exprs) }) } comptime fn modify_unary_op(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_unary_op().map(|expr: (UnaryOp, Expr)| { - let (op, rhs) = expr; + expr.as_unary_op().map(|(op, rhs)| { let rhs = rhs.modify(f); new_unary_op(op, rhs) }) } comptime fn modify_unsafe(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { - expr.as_unsafe().map(|exprs: [Expr]| { - let exprs = exprs.map(|expr: Expr| expr.modify(f)); + expr.as_unsafe().map(|exprs| { + let exprs = exprs.map(|expr| expr.modify(f)); new_unsafe(exprs) }) } comptime fn modify_expressions(exprs: [Expr], f: fn[Env](Expr) -> Option) -> [Expr] { - exprs.map(|expr: Expr| expr.modify(f)) + exprs.map(|expr| expr.modify(f)) } comptime fn new_array(exprs: [Expr]) -> Expr { @@ -554,12 +533,7 @@ comptime fn new_comptime(exprs: [Expr]) -> Expr { } comptime fn new_constructor(typ: UnresolvedType, fields: [(Quoted, Expr)]) -> Expr { - let fields = fields - .map(|field: (Quoted, Expr)| { - let (name, value) = field; - quote { $name: $value } - }) - .join(quote { , }); + let fields = fields.map(|(name, value)| quote { $name: $value }).join(quote { , }); quote { $typ { $fields }}.as_expr().unwrap() } @@ -590,8 +564,7 @@ comptime fn new_lambda( body: Expr, ) -> Expr { let params = params - .map(|param: (Expr, Option)| { - let (name, typ) = param; + .map(|(name, typ)| { if typ.is_some() { let typ = typ.unwrap(); quote { $name: $typ } @@ -669,9 +642,14 @@ comptime fn new_unary_op(op: UnaryOp, rhs: Expr) -> Expr { comptime fn new_unsafe(exprs: [Expr]) -> Expr { let exprs = join_expressions(exprs, quote { ; }); - quote { unsafe { $exprs }}.as_expr().unwrap() + quote { + /// Safety: generated by macro + unsafe { $exprs } + } + .as_expr() + .unwrap() } comptime fn join_expressions(exprs: [Expr], separator: Quoted) -> Quoted { - exprs.map(|expr: Expr| expr.quoted()).join(separator) + exprs.map(|expr| expr.quoted()).join(separator) } diff --git a/noir/noir-repo/noir_stdlib/src/meta/mod.nr b/noir/noir-repo/noir_stdlib/src/meta/mod.nr index 5d2164a510d..35ba05ba74d 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/mod.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/mod.nr @@ -13,6 +13,8 @@ pub mod typed_expr; pub mod quoted; pub mod unresolved_type; +use crate::default::Default; + /// Calling unquote as a macro (via `unquote!(arg)`) will unquote /// its argument. Since this is the effect `!` already does, `unquote` /// itself does not need to do anything besides return its argument. @@ -52,7 +54,7 @@ pub comptime fn derive(s: StructDefinition, traits: [TraitDefinition]) -> Quoted let mut result = quote {}; for trait_to_derive in traits { - let handler = unsafe { HANDLERS.get(trait_to_derive) }; + let handler = HANDLERS.get(trait_to_derive); assert(handler.is_some(), f"No derive function registered for `{trait_to_derive}`"); let trait_impl = handler.unwrap()(s); @@ -95,14 +97,22 @@ pub comptime fn make_trait_impl( ) -> Quoted { // docs:end:make_trait_impl let typ = s.as_type(); - let impl_generics = s.generics().map(|g| quote { $g }).join(quote {,}); - let where_clause = s.generics().map(|name| quote { $name: $trait_name }).join(quote {,}); + + let mut impl_generics = &[]; + let mut where_clause = &[]; + for g in s.generics() { + let (typ, numeric_type) = g; + impl_generics = impl_generics.push_back(quote { $typ }); + if numeric_type.is_none() { + where_clause = where_clause.push_back(quote { $typ: $trait_name }); + } + } + + let impl_generics = impl_generics.join(quote {, }); + let where_clause = where_clause.join(quote {, }); // `for_each_field(field1) $join_fields_with for_each_field(field2) $join_fields_with ...` - let fields = s.fields().map(|f: (Quoted, Type)| { - let name = f.0; - for_each_field(name) - }); + let fields = s.fields_as_written().map(|(name, _)| for_each_field(name)); let body = body(fields.join(join_fields_with)); quote { @@ -115,6 +125,7 @@ pub comptime fn make_trait_impl( } mod tests { + use crate::meta::ctstring::AsCtString; use crate::meta::derive_via; // docs:start:quote-example @@ -152,7 +163,7 @@ mod tests { comptime fn derive_field_count(s: StructDefinition) -> Quoted { let typ = s.as_type(); - let field_count = s.fields().len(); + let field_count = s.fields_as_written().len(); quote { impl FieldCount for $typ { fn field_count() -> u32 { @@ -171,7 +182,7 @@ mod tests { comptime fn assert_field_is_type(s: StructDefinition, typ: Type) { // Assert the first field in `s` has type `typ` - let fields = s.fields(); + let fields = s.fields([]); assert_eq(fields[0].1, typ); } // docs:end:annotation-arguments-example diff --git a/noir/noir-repo/noir_stdlib/src/meta/op.nr b/noir/noir-repo/noir_stdlib/src/meta/op.nr index afe230429f8..67c9b4cdb64 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/op.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/op.nr @@ -1,3 +1,5 @@ +use crate::hash::Hash; + pub struct UnaryOp { op: Field, } diff --git a/noir/noir-repo/noir_stdlib/src/meta/struct_def.nr b/noir/noir-repo/noir_stdlib/src/meta/struct_def.nr index ba5d0289e73..d561e326fbd 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/struct_def.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/struct_def.nr @@ -1,3 +1,5 @@ +use crate::option::Option; + impl StructDefinition { #[builtin(struct_def_add_attribute)] // docs:start:add_attribute @@ -21,19 +23,30 @@ impl StructDefinition { pub comptime fn has_named_attribute(self, name: str) -> bool {} // docs:end:has_named_attribute - /// Return each generic on this struct. + /// Return (type, option) pairs of each generic in this struct definition. + /// If a generic is numeric, the second element of the pair will contain the numeric type. #[builtin(struct_def_generics)] // docs:start:generics - pub comptime fn generics(self) -> [Type] {} + pub comptime fn generics(self) -> [(Type, Option)] {} // docs:end:generics - /// Returns (name, type) pairs of each field in this struct. Each type is as-is - /// with any generic arguments unchanged. + /// Returns (name, type) pairs of each field in this struct. + /// Any generic types used in each field type is automatically substituted with the + /// provided generic arguments. #[builtin(struct_def_fields)] // docs:start:fields - pub comptime fn fields(self) -> [(Quoted, Type)] {} + pub comptime fn fields(self, generic_args: [Type]) -> [(Quoted, Type)] {} // docs:end:fields + /// Returns (name, type) pairs of each field in this struct. Each type is as-is + /// with any generic arguments unchanged. Unless the field types are not needed, + /// users should generally prefer to use `StructDefinition::fields` over this + /// function if possible. + #[builtin(struct_def_fields_as_written)] + // docs:start:fields_as_written + pub comptime fn fields_as_written(self) -> [(Quoted, Type)] {} + // docs:end:fields_as_written + #[builtin(struct_def_module)] // docs:start:module pub comptime fn module(self) -> Module {} diff --git a/noir/noir-repo/noir_stdlib/src/uint128.nr b/noir/noir-repo/noir_stdlib/src/uint128.nr index 06febb3ee00..9aa01e1ea52 100644 --- a/noir/noir-repo/noir_stdlib/src/uint128.nr +++ b/noir/noir-repo/noir_stdlib/src/uint128.nr @@ -1,5 +1,6 @@ use crate::cmp::{Eq, Ord, Ordering}; use crate::ops::{Add, BitAnd, BitOr, BitXor, Div, Mul, Not, Rem, Shl, Shr, Sub}; +use super::{convert::AsPrimitive, default::Default}; global pow64: Field = 18446744073709551616; //2^64; global pow63: Field = 9223372036854775808; // 2^63; @@ -103,6 +104,7 @@ impl U128 { if ascii < 58 { ascii - 48 } else { + /// Safety: optionally adds 32 and then check (below) the result is in 'a..f' range let ascii = ascii + 32 * (unsafe { U128::uconstrained_check_is_upper_ascii(ascii) as u8 }); assert(ascii >= 97); // enforce >= 'a' @@ -139,8 +141,11 @@ impl U128 { } } - pub fn from_integer(i: T) -> U128 { - let f = crate::as_field(i); + pub fn from_integer(i: T) -> U128 + where + T: AsPrimitive, + { + let f = i.as_(); // Reject values which would overflow a u128 f.assert_max_bit_size::<128>(); let lo = f as u64 as Field; @@ -148,8 +153,11 @@ impl U128 { U128 { lo, hi } } - pub fn to_integer(self) -> T { - crate::from_field(self.lo + self.hi * pow64) + pub fn to_integer(self) -> T + where + Field: AsPrimitive, + { + AsPrimitive::as_(self.lo + self.hi * pow64) } fn wrapping_mul(self: Self, b: U128) -> U128 { @@ -205,6 +213,8 @@ impl Mul for U128 { impl Div for U128 { fn div(self: Self, b: U128) -> U128 { + /// Safety: euclidian division is asserted to be correct: assert(a == b * q + r); and assert(r < b); + /// Furthermore, U128 addition and multiplication ensures that b * q + r does not overflow unsafe { let (q, r) = self.unconstrained_div(b); let a = b * q + r; @@ -217,6 +227,7 @@ impl Div for U128 { impl Rem for U128 { fn rem(self: Self, b: U128) -> U128 { + /// Safety: cf div() above unsafe { let (q, r) = self.unconstrained_div(b); let a = b * q + r; @@ -312,7 +323,15 @@ impl Shr for U128 { } } +impl Default for U128 { + fn default() -> Self { + U128::zero() + } +} + mod tests { + use crate::default::Default; + use crate::ops::Not; use crate::uint128::{pow63, pow64, U128}; #[test] @@ -436,6 +455,7 @@ mod tests { let b = U128::from_u64s_le(0x0, 0xfffffffffffffffe); let c = U128::one(); let d = U128::from_u64s_le(0x0, 0x1); + /// Safety: testing context unsafe { let (q, r) = a.unconstrained_div(b); assert_eq(q, c); @@ -445,12 +465,14 @@ mod tests { let a = U128::from_u64s_le(2, 0); let b = U128::one(); // Check the case where a is a multiple of b + /// Safety: testing context unsafe { let (c, d) = a.unconstrained_div(b); assert_eq((c, d), (a, U128::zero())); } // Check where b is a multiple of a + /// Safety: testing context unsafe { let (c, d) = b.unconstrained_div(a); assert_eq((c, d), (U128::zero(), b)); @@ -459,6 +481,7 @@ mod tests { // Dividing by zero returns 0,0 let a = U128::from_u64s_le(0x1, 0x0); let b = U128::zero(); + /// Safety: testing context unsafe { let (c, d) = a.unconstrained_div(b); assert_eq((c, d), (U128::zero(), U128::zero())); @@ -466,6 +489,7 @@ mod tests { // Dividing 1<<127 by 1<<127 (special case) let a = U128::from_u64s_le(0x0, pow63 as u64); let b = U128::from_u64s_le(0x0, pow63 as u64); + /// Safety: testing context unsafe { let (c, d) = a.unconstrained_div(b); assert_eq((c, d), (U128::one(), U128::zero())); @@ -540,4 +564,9 @@ mod tests { ), ); } + + #[test] + fn test_default() { + assert_eq(U128::default(), U128::zero()); + } } diff --git a/noir/noir-repo/rust-toolchain.toml b/noir/noir-repo/rust-toolchain.toml index fe2949c8458..e647d5cbf46 100644 --- a/noir/noir-repo/rust-toolchain.toml +++ b/noir/noir-repo/rust-toolchain.toml @@ -1,5 +1,5 @@ [toolchain] -channel = "1.74.1" +channel = "1.75.0" components = [ "rust-src" ] targets = [ "wasm32-unknown-unknown", "wasm32-wasi", "aarch64-apple-darwin" ] profile = "default" diff --git a/noir/noir-repo/test_programs/compilation_report.sh b/noir/noir-repo/test_programs/compilation_report.sh index d050e7c9c34..786dbd75fe8 100755 --- a/noir/noir-repo/test_programs/compilation_report.sh +++ b/noir/noir-repo/test_programs/compilation_report.sh @@ -1,6 +1,7 @@ #!/usr/bin/env bash set -e +PARSE_TIME=$(realpath "$(dirname "$0")/parse_time.sh") current_dir=$(pwd) base_path="$current_dir/execution_success" @@ -10,13 +11,14 @@ tests_to_profile=("sha256_regression" "regression_4709" "ram_blowup_regression") echo "{\"compilation_reports\": [ " > $current_dir/compilation_report.json # If there is an argument that means we want to generate a report for only the current directory -if [ "$#" -ne 0 ]; then +if [ "$1" == "1" ]; then base_path="$current_dir" tests_to_profile=(".") fi ITER="1" NUM_ARTIFACTS=${#tests_to_profile[@]} +FLAGS=${FLAGS:- ""} for dir in ${tests_to_profile[@]}; do if [[ " ${excluded_dirs[@]} " =~ " ${dir} " ]]; then @@ -32,19 +34,42 @@ for dir in ${tests_to_profile[@]}; do # The default package to run is the supplied list hardcoded at the top of the script PACKAGE_NAME=$dir # Otherwise default to the current directory as the package we want to run - if [ "$#" -ne 0 ]; then + if [ "$1" == "1" ]; then PACKAGE_NAME=$(basename $current_dir) fi - COMPILE_TIME=$((time nargo compile --force --silence-warnings) 2>&1 | grep real | grep -oE '[0-9]+m[0-9]+.[0-9]+s') - echo -e " {\n \"artifact_name\":\"$PACKAGE_NAME\",\n \"time\":\"$COMPILE_TIME\"" >> $current_dir/compilation_report.json - - if (($ITER == $NUM_ARTIFACTS)); then - echo "}" >> $current_dir/compilation_report.json - else - echo "}, " >> $current_dir/compilation_report.json + NUM_RUNS=$2 + TOTAL_TIME=0 + + for ((i = 1; i <= NUM_RUNS; i++)); do + NOIR_LOG=trace NARGO_LOG_DIR=./tmp nargo compile --force --silence-warnings $FLAGS + done + + TIMES=($(jq -r '. | select(.target == "nargo::cli" and .fields.message == "close") | .fields."time.busy"' ./tmp/*)) + + AVG_TIME=$(awk -v RS=" " -v parse_time="$PARSE_TIME" ' + { + # Times are formatted annoyingly so we need to parse it. + parse_time" "$1 | getline current_time + close(parse_time" "$1) + sum += current_time; + n++; + } + END { + if (n > 0) + printf "%.3f\n", sum / n + else + printf "%.3f\n", 0 + }' <<<"${TIMES[@]}") + + jq -rc "{artifact_name: \"$PACKAGE_NAME\", time: \""$AVG_TIME"s\"}" --null-input >> $current_dir/compilation_report.json + + if (($ITER != $NUM_ARTIFACTS)); then + echo "," >> $current_dir/compilation_report.json fi + rm -rf ./tmp + ITER=$(( $ITER + 1 )) done diff --git a/noir/noir-repo/test_programs/compile_failure/brillig_mut_ref_from_acir/src/main.nr b/noir/noir-repo/test_programs/compile_failure/brillig_mut_ref_from_acir/src/main.nr index f262635e508..d4450ec62b1 100644 --- a/noir/noir-repo/test_programs/compile_failure/brillig_mut_ref_from_acir/src/main.nr +++ b/noir/noir-repo/test_programs/compile_failure/brillig_mut_ref_from_acir/src/main.nr @@ -3,8 +3,7 @@ unconstrained fn mut_ref_identity(value: &mut Field) -> Field { } fn main(mut x: Field, y: pub Field) { - let returned_x = unsafe { - mut_ref_identity(&mut x) - }; + /// Safety: testing context + let returned_x = unsafe { mut_ref_identity(&mut x) }; assert(returned_x == x); } diff --git a/noir/noir-repo/test_programs/compile_failure/regression_5008/src/main.nr b/noir/noir-repo/test_programs/compile_failure/regression_5008/src/main.nr index cf79c22683f..5155900734c 100644 --- a/noir/noir-repo/test_programs/compile_failure/regression_5008/src/main.nr +++ b/noir/noir-repo/test_programs/compile_failure/regression_5008/src/main.nr @@ -2,7 +2,7 @@ struct Bar { value: Field, } -struct Foo{ +struct Foo { bar: &mut Bar, } @@ -13,7 +13,6 @@ impl Foo { fn main() { let foo = Foo { bar: &mut Bar { value: 0 } }; - unsafe { - foo.crash_fn() - }; + /// Safety: testing context + unsafe { foo.crash_fn() }; } diff --git a/noir/noir-repo/test_programs/compile_failure/unconstrained_ref/src/main.nr b/noir/noir-repo/test_programs/compile_failure/unconstrained_ref/src/main.nr index 9a4f42469b5..23dc1d66673 100644 --- a/noir/noir-repo/test_programs/compile_failure/unconstrained_ref/src/main.nr +++ b/noir/noir-repo/test_programs/compile_failure/unconstrained_ref/src/main.nr @@ -4,7 +4,6 @@ unconstrained fn uncon_ref() -> &mut Field { } fn main() { - let e = unsafe { - uncon_ref() - }; + /// Safety: testing context + let e = unsafe { uncon_ref() }; } diff --git a/noir/noir-repo/test_programs/execution_success/acir_inside_brillig_recursion/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/acir_inside_brillig_recursion/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/acir_inside_brillig_recursion/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/acir_inside_brillig_recursion/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/acir_inside_brillig_recursion/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/acir_inside_brillig_recursion/src/main.nr similarity index 89% rename from noir/noir-repo/test_programs/execution_success/acir_inside_brillig_recursion/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/acir_inside_brillig_recursion/src/main.nr index 49b7c00b6b9..12c663b2705 100644 --- a/noir/noir-repo/test_programs/execution_success/acir_inside_brillig_recursion/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/acir_inside_brillig_recursion/src/main.nr @@ -1,4 +1,5 @@ fn main() { + /// Safety: testing context unsafe { assert_eq(fibonacci(3), fibonacci_hint(3)); } diff --git a/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/Nargo.toml new file mode 100644 index 00000000000..6b54cbfca6a --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/Nargo.toml @@ -0,0 +1,6 @@ +[package] +name = "associated_types_implicit" +type = "bin" +authors = [""] + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/Prover.toml b/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/Prover.toml new file mode 100644 index 00000000000..cab679b4b66 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/Prover.toml @@ -0,0 +1 @@ +a = [[0, 1], [2, 3]] diff --git a/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/src/main.nr new file mode 100644 index 00000000000..d04cef51868 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/associated_types_implicit/src/main.nr @@ -0,0 +1,60 @@ +trait Foo { + type Bar; + + fn foo(self) -> Self::Bar; +} + +impl Foo for u64 { + type Bar = u8; + + fn foo(self) -> Self::Bar { + self as u8 + } +} + +fn main() { + // This currently requires a type annotation to find the impl + let three: u64 = 3; + call_foo(three); + + let x: Option> = Option::some(Option::some(0)); + let x_foo = x.foo(); + assert_eq(x_foo, x_foo); // ensure we don't need an additional type annotation for Bar here + + // The `as u8` is still necessary even though we know the object type, + // otherwise we try to search for `u64: Foo`. + // It seems the `Bar = u8` constraint is still there & checked for, but + // the defaulting of the polymorphic integer occurs first. + assert_eq(x.foo(), 0 as u8); +} + +// Ensure we can use `::Bar: Eq` in a function's where clause +fn call_foo(x: T) +where + T: Foo, + ::Bar: Eq, +{ + let y = x.foo(); + assert_eq(y, y); +} + +// Ensure we can use `::Bar: Eq` in a trait impl's where clause +impl Foo for Option +where + T: Foo, + ::Bar: Eq, +{ + type Bar = ::Bar; + + fn foo(self) -> Self::Bar { + self.unwrap().foo() + } +} + +// Ensure we can use `::Bar: Eq` in a trait's where clause +// TODO: Not working, see issue #7024 +// trait Baz where +// T: Foo, +// ::Bar: Eq {} +// +// impl Baz for u32 {} diff --git a/noir/noir-repo/test_programs/compile_success_empty/brillig_cast/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/brillig_cast/src/main.nr index f1a6814f4e7..d7d0b1b8fa5 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/brillig_cast/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/brillig_cast/src/main.nr @@ -2,6 +2,7 @@ // // The features being tested are cast operations on brillig fn main() { + /// Safety: testing context unsafe { bool_casts(); field_casts(); diff --git a/noir/noir-repo/test_programs/compile_success_empty/brillig_field_binary_operations/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/brillig_field_binary_operations/src/main.nr index fedfe48cb0d..f55beb6d1a8 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/brillig_field_binary_operations/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/brillig_field_binary_operations/src/main.nr @@ -1,5 +1,6 @@ // Tests arithmetic operations on fields fn main() { + /// Safety: testing context unsafe { let x = 4; let y = 2; diff --git a/noir/noir-repo/test_programs/compile_success_empty/brillig_integer_binary_operations/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/brillig_integer_binary_operations/src/main.nr index 7ecc21dbd2f..0e00f25e5bb 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/brillig_integer_binary_operations/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/brillig_integer_binary_operations/src/main.nr @@ -3,6 +3,7 @@ fn main() { let x: u32 = 6; let y: u32 = 2; + /// Safety: testing context unsafe { assert((x + y) == add(x, y)); diff --git a/noir/noir-repo/test_programs/compile_success_empty/brillig_modulo/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/brillig_modulo/src/main.nr index 2ad43bdb7eb..841d7d2dfca 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/brillig_modulo/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/brillig_modulo/src/main.nr @@ -2,6 +2,7 @@ // // The features being tested is modulo operations on brillig fn main() { + /// Safety: testing context unsafe { assert(modulo(47, 3) == 2); assert(modulo(2, 3) == 2); diff --git a/noir/noir-repo/test_programs/compile_success_empty/brillig_slice_input/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/brillig_slice_input/src/main.nr index 436a939767e..c43c5fae162 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/brillig_slice_input/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/brillig_slice_input/src/main.nr @@ -16,12 +16,14 @@ unconstrained fn sum_slice(slice: [[Point; 2]]) -> Field { fn main() { let mut slice = &[]; slice = slice.push_back([Point { x: 13, y: 14 }, Point { x: 20, y: 8 }]); + /// Safety: testing context unsafe { let brillig_sum = sum_slice(slice); assert_eq(brillig_sum, 55); } slice = slice.push_back([Point { x: 15, y: 5 }, Point { x: 12, y: 13 }]); + /// Safety: testing context unsafe { let brillig_sum = sum_slice(slice); assert_eq(brillig_sum, 100); diff --git a/noir/noir-repo/test_programs/execution_success/cast_and_shift_global/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/cast_and_shift_global/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/cast_and_shift_global/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/cast_and_shift_global/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/cast_and_shift_global/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/cast_and_shift_global/src/main.nr similarity index 100% rename from noir/noir-repo/test_programs/execution_success/cast_and_shift_global/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/cast_and_shift_global/src/main.nr diff --git a/noir/noir-repo/test_programs/execution_success/check_large_field_bits/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/check_large_field_bits/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/check_large_field_bits/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/check_large_field_bits/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/check_large_field_bits/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/check_large_field_bits/src/main.nr similarity index 100% rename from noir/noir-repo/test_programs/execution_success/check_large_field_bits/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/check_large_field_bits/src/main.nr diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_as_field/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/comptime_as_field/Nargo.toml new file mode 100644 index 00000000000..294832ba329 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_as_field/Nargo.toml @@ -0,0 +1,6 @@ +[package] +name = "comptime_as_field" +type = "bin" +authors = [""] + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_as_field/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_as_field/src/main.nr new file mode 100644 index 00000000000..e13db54b80b --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_as_field/src/main.nr @@ -0,0 +1,6 @@ +fn main() { + comptime { + let _: U128 = U128::from_integer(1); + } +} + diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_change_type_each_iteration/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_change_type_each_iteration/src/main.nr index 976987fec9e..153eb308911 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/comptime_change_type_each_iteration/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_change_type_each_iteration/src/main.nr @@ -1,3 +1,5 @@ +use std::meta::ctstring::AsCtString; + fn main() { comptime { for i in 9..11 { diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_from_field/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/comptime_from_field/Nargo.toml new file mode 100644 index 00000000000..38a46ba0dbe --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_from_field/Nargo.toml @@ -0,0 +1,6 @@ +[package] +name = "comptime_from_field" +type = "bin" +authors = [""] + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_from_field/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_from_field/src/main.nr new file mode 100644 index 00000000000..722c5c7eb32 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_from_field/src/main.nr @@ -0,0 +1,6 @@ +fn main() { + comptime { + let _: Field = U128::from_hex("0x0").to_integer(); + } +} + diff --git a/noir/noir-repo/test_programs/execution_success/comptime_slice_equality/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/comptime_slice_equality/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/comptime_slice_equality/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/comptime_slice_equality/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/comptime_slice_equality/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_slice_equality/src/main.nr similarity index 100% rename from noir/noir-repo/test_programs/execution_success/comptime_slice_equality/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/comptime_slice_equality/src/main.nr diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_struct_definition/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_struct_definition/src/main.nr index 686ac26025d..41ba76171a8 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/comptime_struct_definition/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_struct_definition/src/main.nr @@ -12,7 +12,7 @@ pub struct I32AndField { comptime fn my_comptime_fn(typ: StructDefinition) { let _ = typ.as_type(); assert_eq(typ.generics().len(), 3); - assert_eq(typ.fields().len(), 2); + assert_eq(typ.fields_as_written().len(), 2); assert_eq(typ.name(), quote { MyType }); } @@ -39,9 +39,29 @@ mod foo { let generics = s.generics(); assert_eq(generics.len(), 1); - assert_eq(generics[0], new_generic); + let (typ, numeric) = generics[0]; + assert_eq(typ, new_generic); + assert(numeric.is_none()); } // docs:end:add-generic-example } -fn main() {} +fn main() { + comptime { + let typ = quote { MyType }.as_type(); + let (struct_def, generics) = typ.as_struct().unwrap(); + + let fields = struct_def.fields(generics); + assert_eq(fields.len(), 2); + + let (field1_name, field1_type) = fields[0]; + let (field2_name, field2_type) = fields[1]; + + assert_eq(field1_name, quote { field1 }); + assert_eq(field2_name, quote { field2 }); + + // Ensure .fields(generics) actually performs substitutions on generics + assert_eq(field1_type, quote { [i8; 10] }.as_type()); + assert_eq(field2_type, quote { (i16, i32) }.as_type()); + } +} diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_trait_constraint/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_trait_constraint/src/main.nr index 2f24675e8c7..39ac1b5cde1 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/comptime_trait_constraint/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_trait_constraint/src/main.nr @@ -1,4 +1,4 @@ -use std::hash::Hasher; +use std::hash::{Hash, Hasher}; trait TraitWithGenerics { fn foo(self) -> (A, B); diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_type/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_type/src/main.nr index 887690cb6cb..6a2453ff0f2 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/comptime_type/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_type/src/main.nr @@ -100,7 +100,7 @@ fn main() { let foo = Foo { x: 0 }; let foo_type = type_of(foo); let (struct_definition, generics) = foo_type.as_struct().unwrap(); - let fields = struct_definition.fields(); + let fields = struct_definition.fields(generics); assert_eq(fields.len(), 1); assert_eq(generics.len(), 1); diff --git a/noir/noir-repo/test_programs/compile_success_empty/ctstring/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/ctstring/src/main.nr index 61cd848a436..818ae7f94e8 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/ctstring/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/ctstring/src/main.nr @@ -1,3 +1,5 @@ +use std::meta::ctstring::AsCtString; + fn main() { comptime { let msg1 = "msg1"; diff --git a/noir/noir-repo/test_programs/compile_success_empty/derive_impl/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/derive_impl/src/main.nr index 4396169235d..fe7a7140280 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/derive_impl/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/derive_impl/src/main.nr @@ -7,7 +7,7 @@ comptime fn derive_default(typ: StructDefinition) -> Quoted { ); let type_name = typ.as_type(); - let fields = typ.fields(); + let fields = typ.fields_as_written(); let fields = join(make_field_exprs(fields)); diff --git a/noir/noir-repo/test_programs/compile_success_empty/double_generic_alias_in_path/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/double_generic_alias_in_path/Nargo.toml new file mode 100644 index 00000000000..aaebee8d6ef --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/double_generic_alias_in_path/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "double_generic_alias_in_path" +type = "bin" +authors = [""] +compiler_version = ">=0.32.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/double_generic_alias_in_path/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/double_generic_alias_in_path/src/main.nr new file mode 100644 index 00000000000..09f2e5c4b43 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/double_generic_alias_in_path/src/main.nr @@ -0,0 +1,14 @@ +struct Foo {} + +impl Foo { + fn new() -> Self { + Self {} + } +} + +type FooAlias1 = Foo; +type FooAlias2 = FooAlias1; + +fn main() { + let _ = FooAlias2::new(); +} diff --git a/noir/noir-repo/test_programs/execution_success/empty/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/empty/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/empty/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/empty/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/empty/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/empty/src/main.nr similarity index 100% rename from noir/noir-repo/test_programs/execution_success/empty/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/empty/src/main.nr diff --git a/noir/noir-repo/test_programs/compile_success_empty/eq_derivation_with_numeric_generics/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/eq_derivation_with_numeric_generics/Nargo.toml new file mode 100644 index 00000000000..14b4d5eff4a --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/eq_derivation_with_numeric_generics/Nargo.toml @@ -0,0 +1,6 @@ +[package] +name = "eq_derivation_with_numeric_generics" +type = "bin" +authors = [""] + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/eq_derivation_with_numeric_generics/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/eq_derivation_with_numeric_generics/src/main.nr new file mode 100644 index 00000000000..3307aeb15ad --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/eq_derivation_with_numeric_generics/src/main.nr @@ -0,0 +1,12 @@ +#[derive(Eq)] +struct Foo { + a: [Field; T], + b: u32, +} + +fn main() { + let foo = Foo { a: [0; 10], b: 27 }; + let bar = Foo { a: [0; 10], b: 28 }; + assert(foo != bar); +} + diff --git a/noir/noir-repo/test_programs/compile_success_empty/inject_context_attribute/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/inject_context_attribute/src/main.nr index 963d4cea969..e682ea34b23 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/inject_context_attribute/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/inject_context_attribute/src/main.nr @@ -40,19 +40,16 @@ comptime fn inject_context(f: FunctionDefinition) { } comptime fn mapping_function(expr: Expr, f: FunctionDefinition) -> Option { - expr.as_function_call().and_then(|func_call: (Expr, [Expr])| { - let (name, arguments) = func_call; - name.resolve(Option::some(f)).as_function_definition().and_then( - |function_definition: FunctionDefinition| { - if function_definition.has_named_attribute("inject_context") { - let arguments = arguments.push_front(quote { _context }.as_expr().unwrap()); - let arguments = arguments.map(|arg: Expr| arg.quoted()).join(quote { , }); - Option::some(quote { $name($arguments) }.as_expr().unwrap()) - } else { - Option::none() - } - }, - ) + expr.as_function_call().and_then(|(name, arguments)| { + name.resolve(Option::some(f)).as_function_definition().and_then(|function_definition| { + if function_definition.has_named_attribute("inject_context") { + let arguments = arguments.push_front(quote { _context }.as_expr().unwrap()); + let arguments = arguments.map(|arg| arg.quoted()).join(quote { , }); + Option::some(quote { $name($arguments) }.as_expr().unwrap()) + } else { + Option::none() + } + }) }) } diff --git a/noir/noir-repo/test_programs/execution_success/is_unconstrained/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/is_unconstrained/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/is_unconstrained/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/is_unconstrained/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/is_unconstrained/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/is_unconstrained/src/main.nr similarity index 90% rename from noir/noir-repo/test_programs/execution_success/is_unconstrained/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/is_unconstrained/src/main.nr index d06366cf642..856040b2274 100644 --- a/noir/noir-repo/test_programs/execution_success/is_unconstrained/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/is_unconstrained/src/main.nr @@ -9,6 +9,7 @@ unconstrained fn unconstrained_intermediate() { } fn main() { + /// Safety: testing context unsafe { unconstrained_intermediate(); } diff --git a/noir/noir-repo/test_programs/compile_success_empty/macros_in_comptime/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/macros_in_comptime/src/main.nr index 799091fca09..112ed16c22a 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/macros_in_comptime/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/macros_in_comptime/src/main.nr @@ -6,7 +6,7 @@ global three_field: Field = 3; fn main() { comptime { - unsafe { foo::(5) }; + foo::(5); submodule::bar(); } } @@ -33,8 +33,8 @@ comptime fn foo(x: Field) { break; } - let loop = quote { for _ in 0..0 { break; } }; - unquote!(loop); + let loop_ = quote { for _ in 0..0 { break; } }; + unquote!(loop_); } mod submodule { diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_5428/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/regression_5428/src/main.nr index f01b89cbea4..e9e6c8a5dc8 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/regression_5428/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_5428/src/main.nr @@ -1,3 +1,5 @@ +use std::append::Append; + fn main() { assert_true!(); } diff --git a/noir/noir-repo/test_programs/execution_success/regression_5462/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/regression_5462/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/regression_5462/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/regression_5462/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/regression_5462/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/regression_5462/src/main.nr similarity index 100% rename from noir/noir-repo/test_programs/execution_success/regression_5462/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/regression_5462/src/main.nr diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_7038/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/regression_7038/Nargo.toml new file mode 100644 index 00000000000..3c874d4b6e8 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_7038/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "regression_7038" +type = "bin" +authors = [""] +compiler_version = ">=0.30.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_7038/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/regression_7038/src/main.nr new file mode 100644 index 00000000000..793a3f60807 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_7038/src/main.nr @@ -0,0 +1,40 @@ +trait BigNumTrait {} + +pub struct MyBigNum; + +impl crate::BigNumTrait for MyBigNum {} + +trait CurveParamsTrait +where + BigNum: BigNumTrait, +{ + fn one(); +} + +pub struct BN254Params; +impl CurveParamsTrait for BN254Params { + + fn one() {} +} + +trait BigCurveTrait { + fn two(); +} + +pub struct BigCurve {} + +type BN254 = BigCurve; + +impl BigCurveTrait for BigCurve +where + BigNum: BigNumTrait, + CurveParams: CurveParamsTrait, +{ + fn two() { + let _ = CurveParams::one(); + } +} + +fn main() { + let _ = BN254::two(); +} diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_7038_2/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_2/Nargo.toml new file mode 100644 index 00000000000..f4f23683eb8 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_2/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "regression_7038_2" +type = "bin" +authors = [""] +compiler_version = ">=0.30.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_7038_2/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_2/src/main.nr new file mode 100644 index 00000000000..6a116bb0722 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_2/src/main.nr @@ -0,0 +1,39 @@ +trait BigNumTrait {} + +pub struct MyBigNum; + +impl crate::BigNumTrait for MyBigNum {} + +trait CurveParamsTrait +where + BigNum: BigNumTrait, +{ + // The difference between this and regression_7083 is that here + // this is a default method. + fn one() {} +} + +pub struct BN254Params; +impl CurveParamsTrait for BN254Params {} + +trait BigCurveTrait { + fn two(); +} + +pub struct BigCurve {} + +type BN254 = BigCurve; + +impl BigCurveTrait for BigCurve +where + BigNum: BigNumTrait, + CurveParams: CurveParamsTrait, +{ + fn two() { + let _ = CurveParams::one(); + } +} + +fn main() { + let _ = BN254::two(); +} diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_7038_3/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_3/Nargo.toml new file mode 100644 index 00000000000..65bc946c559 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_3/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "regression_7038_3" +type = "bin" +authors = [""] +compiler_version = ">=0.30.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_7038_3/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_3/src/main.nr new file mode 100644 index 00000000000..1b6bf0b72d5 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_3/src/main.nr @@ -0,0 +1,40 @@ +trait BigNumTrait {} + +pub struct MyBigNum; + +impl crate::BigNumTrait for MyBigNum {} + +trait CurveParamsTrait { + // The difference between this and regression_7038 and regression_7038_2 is that + // here the where clause is on the method, not the trait + fn one() + where + BigNum: BigNumTrait; +} + +pub struct BN254Params; +impl CurveParamsTrait for BN254Params { + fn one() {} +} + +trait BigCurveTrait { + fn two(); +} + +pub struct BigCurve {} + +type BN254 = BigCurve; + +impl BigCurveTrait for BigCurve +where + BigNum: BigNumTrait, + CurveParams: CurveParamsTrait, +{ + fn two() { + let _ = CurveParams::one(); + } +} + +fn main() { + let _ = BN254::two(); +} diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_7038_4/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_4/Nargo.toml new file mode 100644 index 00000000000..435c8094c70 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_4/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "regression_7038_4" +type = "bin" +authors = [""] +compiler_version = ">=0.30.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/regression_7038_4/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_4/src/main.nr new file mode 100644 index 00000000000..524f05a5022 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/regression_7038_4/src/main.nr @@ -0,0 +1,29 @@ +// This program is a reduction of regression_7038_3 that led to a monomorphizer crash +trait BigNumTrait {} + +trait CurveParamsTrait { + fn one() + where + BigNum: BigNumTrait; +} + +pub struct MyBigNum; + +impl BigNumTrait for MyBigNum {} + +pub struct Params; +impl CurveParamsTrait for Params { + + fn one() {} +} + +fn foo() +where + C: CurveParamsTrait, +{ + let _ = C::one(); +} + +fn main() { + foo::(); +} diff --git a/noir/noir-repo/test_programs/compile_success_empty/serialize/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/serialize/src/main.nr index 66c79f9fc9d..a11fdf570d0 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/serialize/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/serialize/src/main.nr @@ -5,13 +5,12 @@ trait Serialize { fn serialize(self) -> [Field; Self::Size]; } -impl Serialize for (A, B) +impl Serialize for (A, B) where - A: Serialize, - B: Serialize, + A: Serialize, + B: Serialize, { - // let Size = ::Size + ::Size; - let Size: u32 = AS + BS; + let Size = ::Size + ::Size; fn serialize(self: Self) -> [Field; Self::Size] { let mut array: [Field; Self::Size] = std::mem::zeroed(); @@ -28,12 +27,11 @@ where } } -impl Serialize for [T; N] +impl Serialize for [T; N] where - T: Serialize, + T: Serialize, { - // let Size = ::Size * N; - let Size: u32 = TS * N; + let Size = ::Size * N; fn serialize(self: Self) -> [Field; Self::Size] { let mut array: [Field; Self::Size] = std::mem::zeroed(); diff --git a/noir/noir-repo/test_programs/compile_success_empty/trait_default_method_uses_op/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/trait_default_method_uses_op/Nargo.toml new file mode 100644 index 00000000000..d8b882c1696 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/trait_default_method_uses_op/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "trait_default_method_uses_op" +type = "bin" +authors = [""] +compiler_version = ">=0.32.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_empty/trait_default_method_uses_op/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/trait_default_method_uses_op/src/main.nr new file mode 100644 index 00000000000..415f36fe207 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/trait_default_method_uses_op/src/main.nr @@ -0,0 +1,7 @@ +pub trait Foo: Eq { + fn foo(self) -> bool { + self == self + } +} + +fn main() {} diff --git a/noir/noir-repo/test_programs/compile_success_empty/trait_function_calls/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/trait_function_calls/src/main.nr index f9a338bfa47..bdc4a3d8c90 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/trait_function_calls/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/trait_function_calls/src/main.nr @@ -23,25 +23,32 @@ // 1a) trait default method -> trait default method trait Trait1a { fn trait_method1(self) -> Field { - self.trait_method2() * 7892 - self.vl + self.trait_method2() * 7892 - self.vl() } fn trait_method2(self) -> Field { let _ = self; 43278 } + fn vl(self) -> Field; } struct Struct1a { vl: Field, } -impl Trait1a for Struct1a {} +impl Trait1a for Struct1a { + fn vl(self) -> Field { + self.vl + } +} // 1b) trait default method -> trait overriden method trait Trait1b { fn trait_method1(self) -> Field { - self.trait_method2() * 2832 - self.vl + self.trait_method2() * 2832 - self.vl() } fn trait_method2(self) -> Field { + let _ = self; 9323 } + fn vl(self) -> Field; } struct Struct1b { vl: Field, @@ -51,13 +58,17 @@ impl Trait1b for Struct1b { let _ = self; 2394 } + fn vl(self) -> Field { + self.vl + } } // 1c) trait default method -> trait overriden (no default) method trait Trait1c { fn trait_method1(self) -> Field { - self.trait_method2() * 7635 - self.vl + self.trait_method2() * 7635 - self.vl() } fn trait_method2(self) -> Field; + fn vl(self) -> Field; } struct Struct1c { vl: Field, @@ -67,16 +78,20 @@ impl Trait1c for Struct1c { let _ = self; 5485 } + fn vl(self) -> Field { + self.vl + } } // 1d) trait overriden method -> trait default method trait Trait1d { fn trait_method1(self) -> Field { - self.trait_method2() * 2825 - self.vl + self.trait_method2() * 2825 - self.vl() } fn trait_method2(self) -> Field { let _ = self; 29341 } + fn vl(self) -> Field; } struct Struct1d { vl: Field, @@ -85,15 +100,20 @@ impl Trait1d for Struct1d { fn trait_method1(self) -> Field { self.trait_method2() * 9342 - self.vl } + fn vl(self) -> Field { + self.vl + } } // 1e) trait overriden method -> trait overriden method trait Trait1e { fn trait_method1(self) -> Field { - self.trait_method2() * 85465 - self.vl + self.trait_method2() * 85465 - self.vl() } fn trait_method2(self) -> Field { + let _ = self; 2381 } + fn vl(self) -> Field; } struct Struct1e { vl: Field, @@ -106,13 +126,17 @@ impl Trait1e for Struct1e { let _ = self; 58945 } + fn vl(self) -> Field { + self.vl + } } // 1f) trait overriden method -> trait overriden (no default) method trait Trait1f { fn trait_method1(self) -> Field { - self.trait_method2() * 43257 - self.vl + self.trait_method2() * 43257 - self.vl() } fn trait_method2(self) -> Field; + fn vl(self) -> Field; } struct Struct1f { vl: Field, @@ -125,6 +149,9 @@ impl Trait1f for Struct1f { let _ = self; 5748 } + fn vl(self) -> Field { + self.vl + } } // 1g) trait overriden (no default) method -> trait default method trait Trait1g { @@ -146,6 +173,7 @@ impl Trait1g for Struct1g { trait Trait1h { fn trait_method1(self) -> Field; fn trait_method2(self) -> Field { + let _ = self; 7823 } } @@ -182,24 +210,30 @@ impl Trait1i for Struct1i { // 2a) trait default method -> trait default function trait Trait2a { fn trait_method1(self) -> Field { - Self::trait_function2() * 2385 - self.vl + Self::trait_function2() * 2385 - self.vl() } fn trait_function2() -> Field { 7843 } + fn vl(self) -> Field; } struct Struct2a { vl: Field, } -impl Trait2a for Struct2a {} +impl Trait2a for Struct2a { + fn vl(self) -> Field { + self.vl + } +} // 2b) trait default method -> trait overriden function trait Trait2b { fn trait_method1(self) -> Field { - Self::trait_function2() * 6583 - self.vl + Self::trait_function2() * 6583 - self.vl() } fn trait_function2() -> Field { 3752 } + fn vl(self) -> Field; } struct Struct2b { vl: Field, @@ -208,13 +242,17 @@ impl Trait2b for Struct2b { fn trait_function2() -> Field { 8477 } + fn vl(self) -> Field { + self.vl + } } // 2c) trait default method -> trait overriden (no default) function trait Trait2c { fn trait_method1(self) -> Field { - Self::trait_function2() * 2831 - self.vl + Self::trait_function2() * 2831 - self.vl() } fn trait_function2() -> Field; + fn vl(self) -> Field; } struct Struct2c { vl: Field, @@ -223,15 +261,19 @@ impl Trait2c for Struct2c { fn trait_function2() -> Field { 8342 } + fn vl(self) -> Field { + self.vl + } } // 2d) trait overriden method -> trait default function trait Trait2d { fn trait_method1(self) -> Field { - Self::trait_function2() * 924 - self.vl + Self::trait_function2() * 924 - self.vl() } fn trait_function2() -> Field { 384 } + fn vl(self) -> Field; } struct Struct2d { vl: Field, @@ -240,15 +282,19 @@ impl Trait2d for Struct2d { fn trait_method1(self) -> Field { Self::trait_function2() * 3984 - self.vl } + fn vl(self) -> Field { + self.vl + } } // 2e) trait overriden method -> trait overriden function trait Trait2e { fn trait_method1(self) -> Field { - Self::trait_function2() * 3642 - self.vl + Self::trait_function2() * 3642 - self.vl() } fn trait_function2() -> Field { 97342 } + fn vl(self) -> Field; } struct Struct2e { vl: Field, @@ -260,13 +306,17 @@ impl Trait2e for Struct2e { fn trait_function2() -> Field { 39400 } + fn vl(self) -> Field { + self.vl + } } // 2f) trait overriden method -> trait overriden (no default) function trait Trait2f { fn trait_method1(self) -> Field { - Self::trait_function2() * 2783 - self.vl + Self::trait_function2() * 2783 - self.vl() } fn trait_function2() -> Field; + fn vl(self) -> Field; } struct Struct2f { vl: Field, @@ -278,6 +328,9 @@ impl Trait2f for Struct2f { fn trait_function2() -> Field { 72311 } + fn vl(self) -> Field { + self.vl + } } // 2g) trait overriden (no default) method -> trait default function trait Trait2g { @@ -332,25 +385,32 @@ impl Trait2i for Struct2i { // 3a) trait default function -> trait default method trait Trait3a { fn trait_function1(a: Field, b: Self) -> Field { - b.trait_method2() * 8344 - b.vl + a + b.trait_method2() * 8344 - b.vl() + a } fn trait_method2(self) -> Field { let _ = self; 19212 } + fn vl(self) -> Field; } struct Struct3a { vl: Field, } -impl Trait3a for Struct3a {} +impl Trait3a for Struct3a { + fn vl(self) -> Field { + self.vl + } +} // 3b) trait default function -> trait overriden method trait Trait3b { fn trait_function1(a: Field, b: Self) -> Field { - b.trait_method2() * 9233 - b.vl + a + b.trait_method2() * 9233 - b.vl() + a } fn trait_method2(self) -> Field { + let _ = self; 9111 } + fn vl(self) -> Field; } struct Struct3b { vl: Field, @@ -360,13 +420,17 @@ impl Trait3b for Struct3b { let _ = self; 2392 } + fn vl(self) -> Field { + self.vl + } } // 3c) trait default function -> trait overriden (no default) method trait Trait3c { fn trait_function1(a: Field, b: Self) -> Field { - b.trait_method2() * 2822 - b.vl + a + b.trait_method2() * 2822 - b.vl() + a } fn trait_method2(self) -> Field; + fn vl(self) -> Field; } struct Struct3c { vl: Field, @@ -376,16 +440,20 @@ impl Trait3c for Struct3c { let _ = self; 7743 } + fn vl(self) -> Field { + self.vl + } } // 3d) trait overriden function -> trait default method trait Trait3d { fn trait_function1(a: Field, b: Self) -> Field { - b.trait_method2() * 291 - b.vl + a + b.trait_method2() * 291 - b.vl() + a } fn trait_method2(self) -> Field { let _ = self; 3328 } + fn vl(self) -> Field; } struct Struct3d { vl: Field, @@ -394,15 +462,20 @@ impl Trait3d for Struct3d { fn trait_function1(a: Field, b: Self) -> Field { b.trait_method2() * 4933 - b.vl + a } + fn vl(self) -> Field { + self.vl + } } // 3e) trait overriden function -> trait overriden method trait Trait3e { fn trait_function1(a: Field, b: Self) -> Field { - b.trait_method2() * 71231 - b.vl + a + b.trait_method2() * 71231 - b.vl() + a } fn trait_method2(self) -> Field { + let _ = self; 373 } + fn vl(self) -> Field; } struct Struct3e { vl: Field, @@ -415,13 +488,17 @@ impl Trait3e for Struct3e { let _ = self; 80002 } + fn vl(self) -> Field { + self.vl + } } // 3f) trait overriden function -> trait overriden (no default) method trait Trait3f { fn trait_function1(a: Field, b: Self) -> Field { - b.trait_method2() * 28223 - b.vl + a + b.trait_method2() * 28223 - b.vl() + a } fn trait_method2(self) -> Field; + fn vl(self) -> Field; } struct Struct3f { vl: Field, @@ -434,6 +511,9 @@ impl Trait3f for Struct3f { let _ = self; 63532 } + fn vl(self) -> Field { + self.vl + } } // 3g) trait overriden (no default) function -> trait default method trait Trait3g { @@ -455,6 +535,7 @@ impl Trait3g for Struct3g { trait Trait3h { fn trait_function1(a: Field, b: Self) -> Field; fn trait_method2(self) -> Field { + let _ = self; 293 } } diff --git a/noir/noir-repo/test_programs/compile_success_empty/trait_generics/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/trait_generics/src/main.nr index 08302ded68c..e8b57b6fe6f 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/trait_generics/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/trait_generics/src/main.nr @@ -24,7 +24,7 @@ where T: MyInto, { fn into(self) -> [U; N] { - self.map(|x: T| x.into()) + self.map(|x| x.into()) } } diff --git a/noir/noir-repo/test_programs/execution_success/trait_inheritance/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/trait_inheritance/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/trait_inheritance/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/trait_inheritance/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/trait_inheritance/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/trait_inheritance/src/main.nr similarity index 100% rename from noir/noir-repo/test_programs/execution_success/trait_inheritance/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/trait_inheritance/src/main.nr diff --git a/noir/noir-repo/test_programs/execution_success/unit_value/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/unit_value/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/unit_value/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_empty/unit_value/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/unit_value/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/unit_value/src/main.nr similarity index 100% rename from noir/noir-repo/test_programs/execution_success/unit_value/src/main.nr rename to noir/noir-repo/test_programs/compile_success_empty/unit_value/src/main.nr diff --git a/noir/noir-repo/test_programs/compile_success_empty/unquote_struct/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/unquote_struct/src/main.nr index d4ab275858c..12c683a94a8 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/unquote_struct/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/unquote_struct/src/main.nr @@ -10,14 +10,7 @@ fn foo(x: Field, y: u32) -> u32 { // Given a function, wrap its parameters in a struct definition comptime fn output_struct(f: FunctionDefinition) -> Quoted { - let fields = f - .parameters() - .map(|param: (Quoted, Type)| { - let name = param.0; - let typ = param.1; - quote { $name: $typ, } - }) - .join(quote {}); + let fields = f.parameters().map(|(name, typ)| quote { $name: $typ, }).join(quote {}); quote { struct Foo { $fields } diff --git a/noir/noir-repo/test_programs/compile_success_no_bug/check_uncostrained_regression/Nargo.toml b/noir/noir-repo/test_programs/compile_success_no_bug/check_unconstrained_regression/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/compile_success_no_bug/check_uncostrained_regression/Nargo.toml rename to noir/noir-repo/test_programs/compile_success_no_bug/check_unconstrained_regression/Nargo.toml diff --git a/noir/noir-repo/test_programs/compile_success_no_bug/check_uncostrained_regression/src/main.nr b/noir/noir-repo/test_programs/compile_success_no_bug/check_unconstrained_regression/src/main.nr similarity index 55% rename from noir/noir-repo/test_programs/compile_success_no_bug/check_uncostrained_regression/src/main.nr rename to noir/noir-repo/test_programs/compile_success_no_bug/check_unconstrained_regression/src/main.nr index 33b84c2b702..174b68fd162 100644 --- a/noir/noir-repo/test_programs/compile_success_no_bug/check_uncostrained_regression/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_no_bug/check_unconstrained_regression/src/main.nr @@ -1,23 +1,26 @@ -struct Trigger{ +struct Trigger { x: u32, y: Field, - z: [Field;3], + z: [Field; 3], } -struct ResultType{ +struct ResultType { a: u32, b: Field, - c: [Field;3], + c: [Field; 3], } unconstrained fn convert(trigger: Trigger) -> ResultType { - let result= ResultType { a: trigger.x + 1, b: trigger.y - 1 + trigger.z[2], c: [trigger.z[0], 0, trigger.z[1]] }; + let result = ResultType { + a: trigger.x + 1, + b: trigger.y - 1 + trigger.z[2], + c: [trigger.z[0], 0, trigger.z[1]], + }; result } impl Trigger { fn execute(self) -> ResultType { - let result = unsafe { - convert(self) - }; + /// Safety: testing context + let result = unsafe { convert(self) }; assert(result.a == self.x + 1); assert(result.b == self.y - 1 + self.z[2]); assert(result.c[1] == 0); diff --git a/noir/noir-repo/test_programs/compile_success_no_bug/regression_7062/Nargo.toml b/noir/noir-repo/test_programs/compile_success_no_bug/regression_7062/Nargo.toml new file mode 100644 index 00000000000..0e11219ad98 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_no_bug/regression_7062/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "regression_7062" +type = "bin" +authors = [""] +compiler_version = ">=0.31.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/compile_success_no_bug/regression_7062/src/main.nr b/noir/noir-repo/test_programs/compile_success_no_bug/regression_7062/src/main.nr new file mode 100644 index 00000000000..c640062b45b --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_no_bug/regression_7062/src/main.nr @@ -0,0 +1,8 @@ +fn main(args: [Field; 2]) { + /// Safety: n/a + unsafe { store(args) }; + // Dummy test to remove the 'underconstraint bug' + assert(args[0] + args[1] != 0); +} + +pub unconstrained fn store(_: [Field]) {} \ No newline at end of file diff --git a/noir/noir-repo/test_programs/compile_success_with_bug/underconstrained_value_detector_5425/src/main.nr b/noir/noir-repo/test_programs/compile_success_with_bug/underconstrained_value_detector_5425/src/main.nr index 1d9269eebd5..22e2bc0b49d 100644 --- a/noir/noir-repo/test_programs/compile_success_with_bug/underconstrained_value_detector_5425/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_with_bug/underconstrained_value_detector_5425/src/main.nr @@ -9,6 +9,7 @@ unconstrained fn maximum_price(options: [u32; 3]) -> u32 { } fn main(sandwiches: pub [u32; 3], drinks: pub [u32; 3], snacks: pub [u32; 3], best_value: u32) { + /// Safety: testing context unsafe { let meal_deal_cost: u32 = 390; let most_expensive_sandwich = maximum_price(sandwiches); diff --git a/noir/noir-repo/test_programs/execution_failure/brillig_assert_fail/src/main.nr b/noir/noir-repo/test_programs/execution_failure/brillig_assert_fail/src/main.nr index 07256f0c398..18e6422361c 100644 --- a/noir/noir-repo/test_programs/execution_failure/brillig_assert_fail/src/main.nr +++ b/noir/noir-repo/test_programs/execution_failure/brillig_assert_fail/src/main.nr @@ -1,12 +1,9 @@ // Tests a very simple program. -// +// // The features being tested is using assert on brillig fn main(x: Field) { - assert( - 1 == unsafe { - conditional(x as bool) - } - ); + /// Safety: testing context + assert(1 == unsafe { conditional(x as bool) }); } unconstrained fn conditional(x: bool) -> Field { diff --git a/noir/noir-repo/test_programs/execution_failure/brillig_assert_msg_runtime/src/main.nr b/noir/noir-repo/test_programs/execution_failure/brillig_assert_msg_runtime/src/main.nr index 4dd06ceb743..9c07660217b 100644 --- a/noir/noir-repo/test_programs/execution_failure/brillig_assert_msg_runtime/src/main.nr +++ b/noir/noir-repo/test_programs/execution_failure/brillig_assert_msg_runtime/src/main.nr @@ -1,9 +1,6 @@ fn main(x: Field) { - assert( - 1 == unsafe { - conditional(x) - } - ); + /// Safety: testing context + assert(1 == unsafe { conditional(x) }); } unconstrained fn conditional(x: Field) -> Field { diff --git a/noir/noir-repo/test_programs/execution_failure/fold_nested_brillig_assert_fail/src/main.nr b/noir/noir-repo/test_programs/execution_failure/fold_nested_brillig_assert_fail/src/main.nr index 1c1563ffe1d..bbb6f8ff422 100644 --- a/noir/noir-repo/test_programs/execution_failure/fold_nested_brillig_assert_fail/src/main.nr +++ b/noir/noir-repo/test_programs/execution_failure/fold_nested_brillig_assert_fail/src/main.nr @@ -1,5 +1,5 @@ // Tests a very simple program. -// +// // The features being tested is using assert on brillig that is triggered through nested ACIR calls. // We want to make sure we get a call stack from the original call in main to the failed assert. fn main(x: Field) { @@ -14,6 +14,7 @@ fn fold_conditional_wrapper(x: bool) -> Field { #[fold] fn fold_conditional(x: bool) -> Field { + /// Safety: testing context unsafe { conditional_wrapper(x) } diff --git a/noir/noir-repo/test_programs/execution_failure/mocks_in_execution/Nargo.toml b/noir/noir-repo/test_programs/execution_failure/mocks_in_execution/Nargo.toml new file mode 100644 index 00000000000..c83da9a3d0c --- /dev/null +++ b/noir/noir-repo/test_programs/execution_failure/mocks_in_execution/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "mocks_in_execution" +type = "bin" +authors = [""] +compiler_version = ">=0.23.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/execution_failure/mocks_in_execution/Prover.toml b/noir/noir-repo/test_programs/execution_failure/mocks_in_execution/Prover.toml new file mode 100644 index 00000000000..e69de29bb2d diff --git a/noir/noir-repo/test_programs/execution_failure/mocks_in_execution/src/main.nr b/noir/noir-repo/test_programs/execution_failure/mocks_in_execution/src/main.nr new file mode 100644 index 00000000000..b43068ab251 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_failure/mocks_in_execution/src/main.nr @@ -0,0 +1,5 @@ +fn main() { + // Trying to use a mock in `nargo execute` should fail. + let mock = unsafe { std::test::OracleMock::mock("foo") }; + assert_eq(mock.id, 0); +} diff --git a/noir/noir-repo/test_programs/execution_failure/regression_5202/src/main.nr b/noir/noir-repo/test_programs/execution_failure/regression_5202/src/main.nr index fbcdba43355..0c3a9693958 100644 --- a/noir/noir-repo/test_programs/execution_failure/regression_5202/src/main.nr +++ b/noir/noir-repo/test_programs/execution_failure/regression_5202/src/main.nr @@ -2,7 +2,11 @@ trait ToField { fn to_field(self) -> Field; } -impl ToField for bool { fn to_field(self) -> Field { self as Field } } +impl ToField for bool { + fn to_field(self) -> Field { + self as Field + } +} unconstrained fn get_unconstrained_option() -> Option { Option::some(13) @@ -13,20 +17,17 @@ unconstrained fn should_i_assert() -> bool { } fn get_magical_boolean() -> bool { - let option = unsafe { - get_unconstrained_option() - }; + /// Safety: testing context + let option = unsafe { get_unconstrained_option() }; let pre_assert = option.is_some().to_field(); - if unsafe { - should_i_assert() - } { + /// Safety: testing context + if unsafe { should_i_assert() } { // Note that `should_i_assert` is unconstrained, so Noir should not be able to infer // any behavior from the contents of this block. In this case it is actually false, so the // assertion below is not even executed (if it did it'd fail since the values are not equal). - - assert_eq(option, Option::some(42)); /// <- this seems to be the trigger for the bug + assert_eq(option, Option::some(42)); // <- this seems to be the trigger for the bug } // In my testing, the `option` value exhibits weird behavior from this point on, as if it had been mutated diff --git a/noir/noir-repo/test_programs/execution_report.sh b/noir/noir-repo/test_programs/execution_report.sh new file mode 100755 index 00000000000..827b7806d37 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_report.sh @@ -0,0 +1,84 @@ +#!/usr/bin/env bash +set -e + +PARSE_TIME=$(realpath "$(dirname "$0")/parse_time.sh") +current_dir=$(pwd) +base_path="$current_dir/execution_success" + +# Tests to be profiled for execution report +tests_to_profile=("sha256_regression" "regression_4709" "ram_blowup_regression") + +echo "{\"execution_reports\": [ " > $current_dir/execution_report.json + +# If there is an argument that means we want to generate a report for only the current directory +if [ "$1" == "1" ]; then + base_path="$current_dir" + tests_to_profile=(".") +fi + +ITER="1" +NUM_ARTIFACTS=${#tests_to_profile[@]} + +for dir in ${tests_to_profile[@]}; do + if [[ " ${excluded_dirs[@]} " =~ " ${dir} " ]]; then + continue + fi + + if [[ ${CI-false} = "true" ]] && [[ " ${ci_excluded_dirs[@]} " =~ " ${dir} " ]]; then + continue + fi + + cd $base_path/$dir + + # The default package to run is the supplied list hardcoded at the top of the script + PACKAGE_NAME=$dir + # Otherwise default to the current directory as the package we want to run + if [ "$1" == "1" ]; then + PACKAGE_NAME=$(basename $current_dir) + fi + + # Check whether a compilation artifact exists. + # Any programs part of this benchmark should already be compiled. + # We want to make sure that compilation time is not included in the execution time. + if [ ! -e ./target/*.json ]; then + echo "Missing compilation artifact for $PACKAGE_NAME" + exit 1 + fi + + + NUM_RUNS=$2 + TOTAL_TIME=0 + + for ((i = 1; i <= NUM_RUNS; i++)); do + NOIR_LOG=trace NARGO_LOG_DIR=./tmp nargo execute --silence-warnings + done + + TIMES=($(jq -r '. | select(.target == "nargo::ops::execute" and .fields.message == "close") | .fields."time.busy"' ./tmp/*)) + + AVG_TIME=$(awk -v RS=" " -v parse_time="$PARSE_TIME" ' + { + # Times are formatted annoyingly so we need to parse it. + parse_time" "$1 | getline current_time + close(parse_time" "$1) + sum += current_time; + n++; + } + END { + if (n > 0) + printf "%.3f\n", sum / n + else + printf "%.3f\n", 0 + }' <<<"${TIMES[@]}") + + jq -rc "{artifact_name: \"$PACKAGE_NAME\", time: \""$AVG_TIME"s\"}" --null-input >> $current_dir/execution_report.json + + if (($ITER != $NUM_ARTIFACTS)); then + echo "," >> $current_dir/execution_report.json + fi + + rm -rf ./tmp + + ITER=$(( $ITER + 1 )) +done + +echo "]}" >> $current_dir/execution_report.json diff --git a/noir/noir-repo/test_programs/execution_success/acir_inside_brillig_recursion/Prover.toml b/noir/noir-repo/test_programs/execution_success/acir_inside_brillig_recursion/Prover.toml deleted file mode 100644 index 8b137891791..00000000000 --- a/noir/noir-repo/test_programs/execution_success/acir_inside_brillig_recursion/Prover.toml +++ /dev/null @@ -1 +0,0 @@ - diff --git a/noir/noir-repo/test_programs/execution_success/aes128_encrypt/src/main.nr b/noir/noir-repo/test_programs/execution_success/aes128_encrypt/src/main.nr index 3d3992971f7..11f9512444e 100644 --- a/noir/noir-repo/test_programs/execution_success/aes128_encrypt/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/aes128_encrypt/src/main.nr @@ -22,16 +22,18 @@ unconstrained fn decode_hex(s: str) -> [u8; M] { unconstrained fn cipher(plaintext: [u8; 12], iv: [u8; 16], key: [u8; 16]) -> [u8; 16] { let result = std::aes128::aes128_encrypt(plaintext, iv, key); - result.as_array() + result } fn main(inputs: str<12>, iv: str<16>, key: str<16>, output: str<32>) { let result: [u8; 16] = - std::aes128::aes128_encrypt(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()).as_array(); + std::aes128::aes128_encrypt(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()); + /// Safety: testing context let output_bytes: [u8; 16] = unsafe { decode_hex(output) }; assert(result == output_bytes); + /// Safety: testing context let unconstrained_result = unsafe { cipher(inputs.as_bytes(), iv.as_bytes(), key.as_bytes()) }; assert(unconstrained_result == output_bytes); } diff --git a/noir/noir-repo/test_programs/execution_success/array_to_slice_constant_length/src/main.nr b/noir/noir-repo/test_programs/execution_success/array_to_slice_constant_length/src/main.nr index 09f99622939..1d29db4973a 100644 --- a/noir/noir-repo/test_programs/execution_success/array_to_slice_constant_length/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/array_to_slice_constant_length/src/main.nr @@ -4,6 +4,7 @@ unconstrained fn return_array(val: Field) -> [Field; 1] { } fn main(val: Field) { + /// Safety: testing context unsafe { let array = return_array(val); assert_constant(array.as_slice().len()); diff --git a/noir/noir-repo/test_programs/execution_success/bigint/src/main.nr b/noir/noir-repo/test_programs/execution_success/bigint/src/main.nr index 5a8b9fb67ef..2095c92349a 100644 --- a/noir/noir-repo/test_programs/execution_success/bigint/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/bigint/src/main.nr @@ -17,6 +17,7 @@ fn main(mut x: [u8; 5], y: [u8; 5]) { let c = if x[0] != 0 { test_unconstrained1(a, b) } else { + /// Safety: testing context unsafe { test_unconstrained2(a, b) } diff --git a/noir/noir-repo/test_programs/execution_success/brillig_acir_as_brillig/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_acir_as_brillig/src/main.nr index 1a595ecfb38..a327dfd7533 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_acir_as_brillig/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_acir_as_brillig/src/main.nr @@ -1,4 +1,5 @@ fn main(x: u32) { + /// Safety: testing context unsafe { assert(entry_point(x) == 2); swap_entry_point(x, x + 1); diff --git a/noir/noir-repo/test_programs/execution_success/brillig_arrays/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_arrays/src/main.nr index 00d43e2b10d..f0524842ac6 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_arrays/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_arrays/src/main.nr @@ -2,6 +2,7 @@ // // The features being tested are array reads and writes fn main(x: [Field; 3]) { + /// Safety: testing context unsafe { read_array(x); read_write_array(x); diff --git a/noir/noir-repo/test_programs/execution_success/brillig_blake2s/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_blake2s/src/main.nr index c4412bd2bba..7e567b84139 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_blake2s/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_blake2s/src/main.nr @@ -2,6 +2,7 @@ // // The features being tested is blake2s in brillig fn main(x: [u8; 5], result: [u8; 32]) { + /// Safety: testing context unsafe { assert(blake2s(x) == result); } diff --git a/noir/noir-repo/test_programs/execution_success/brillig_calls/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_calls/src/main.nr index c76d7651b0a..5bd3458da51 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_calls/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_calls/src/main.nr @@ -3,6 +3,7 @@ // The features being tested is brillig calls fn main(x: u32) { unsafe { + /*@safety : testing context*/ assert(entry_point(x) == 2); swap_entry_point(x, x + 1); assert(deep_entry_point(x) == 4); diff --git a/noir/noir-repo/test_programs/execution_success/brillig_calls_array/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_calls_array/src/main.nr index 90f588ab988..b27eb30da17 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_calls_array/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_calls_array/src/main.nr @@ -2,6 +2,7 @@ // // The features being tested is brillig calls passing arrays around fn main(x: [u32; 3]) { + /// Safety: testing context unsafe { assert(entry_point(x) == 9); another_entry_point(x); diff --git a/noir/noir-repo/test_programs/execution_success/brillig_calls_conditionals/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_calls_conditionals/src/main.nr index 0fbfd84968f..31b70cb12cf 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_calls_conditionals/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_calls_conditionals/src/main.nr @@ -2,6 +2,7 @@ // // The features being tested is brillig calls with conditionals fn main(x: [u32; 3]) { + /// Safety: testing context unsafe { assert(entry_point(x[0]) == 7); assert(entry_point(x[1]) == 8); diff --git a/noir/noir-repo/test_programs/execution_success/brillig_conditional/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_conditional/src/main.nr index 17484f6e8c5..024ee52d5af 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_conditional/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_conditional/src/main.nr @@ -2,6 +2,7 @@ // // The features being tested is basic conditonal on brillig fn main(x: Field) { + /// Safety: testing context unsafe { assert(4 == conditional(x == 1)); } diff --git a/noir/noir-repo/test_programs/execution_success/brillig_fns_as_values/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_fns_as_values/src/main.nr index 03a92b4b10d..ff1c36c9ddf 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_fns_as_values/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_fns_as_values/src/main.nr @@ -3,6 +3,7 @@ struct MyStruct { } fn main(x: u32) { + /// Safety: testing context unsafe { assert(wrapper(increment, x) == x + 1); assert(wrapper(increment_acir, x) == x + 1); diff --git a/noir/noir-repo/test_programs/execution_success/brillig_identity_function/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_identity_function/src/main.nr index f3b45d0de4e..b676c46120e 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_identity_function/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_identity_function/src/main.nr @@ -6,6 +6,7 @@ struct myStruct { // // The features being tested is the identity function in Brillig fn main(x: Field) { + /// Safety: testing context unsafe { assert(x == identity(x)); // TODO: add support for array comparison diff --git a/noir/noir-repo/test_programs/execution_success/brillig_nested_arrays/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_nested_arrays/src/main.nr index 77ab4ea19a6..feb5433738f 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_nested_arrays/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_nested_arrays/src/main.nr @@ -28,6 +28,7 @@ unconstrained fn create_and_assert_inside_brillig(x: Field, y: Field) { } fn main(x: Field, y: Field) { + /// Safety: testing context unsafe { let header = Header { params: [1, 2, 3] }; let note0 = MyNote { array: [1, 2], plain: 3, header }; diff --git a/noir/noir-repo/test_programs/execution_success/brillig_not/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_not/src/main.nr index 4e61b6a54ea..6dc91f86127 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_not/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_not/src/main.nr @@ -2,6 +2,7 @@ // // The features being tested is not instruction on brillig fn main(x: Field, y: Field) { + /// Safety: testing context unsafe { assert(false == not_operator(x as bool)); assert(true == not_operator(y as bool)); diff --git a/noir/noir-repo/test_programs/execution_success/brillig_oracle/Prover.toml b/noir/noir-repo/test_programs/execution_success/brillig_oracle/Prover.toml deleted file mode 100644 index 161f4fb62c0..00000000000 --- a/noir/noir-repo/test_programs/execution_success/brillig_oracle/Prover.toml +++ /dev/null @@ -1,2 +0,0 @@ -_x = "10" - diff --git a/noir/noir-repo/test_programs/execution_success/brillig_recursion/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_recursion/src/main.nr index 5354777a1de..e7140ffa06c 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_recursion/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_recursion/src/main.nr @@ -2,6 +2,7 @@ // // The feature being tested is brillig recursion fn main(x: u32) { + /// Safety: testing context unsafe { assert(fibonacci(x) == 55); } diff --git a/noir/noir-repo/test_programs/execution_success/brillig_uninitialized_arrays/src/main.nr b/noir/noir-repo/test_programs/execution_success/brillig_uninitialized_arrays/src/main.nr index d4b74162cfb..39a13440ca9 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_uninitialized_arrays/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/brillig_uninitialized_arrays/src/main.nr @@ -1,4 +1,5 @@ fn main(x: Field, y: Field) -> pub Field { + /// Safety: testing context unsafe { let notes = create_notes(x, y); sum_x(notes, x, y) diff --git a/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/Nargo.toml b/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/Nargo.toml new file mode 100644 index 00000000000..5c2623b3a1c --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "encrypted_log_regression" +type = "bin" +authors = [""] +compiler_version = ">=0.31.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/Prover.toml b/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/Prover.toml new file mode 100644 index 00000000000..7ca8c21a692 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/Prover.toml @@ -0,0 +1,9 @@ +# Using the smaller sizes defined in `main.nr`. +# The reason this program is in the `execution_success` directory is because +# `rebuild.sh` only goes over these programs, but all we really care about is +# any potential future bytecode size regression. +eph_pk_bytes = [1, 2, 3] +incoming_header_ciphertext = [1, 2] +incoming_body_ciphertext = [9, 8, 7, 6, 5, 4, 3, 2, 1] +flag = true +return = [1, 2, 3, 1, 2, 0, 9, 8, 7, 6, 5, 4, 3, 2, 1] diff --git a/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/src/main.nr b/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/src/main.nr new file mode 100644 index 00000000000..c65f580b0c8 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/encrypted_log_regression/src/main.nr @@ -0,0 +1,94 @@ +// The code below is inspired by [compute_encrypted_log](https://github.com/AztecProtocol/aztec-packages/blob/b42756bc10175fea9eb60544759e9dbe41ae5e76/noir-projects/aztec-nr/aztec/src/encrypted_logs/payload.nr#L111) +// which resulted in a bytecode size blowup when compiled to ACIR, see https://github.com/noir-lang/noir/issues/6929 +// The issue was around `encrypted_bytes[offset + i]` generating large amounts of gates, as per the `flamegraph.sh` tool in aztec-packages. +// The details around encryption and addresses have been stripped away, focusing on just copying bytes of equivalent size arrays. + +// Original values which resulted in huge bytecode even on this example (500K long SSA) +// global PRIVATE_LOG_SIZE_IN_FIELDS: u32 = 18; +// global ENCRYPTED_PAYLOAD_SIZE_IN_BYTES: u32 = (PRIVATE_LOG_SIZE_IN_FIELDS - 1) * 31; +// global EPH_PK_SIZE: u32 = 32; +// global HEADER_SIZE: u32 = 48; +// global OVERHEAD_PADDING: u32 = 15; + +// Using the same formulas with smaller numbers; the effect is the same, but the SSA is more manageable. +global PRIVATE_LOG_SIZE_IN_FIELDS: u32 = 4; +global ENCRYPTED_PAYLOAD_SIZE_IN_BYTES: u32 = (PRIVATE_LOG_SIZE_IN_FIELDS - 1) * 5; +global EPH_PK_SIZE: u32 = 3; +global HEADER_SIZE: u32 = 2; +global OVERHEAD_PADDING: u32 = 1; + +// Unused because encryption didn't play a role: +// global OVERHEAD_SIZE: u32 = EPH_PK_SIZE + HEADER_SIZE + OVERHEAD_PADDING; +// global PLAINTEXT_LENGTH_SIZE: u32 = 2; +// global MAX_PRIVATE_LOG_PLAINTEXT_SIZE_IN_BYTES: u32 = +// ENCRYPTED_PAYLOAD_SIZE_IN_BYTES - OVERHEAD_SIZE - PLAINTEXT_LENGTH_SIZE - 1 /* aes padding */; + +global BODY_SIZE: u32 = + ENCRYPTED_PAYLOAD_SIZE_IN_BYTES - EPH_PK_SIZE - HEADER_SIZE - OVERHEAD_PADDING; + +fn main( + eph_pk_bytes: [u8; EPH_PK_SIZE], + incoming_header_ciphertext: [u8; HEADER_SIZE], + incoming_body_ciphertext: [u8; BODY_SIZE], + flag: bool, +) -> pub [u8; ENCRYPTED_PAYLOAD_SIZE_IN_BYTES] { + compute_encrypted_log( + eph_pk_bytes, + incoming_header_ciphertext, + incoming_body_ciphertext, + flag, + ) +} + +fn compute_encrypted_log( + eph_pk_bytes: [u8; EPH_PK_SIZE], + incoming_header_ciphertext: [u8; HEADER_SIZE], + incoming_body_ciphertext: [u8; BODY_SIZE], + flag: bool, +) -> [u8; M] { + let mut encrypted_bytes = [0; M]; + let mut offset = 0; + + // NOTE: Adding a conditional variable can result in the array being fully copied, item by item, + // in each iteration in the second loop that copies incoming_body_ciphertext into encrypted_bytes. + // Depending on where we place the `flag` we either get the item-by-item copying (blowup), + // or just a single array item gets read and a new array constructed in each iteration (no blowup). + + // If the `flag` is here then it blows up. + if flag { + // eph_pk + for i in 0..EPH_PK_SIZE { + encrypted_bytes[offset + i] = eph_pk_bytes[i]; + } + offset += EPH_PK_SIZE; + + // If the `flag` is here then it blows up. + // if flag { + + // incoming_header + for i in 0..HEADER_SIZE { + encrypted_bytes[offset + i] = incoming_header_ciphertext[i]; + } + offset += HEADER_SIZE; + + // Padding. + offset += OVERHEAD_PADDING; + + // If the `flag` is here then it does not blow up. + //if flag { + // incoming_body + // Then we fill in the rest as the incoming body ciphertext + let size = M - offset; + + // NOTE: This made the bytecode size blowup disappear in aztec packages, + // but in this reproduction the size seems to be statically known regardless. + // let size = M - 32 - HEADER_SIZE - OVERHEAD_PADDING; + + assert_eq(size, incoming_body_ciphertext.len(), "ciphertext length mismatch"); + for i in 0..size { + encrypted_bytes[offset + i] = incoming_body_ciphertext[i]; + } + } + + encrypted_bytes +} diff --git a/noir/noir-repo/test_programs/execution_success/global_consts/src/main.nr b/noir/noir-repo/test_programs/execution_success/global_consts/src/main.nr index 2eaab810d6a..9f84af35cba 100644 --- a/noir/noir-repo/test_programs/execution_success/global_consts/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/global_consts/src/main.nr @@ -25,7 +25,7 @@ unconstrained fn calculate_global_value() -> Field { } // Regression test for https://github.com/noir-lang/noir/issues/4318 -global CALCULATED_GLOBAL: Field = unsafe { calculate_global_value() }; +global CALCULATED_GLOBAL: Field = calculate_global_value(); fn main( a: [Field; M + N - N], diff --git a/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/Nargo.toml b/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/Nargo.toml new file mode 100644 index 00000000000..50b4902de22 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/Nargo.toml @@ -0,0 +1,6 @@ +[package] +name = "global_var_regression_simple" +type = "bin" +authors = [""] + +[dependencies] \ No newline at end of file diff --git a/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/Prover.toml b/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/Prover.toml new file mode 100644 index 00000000000..2c1854573a4 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/Prover.toml @@ -0,0 +1,2 @@ +x = 1 +y = 2 diff --git a/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/src/main.nr b/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/src/main.nr new file mode 100644 index 00000000000..b1bf753a73c --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/global_var_regression_simple/src/main.nr @@ -0,0 +1,25 @@ +global EXPONENTIATE: [[Field; 2]; 2] = [[1, 1], [0, 0]]; + +fn main(x: Field, y: pub Field) { + let mut acc: Field = 0; + for i in 0..2 { + for j in 0..2 { + acc += EXPONENTIATE[i][j]; + } + } + assert(!acc.lt(x)); + assert(x != y); + + dummy_again(x, y); +} + +fn dummy_again(x: Field, y: Field) { + let mut acc: Field = 0; + for i in 0..2 { + for j in 0..2 { + acc += EXPONENTIATE[i][j]; + } + } + assert(!acc.lt(x)); + assert(x != y); +} diff --git a/noir/noir-repo/test_programs/execution_success/hint_black_box/src/main.nr b/noir/noir-repo/test_programs/execution_success/hint_black_box/src/main.nr index 1109c54301f..ed2dc2d3760 100644 --- a/noir/noir-repo/test_programs/execution_success/hint_black_box/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/hint_black_box/src/main.nr @@ -2,12 +2,12 @@ use std::hint::black_box; fn main(a: u32, b: u32) { // This version unrolls into a number of additions - assert_eq(loop(5, a), b); + assert_eq(loop_(5, a), b); // This version simplifies into a single `constraint 50 == b` - assert_eq(loop(5, 10), b); + assert_eq(loop_(5, 10), b); // This version should not simplify down to a single constraint, // it should treat 10 as opaque: - assert_eq(loop(5, black_box(10)), b); + assert_eq(loop_(5, black_box(10)), b); // Check array handling. let arr = [a, a, a, a, a]; @@ -23,6 +23,7 @@ fn main(a: u32, b: u32) { //assert_eq(slice_sum(black_box(arr.as_slice())), b); // But we can pass a blackboxed slice to Brillig. + /// Safety: testing context let s = unsafe { brillig_slice_sum(black_box(arr.as_slice())) }; assert_eq(s, b); @@ -53,7 +54,7 @@ fn main(a: u32, b: u32) { //assert_eq(c, b); } -fn loop(n: u32, k: u32) -> u32 { +fn loop_(n: u32, k: u32) -> u32 { let mut sum = 0; for _ in 0..n { sum = sum + k; diff --git a/noir/noir-repo/test_programs/execution_success/is_unconstrained/Prover.toml b/noir/noir-repo/test_programs/execution_success/is_unconstrained/Prover.toml deleted file mode 100644 index 8b137891791..00000000000 --- a/noir/noir-repo/test_programs/execution_success/is_unconstrained/Prover.toml +++ /dev/null @@ -1 +0,0 @@ - diff --git a/noir/noir-repo/test_programs/execution_success/loop/src/main.nr b/noir/noir-repo/test_programs/execution_success/loop/src/main.nr index b3be4c4c3ff..3e1f24742ec 100644 --- a/noir/noir-repo/test_programs/execution_success/loop/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/loop/src/main.nr @@ -2,12 +2,12 @@ // // The features being tested is basic looping. fn main(six_as_u32: u32) { - assert_eq(loop(4), six_as_u32); + assert_eq(loop_excl(4), six_as_u32); assert_eq(loop_incl(3), six_as_u32); assert(plain_loop() == six_as_u32); } -fn loop(x: u32) -> u32 { +fn loop_excl(x: u32) -> u32 { let mut sum = 0; for i in 0..x { sum = sum + i; diff --git a/noir/noir-repo/test_programs/execution_success/loop_invariant_regression/src/main.nr b/noir/noir-repo/test_programs/execution_success/loop_invariant_regression/src/main.nr index c28ce063116..61f8b1bedba 100644 --- a/noir/noir-repo/test_programs/execution_success/loop_invariant_regression/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/loop_invariant_regression/src/main.nr @@ -1,11 +1,11 @@ // Tests a simple loop where we expect loop invariant instructions // to be hoisted to the loop's pre-header block. fn main(x: u32, y: u32) { - loop(4, x, y); + loop_(4, x, y); array_read_loop(4, x); } -fn loop(upper_bound: u32, x: u32, y: u32) { +fn loop_(upper_bound: u32, x: u32, y: u32) { for _ in 0..upper_bound { let mut z = x * y; z = z * x; diff --git a/noir/noir-repo/test_programs/execution_success/loop_keyword/Nargo.toml b/noir/noir-repo/test_programs/execution_success/loop_keyword/Nargo.toml new file mode 100644 index 00000000000..8189b407cd9 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/loop_keyword/Nargo.toml @@ -0,0 +1,5 @@ +[package] +name = "loop_keyword" +type = "bin" +authors = [""] +[dependencies] diff --git a/noir/noir-repo/test_programs/execution_success/loop_keyword/src/main.nr b/noir/noir-repo/test_programs/execution_success/loop_keyword/src/main.nr new file mode 100644 index 00000000000..b038ae22343 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/loop_keyword/src/main.nr @@ -0,0 +1,52 @@ +fn main() { + /// Safety: test code + unsafe { + check_loop(); + } + + check_comptime_loop(); +} + +unconstrained fn check_loop() { + let mut i = 0; + let mut sum = 0; + + loop { + if i == 4 { + break; + } + + if i == 2 { + i += 1; + continue; + } + + sum += i; + i += 1; + } + + assert_eq(sum, 1 + 3); +} + +fn check_comptime_loop() { + comptime { + let mut i = 0; + let mut sum = 0; + + loop { + if i == 4 { + break; + } + + if i == 2 { + i += 1; + continue; + } + + sum += i; + i += 1; + } + + assert_eq(sum, 1 + 3); + } +} diff --git a/noir/noir-repo/test_programs/execution_success/nested_arrays_from_brillig/src/main.nr b/noir/noir-repo/test_programs/execution_success/nested_arrays_from_brillig/src/main.nr index f8070aa3ef4..f724d0cdedd 100644 --- a/noir/noir-repo/test_programs/execution_success/nested_arrays_from_brillig/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/nested_arrays_from_brillig/src/main.nr @@ -20,6 +20,7 @@ unconstrained fn create_inside_brillig(values: [Field; 6]) -> [MyNote; 2] { } fn main(values: [Field; 6]) { + /// Safety: testing context let notes = unsafe { create_inside_brillig(values) }; assert(access_nested(notes) == (2 + 4 + 3 + 1)); } diff --git a/noir/noir-repo/test_programs/execution_success/reference_counts/src/main.nr b/noir/noir-repo/test_programs/execution_success/reference_counts/src/main.nr index 68b9f2407b9..8de4d0f2508 100644 --- a/noir/noir-repo/test_programs/execution_success/reference_counts/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/reference_counts/src/main.nr @@ -2,7 +2,7 @@ use std::mem::array_refcount; fn main() { let mut array = [0, 1, 2]; - assert_refcount(array, 2); + assert_refcount(array, 1); borrow(array, array_refcount(array)); borrow_mut(&mut array, array_refcount(array)); diff --git a/noir/noir-repo/test_programs/execution_success/reference_only_used_as_alias/src/main.nr b/noir/noir-repo/test_programs/execution_success/reference_only_used_as_alias/src/main.nr index 8b5b804cf94..3bbf1cf9ccb 100644 --- a/noir/noir-repo/test_programs/execution_success/reference_only_used_as_alias/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/reference_only_used_as_alias/src/main.nr @@ -61,6 +61,7 @@ where Event: EventInterface, { |e: Event| { + /// Safety: testing context unsafe { func(context.a); } diff --git a/noir/noir-repo/test_programs/execution_success/regression_5435/src/main.nr b/noir/noir-repo/test_programs/execution_success/regression_5435/src/main.nr index 895a6983ad9..a83ca055aab 100644 --- a/noir/noir-repo/test_programs/execution_success/regression_5435/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/regression_5435/src/main.nr @@ -3,6 +3,7 @@ fn main(input: Field, enable: bool) { let hash = no_predicate_function(input); // `EnableSideEffects` instruction from above instruction leaks out and removes the predicate from this call, // resulting in execution failure. + /// Safety: testing context unsafe { fail(hash) }; } } diff --git a/noir/noir-repo/test_programs/execution_success/regression_6451/src/main.nr b/noir/noir-repo/test_programs/execution_success/regression_6451/src/main.nr index b13b6c25a7e..edef71020e4 100644 --- a/noir/noir-repo/test_programs/execution_success/regression_6451/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/regression_6451/src/main.nr @@ -10,6 +10,7 @@ fn main(x: Field) { value.assert_max_bit_size::<1>(); // Regression test for #6447 (Aztec Packages issue #9488) + /// Safety: testing context let y = unsafe { empty(x + 1) }; let z = y + x + 1; let z1 = z + y; diff --git a/noir/noir-repo/test_programs/execution_success/regression_6674_3/src/main.nr b/noir/noir-repo/test_programs/execution_success/regression_6674_3/src/main.nr index 2c87a4c679c..57a3df5fcac 100644 --- a/noir/noir-repo/test_programs/execution_success/regression_6674_3/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/regression_6674_3/src/main.nr @@ -166,7 +166,7 @@ pub unconstrained fn sort_by_counter_desc(array: [T; N]) -> [T; N pub unconstrained fn sort_by(array: [T; N]) -> [T; N] { let mut result = array; - unsafe { get_sorting_index(array) }; + get_sorting_index(array); result } diff --git a/noir/noir-repo/test_programs/execution_success/regression_6834/Nargo.toml b/noir/noir-repo/test_programs/execution_success/regression_6834/Nargo.toml new file mode 100644 index 00000000000..edc3257d52d --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/regression_6834/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "regression_6834" +version = "0.1.0" +type = "bin" +authors = [""] + +[dependencies] diff --git a/noir/noir-repo/test_programs/execution_success/regression_6834/Prover.toml b/noir/noir-repo/test_programs/execution_success/regression_6834/Prover.toml new file mode 100644 index 00000000000..9ef840487ad --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/regression_6834/Prover.toml @@ -0,0 +1,2 @@ +year = 1 +min_age = 1 diff --git a/noir/noir-repo/test_programs/execution_success/regression_6834/src/main.nr b/noir/noir-repo/test_programs/execution_success/regression_6834/src/main.nr new file mode 100644 index 00000000000..cea8d163ab5 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/regression_6834/src/main.nr @@ -0,0 +1,9 @@ +fn main(year: u32, min_age: u8) -> pub u32 { + if min_age == 0 { + (year - 2) / 4 + } else if year > 1 { + (year - 2) / 4 + } else { + 0 + } +} diff --git a/noir/noir-repo/test_programs/execution_success/regression_bignum/Nargo.toml b/noir/noir-repo/test_programs/execution_success/regression_bignum/Nargo.toml new file mode 100644 index 00000000000..5cadd364e43 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/regression_bignum/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "regression_bignum" +version = "0.1.0" +type = "bin" +authors = [""] + +[dependencies] diff --git a/noir/noir-repo/test_programs/execution_success/regression_bignum/src/main.nr b/noir/noir-repo/test_programs/execution_success/regression_bignum/src/main.nr new file mode 100644 index 00000000000..013ab6b2023 --- /dev/null +++ b/noir/noir-repo/test_programs/execution_success/regression_bignum/src/main.nr @@ -0,0 +1,51 @@ +fn main() { + let numerator = + [790096867046896348, 1063071665130103641, 602707730209562162, 996751591622961462, 28650, 0]; + unsafe { __udiv_mod(numerator) }; + + let denominator = [12, 0, 0, 0, 0, 0]; + let result = unsafe { __validate_gt_remainder(denominator) }; + assert(result[4] == 0); + assert(result[5] == 0); +} + +unconstrained fn __udiv_mod(remainder_u60: [u64; 6]) { + let bit_difference = get_msb(remainder_u60); + let accumulator_u60: [u64; 6] = shl(bit_difference); +} + +unconstrained fn __validate_gt_remainder(a_u60: [u64; 6]) -> [u64; 6] { + let mut addend_u60: [u64; 6] = [0; 6]; + let mut result_u60: [u64; 6] = [0; 6]; + + for i in 0..6 { + result_u60[i] = a_u60[i] + addend_u60[i]; + } + + result_u60 +} + +unconstrained fn get_msb(val: [u64; 6]) -> u32 { + let mut count = 0; + for i in 0..6 { + let v = val[(6 - 1) - i]; + if (v > 0) { + count = 60 * ((6 - 1) - i); + break; + } + } + count +} + +unconstrained fn shl(shift: u32) -> [u64; 6] { + let num_shifted_limbs = shift / 60; + let limb_shift = (shift % 60) as u8; + + let mut result = [0; 6]; + result[num_shifted_limbs] = (1 << limb_shift); + + for i in 1..(6 - num_shifted_limbs) { + result[i + num_shifted_limbs] = 0; + } + result +} diff --git a/noir/noir-repo/test_programs/execution_success/regression_unsafe_no_predicates/src/main.nr b/noir/noir-repo/test_programs/execution_success/regression_unsafe_no_predicates/src/main.nr index fc1e55ee641..b4dd6a57e80 100644 --- a/noir/noir-repo/test_programs/execution_success/regression_unsafe_no_predicates/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/regression_unsafe_no_predicates/src/main.nr @@ -7,6 +7,7 @@ fn main(x: u8, nest: bool) { #[no_predicates] pub fn unsafe_assert(msg: [u8; N]) -> u8 { + /// Safety: testing context let block = unsafe { get_block(msg) }; verify_block(msg, block); block[0] diff --git a/noir/noir-repo/test_programs/execution_success/sha2_byte/src/main.nr b/noir/noir-repo/test_programs/execution_success/sha2_byte/src/main.nr index 1b4bed1a643..a1663642c69 100644 --- a/noir/noir-repo/test_programs/execution_success/sha2_byte/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/sha2_byte/src/main.nr @@ -1,8 +1,8 @@ // Test Noir implementations of SHA256 and SHA512 on a one-byte message. fn main(x: Field, result256: [u8; 32], result512: [u8; 64]) { - let digest256 = std::sha256::digest([x as u8]); + let digest256 = std::hash::sha256([x as u8]); assert(digest256 == result256); - let digest512 = std::sha512::digest([x as u8]); + let digest512 = std::hash::sha512::digest([x as u8]); assert(digest512 == result512); } diff --git a/noir/noir-repo/test_programs/execution_success/slice_regex/src/main.nr b/noir/noir-repo/test_programs/execution_success/slice_regex/src/main.nr index 4ba33e83903..67901f10f29 100644 --- a/noir/noir-repo/test_programs/execution_success/slice_regex/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/slice_regex/src/main.nr @@ -21,19 +21,19 @@ impl Eq for Match { // impl From for str trait Regex { - fn match(self, input: [u8]) -> Match; + fn find_match(self, input: [u8]) -> Match; } // Empty impl Regex for () { - fn match(_self: Self, input: [u8]) -> Match { + fn find_match(_self: Self, input: [u8]) -> Match { Match::empty(input) } } // Exact impl Regex for str { - fn match(self, input: [u8]) -> Match { + fn find_match(self, input: [u8]) -> Match { let mut leftover = input; let mut matches_input = true; let self_as_bytes = self.as_bytes(); @@ -60,10 +60,10 @@ where T: Regex, U: Regex, { - fn match(self, input: [u8]) -> Match { - let lhs_result = self.0.match(input); + fn find_match(self, input: [u8]) -> Match { + let lhs_result = self.0.find_match(input); if lhs_result.succeeded { - let rhs_result = self.1.match(lhs_result.leftover); + let rhs_result = self.1.find_match(lhs_result.leftover); if rhs_result.succeeded { Match { succeeded: true, @@ -88,11 +88,11 @@ impl Regex for Repeated where T: Regex, { - fn match(self, input: [u8]) -> Match { + fn find_match(self, input: [u8]) -> Match { let mut result = Match::empty(input); for _ in 0..N { if result.succeeded { - let next_result = self.inner.match(result.leftover); + let next_result = self.inner.find_match(result.leftover); result = Match { succeeded: next_result.succeeded, match_ends: result.match_ends + next_result.match_ends, @@ -114,12 +114,12 @@ where T: Regex, U: Regex, { - fn match(self, input: [u8]) -> Match { - let lhs_result = self.lhs.match(input); + fn find_match(self, input: [u8]) -> Match { + let lhs_result = self.lhs.find_match(input); if lhs_result.succeeded { lhs_result } else { - self.rhs.match(input) + self.rhs.find_match(input) } } } @@ -132,8 +132,8 @@ impl Regex for Question where T: Regex, { - fn match(self, input: [u8]) -> Match { - Or { lhs: self.inner, rhs: () }.match(input) + fn find_match(self, input: [u8]) -> Match { + Or { lhs: self.inner, rhs: () }.find_match(input) } } @@ -146,9 +146,9 @@ impl Regex for Star where T: Regex, { - fn match(self, input: [u8]) -> Match { + fn find_match(self, input: [u8]) -> Match { let regex: Repeated<_, N> = Repeated { inner: Question { inner: self.inner } }; - regex.match(input) + regex.find_match(input) } } @@ -161,10 +161,10 @@ impl Regex for Plus where T: Regex, { - fn match(self, input: [u8]) -> Match { + fn find_match(self, input: [u8]) -> Match { std::static_assert(N_PRED + 1 == N, "N - 1 != N_PRED"); let star: Star = Star { inner: self.inner }; - (self.inner, star).match(input) + (self.inner, star).find_match(input) } } @@ -173,23 +173,23 @@ fn main() { let graey_regex = ("gr", (Or { lhs: "a", rhs: "e" }, "y")); // NOTE: leftover ignored in Eq: Match - let result = graey_regex.match("gray".as_bytes().as_slice()); + let result = graey_regex.find_match("gray".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 4, leftover: &[] }); // NOTE: leftover ignored in Eq: Match - let result = graey_regex.match("grey".as_bytes().as_slice()); + let result = graey_regex.find_match("grey".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 4, leftover: &[] }); // colou?r let colour_regex = ("colo", (Question { inner: "u" }, "r")); - let result = colour_regex.match("color".as_bytes().as_slice()); + let result = colour_regex.find_match("color".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 5, leftover: &[] }); - let result = colour_regex.match("colour".as_bytes().as_slice()); + let result = colour_regex.find_match("colour".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 6, leftover: &[] }); @@ -197,35 +197,35 @@ fn main() { // EMPTY{3} let three_empties_regex: Repeated<(), 3> = Repeated { inner: () }; - let result = three_empties_regex.match("111".as_bytes().as_slice()); + let result = three_empties_regex.find_match("111".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 0, leftover: &[] }); // 1{0} let zero_ones_regex: Repeated, 0> = Repeated { inner: "1" }; - let result = zero_ones_regex.match("111".as_bytes().as_slice()); + let result = zero_ones_regex.find_match("111".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 0, leftover: &[] }); // 1{1} let one_ones_regex: Repeated, 1> = Repeated { inner: "1" }; - let result = one_ones_regex.match("111".as_bytes().as_slice()); + let result = one_ones_regex.find_match("111".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 1, leftover: &[] }); // 1{2} let two_ones_regex: Repeated, 2> = Repeated { inner: "1" }; - let result = two_ones_regex.match("111".as_bytes().as_slice()); + let result = two_ones_regex.find_match("111".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 2, leftover: &[] }); // 1{3} let three_ones_regex: Repeated, 3> = Repeated { inner: "1" }; - let result = three_ones_regex.match("1111".as_bytes().as_slice()); + let result = three_ones_regex.find_match("1111".as_bytes().as_slice()); println(result); assert_eq(result, Match { succeeded: true, match_ends: 3, leftover: &[] }); // TODO(https://github.com/noir-lang/noir/issues/6285): re-enable these cases and complete the test using array_regex below @@ -233,15 +233,15 @@ fn main() { // // 1* // let ones_regex: Star, 5> = Star { inner: "1" }; // - // let result = ones_regex.match("11000".as_bytes().as_slice()); + // let result = ones_regex.find_match("11000".as_bytes().as_slice()); // println(result); // assert_eq(result, Match { succeeded: true, match_ends: 2, leftover: &[] }); // - // let result = ones_regex.match("11".as_bytes().as_slice()); + // let result = ones_regex.find_match("11".as_bytes().as_slice()); // println(result); // assert_eq(result, Match { succeeded: true, match_ends: 2, leftover: &[] }); // - // let result = ones_regex.match("111111".as_bytes().as_slice()); + // let result = ones_regex.find_match("111111".as_bytes().as_slice()); // println(result); // assert_eq(result, Match { succeeded: true, match_ends: 5, leftover: &[] }); // @@ -249,28 +249,28 @@ fn main() { // // 1+ // let nonempty_ones_regex: Plus, 5, 4> = Plus { inner: "1" }; // - // let result = nonempty_ones_regex.match("111111".as_bytes().as_slice()); + // let result = nonempty_ones_regex.find_match("111111".as_bytes().as_slice()); // println(result); // assert_eq(result, Match { succeeded: true, match_ends: 5, leftover: &[] }); // // // 2^n-1 in binary: 1+0 // let pred_pow_two_regex = (nonempty_ones_regex, "0"); // - // let result = pred_pow_two_regex.match("1110".as_bytes().as_slice()); + // let result = pred_pow_two_regex.find_match("1110".as_bytes().as_slice()); // println(result); // assert_eq(result, Match { succeeded: true, match_ends: 3, leftover: &[] }); // // // (0|1)* // let binary_regex: Star, str<1>>, 5> = Star { inner: Or { lhs: "0", rhs: "1" } }; // - // let result = binary_regex.match("110100".as_bytes().as_slice()); + // let result = binary_regex.find_match("110100".as_bytes().as_slice()); // println(result); // assert_eq(result, Match { succeeded: true, match_ends: 5, leftover: &[] }); // // // even numbers in binary: 1(0|1)*0 // let even_binary_regex = ("1", (binary_regex, "0")); // - // let result = even_binary_regex.match("1111110".as_bytes().as_slice()); + // let result = even_binary_regex.find_match("1111110".as_bytes().as_slice()); // println(result); // assert_eq(result, Match { succeeded: true, match_ends: 6, leftover: &[] }); // 2-letter capitalized words: [A-Z][a-z] @@ -290,7 +290,7 @@ fn main() { // ) // ); // - // let result = foo_regex.match("colo".as_bytes().as_slice()); + // let result = foo_regex.find_match("colo".as_bytes().as_slice()); // println(result); // assert_eq(result, Match { // succeeded: true, @@ -387,19 +387,19 @@ fn main() { // // trait Regex { // // Perform a match without backtracking -// fn match(self, input: Bvec) -> Match; +// fn find_match(self, input: Bvec) -> Match; // } // // // Empty // impl Regex for () { -// fn match(_self: Self, input: Bvec) -> Match { +// fn find_match(_self: Self, input: Bvec) -> Match { // Match::empty(input) // } // } // // // Exact // impl Regex for str { -// fn match(self, input: Bvec) -> Match { +// fn find_match(self, input: Bvec) -> Match { // let mut leftover = input; // let mut matches_input = true; // let self_as_bytes = self.as_bytes(); @@ -430,10 +430,10 @@ fn main() { // // // And // impl Regex for (T, U) where T: Regex, U: Regex { -// fn match(self, input: Bvec) -> Match { -// let lhs_result = self.0.match(input); +// fn find_match(self, input: Bvec) -> Match { +// let lhs_result = self.0.find_match(input); // if lhs_result.succeeded { -// let rhs_result = self.1.match(lhs_result.leftover); +// let rhs_result = self.1.find_match(lhs_result.leftover); // if rhs_result.succeeded { // Match { // succeeded: true, @@ -463,11 +463,11 @@ fn main() { // } // // impl Regex for Repeated where T: Regex { -// fn match(self, input: Bvec) -> Match { +// fn find_match(self, input: Bvec) -> Match { // let mut result = Match::empty(input); // for _ in 0..M { // if result.succeeded { -// let next_result = self.inner.match(result.leftover); +// let next_result = self.inner.find_match(result.leftover); // result = Match { // succeeded: next_result.succeeded, // match_ends: result.match_ends + next_result.match_ends, @@ -485,12 +485,12 @@ fn main() { // } // // impl Regex for Or where T: Regex, U: Regex { -// fn match(self, input: Bvec) -> Match { -// let lhs_result = self.lhs.match(input); +// fn find_match(self, input: Bvec) -> Match { +// let lhs_result = self.lhs.find_match(input); // if lhs_result.succeeded { // lhs_result // } else { -// self.rhs.match(input) +// self.rhs.find_match(input) // } // } // } @@ -500,11 +500,11 @@ fn main() { // } // // impl Regex for Question where T: Regex { -// fn match(self, input: Bvec) -> Match { +// fn find_match(self, input: Bvec) -> Match { // Or { // lhs: self.inner, // rhs: (), -// }.match(input) +// }.find_match(input) // } // } // @@ -514,11 +514,11 @@ fn main() { // } // // impl Regex for Star where T: Regex { -// fn match(self, input: Bvec) -> Match { +// fn find_match(self, input: Bvec) -> Match { // let regex: Repeated<_, M> = Repeated { // inner: Question { inner: self.inner }, // }; -// regex.match(input) +// regex.find_match(input) // } // } // @@ -528,13 +528,13 @@ fn main() { // } // // impl Regex for Plus where T: Regex { -// fn match(self, input: Bvec) -> Match { +// fn find_match(self, input: Bvec) -> Match { // std::static_assert(M_PRED + 1 == M, "M - 1 != M_PRED"); // let star: Star = Star { inner: self.inner }; // ( // self.inner, // star -// ).match(input) +// ).find_match(input) // } // } // @@ -544,11 +544,11 @@ fn main() { // } // // impl Regex for AnyOf where T: Regex { -// fn match(self, input: Bvec) -> Match { +// fn find_match(self, input: Bvec) -> Match { // let mut result = Match::failed(input); // for i in 0..M { // if !result.succeeded { -// result = self.inner[i].match(result.leftover); +// result = self.inner[i].find_match(result.leftover); // } // } // result @@ -611,13 +611,13 @@ fn main() { // // gr(a|e)y // let graey_regex = ("gr", (Or { lhs: "a", rhs: "e" }, "y")); // -// let result = graey_regex.match(Bvec::new("gray".as_bytes())); +// let result = graey_regex.find_match(Bvec::new("gray".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 4); // assert_eq(result.leftover.len, 0); // -// let result = graey_regex.match(Bvec::new("grey".as_bytes())); +// let result = graey_regex.find_match(Bvec::new("grey".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 4); @@ -626,13 +626,13 @@ fn main() { // // colou?r // let colour_regex = ("colo", (Question { inner: "u" }, "r")); // -// let result = colour_regex.match(Bvec::new("color".as_bytes())); +// let result = colour_regex.find_match(Bvec::new("color".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 5); // assert_eq(result.leftover.len, 0); // -// let result = colour_regex.match(Bvec::new("colour".as_bytes())); +// let result = colour_regex.find_match(Bvec::new("colour".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 6); @@ -642,7 +642,7 @@ fn main() { // // EMPTY{3} // let three_empties_regex: Repeated<(), 3> = Repeated { inner: () }; // -// let result = three_empties_regex.match(Bvec::new("111".as_bytes())); +// let result = three_empties_regex.find_match(Bvec::new("111".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 0); @@ -651,7 +651,7 @@ fn main() { // // 1{0} // let zero_ones_regex: Repeated, 0> = Repeated { inner: "1" }; // -// let result = zero_ones_regex.match(Bvec::new("111".as_bytes())); +// let result = zero_ones_regex.find_match(Bvec::new("111".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 0); @@ -660,7 +660,7 @@ fn main() { // // 1{1} // let one_ones_regex: Repeated, 1> = Repeated { inner: "1" }; // -// let result = one_ones_regex.match(Bvec::new("111".as_bytes())); +// let result = one_ones_regex.find_match(Bvec::new("111".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 1); @@ -669,7 +669,7 @@ fn main() { // // 1{2} // let two_ones_regex: Repeated, 2> = Repeated { inner: "1" }; // -// let result = two_ones_regex.match(Bvec::new("111".as_bytes())); +// let result = two_ones_regex.find_match(Bvec::new("111".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 2); @@ -678,7 +678,7 @@ fn main() { // // 1{3} // let three_ones_regex: Repeated, 3> = Repeated { inner: "1" }; // -// let result = three_ones_regex.match(Bvec::new("1111".as_bytes())); +// let result = three_ones_regex.find_match(Bvec::new("1111".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 3); @@ -687,19 +687,19 @@ fn main() { // // 1* // let ones_regex: Star, 5> = Star { inner: "1" }; // -// let result = ones_regex.match(Bvec::new("11000".as_bytes())); +// let result = ones_regex.find_match(Bvec::new("11000".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 2); // assert_eq(result.leftover.len, 3); // -// let result = ones_regex.match(Bvec::new("11".as_bytes())); +// let result = ones_regex.find_match(Bvec::new("11".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 2); // assert_eq(result.leftover.len, 0); // -// let result = ones_regex.match(Bvec::new("111111".as_bytes())); +// let result = ones_regex.find_match(Bvec::new("111111".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 5); @@ -708,7 +708,7 @@ fn main() { // // 1+ // let nonempty_ones_regex: Plus, 5, 4> = Plus { inner: "1" }; // -// let result = nonempty_ones_regex.match(Bvec::new("111111".as_bytes())); +// let result = nonempty_ones_regex.find_match(Bvec::new("111111".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 5); @@ -717,7 +717,7 @@ fn main() { // // 2^n-1 in binary: 1+0 // let pred_pow_two_regex = (nonempty_ones_regex, "0"); // -// let result = pred_pow_two_regex.match(Bvec::new("1110".as_bytes())); +// let result = pred_pow_two_regex.find_match(Bvec::new("1110".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 4); @@ -726,7 +726,7 @@ fn main() { // // (0|1)* // let binary_regex: Star, str<1>>, 5> = Star { inner: Or { lhs: "0", rhs: "1" } }; // -// let result = binary_regex.match(Bvec::new("110100".as_bytes())); +// let result = binary_regex.find_match(Bvec::new("110100".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 5); @@ -735,7 +735,7 @@ fn main() { // // even numbers in binary: 1(0|1)*0 // let even_binary_regex = ("1", (binary_regex, "0")); // -// let result = even_binary_regex.match(Bvec::new("1111110".as_bytes())); +// let result = even_binary_regex.find_match(Bvec::new("1111110".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 7); @@ -758,13 +758,13 @@ fn main() { // ] // }; // -// let result = digit_regex.match(Bvec::new("157196345823795".as_bytes())); +// let result = digit_regex.find_match(Bvec::new("157196345823795".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 1); // assert_eq(result.leftover.len, 14); // -// let result = digit_regex.match(Bvec::new("hi".as_bytes())); +// let result = digit_regex.find_match(Bvec::new("hi".as_bytes())); // println(result); // assert(!result.succeeded); // assert_eq(result.match_ends, 0); @@ -774,13 +774,13 @@ fn main() { // // [0-9]+ // let digits_regex: Plus, 10>, 32, 31> = Plus { inner: digit_regex }; // -// let result = digits_regex.match(Bvec::new("123456789012345".as_bytes())); +// let result = digits_regex.find_match(Bvec::new("123456789012345".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 15); // assert_eq(result.leftover.len, 0); // -// let result = digits_regex.match(Bvec::new("123456789012345 then words".as_bytes())); +// let result = digits_regex.find_match(Bvec::new("123456789012345 then words".as_bytes())); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 15); @@ -791,14 +791,14 @@ fn main() { // // 0\d+ // let backwards_mult_of_10_regex = ("0", digits_regex); // -// let result = backwards_mult_of_10_regex.match(Bvec::new(reverse_array("1230".as_bytes()))); +// let result = backwards_mult_of_10_regex.find_match(Bvec::new(reverse_array("1230".as_bytes()))); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 4); // assert_eq(result.leftover.len, 0); // // let ten_pow_16: str<17> = "10000000000000000"; -// let result = backwards_mult_of_10_regex.match(Bvec::new(reverse_array(ten_pow_16.as_bytes()))); +// let result = backwards_mult_of_10_regex.find_match(Bvec::new(reverse_array(ten_pow_16.as_bytes()))); // println(result); // assert(result.succeeded); // assert_eq(result.match_ends, 17); diff --git a/noir/noir-repo/test_programs/execution_success/u16_support/src/main.nr b/noir/noir-repo/test_programs/execution_success/u16_support/src/main.nr index ca41c708077..d86ad348708 100644 --- a/noir/noir-repo/test_programs/execution_success/u16_support/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/u16_support/src/main.nr @@ -1,5 +1,6 @@ fn main(x: u16) { test_u16(x); + /// Safety: testing context unsafe { test_u16_unconstrained(x); } diff --git a/noir/noir-repo/test_programs/execution_success/uhashmap/src/main.nr b/noir/noir-repo/test_programs/execution_success/uhashmap/src/main.nr index 689ba9d4a04..6163d5b954b 100644 --- a/noir/noir-repo/test_programs/execution_success/uhashmap/src/main.nr +++ b/noir/noir-repo/test_programs/execution_success/uhashmap/src/main.nr @@ -293,6 +293,7 @@ unconstrained fn doc_tests() { // docs:start:get_example fn get_example(map: UHashMap>) { + /// Safety: testing context let x = unsafe { map.get(12) }; if x.is_some() { @@ -318,6 +319,7 @@ fn entries_examples(map: UHashMap {value}"); } diff --git a/noir/noir-repo/test_programs/gates_report.sh b/noir/noir-repo/test_programs/gates_report.sh index 12e9ae0f864..c92346f40e8 100755 --- a/noir/noir-repo/test_programs/gates_report.sh +++ b/noir/noir-repo/test_programs/gates_report.sh @@ -24,13 +24,13 @@ for pathname in $test_dirs; do fi GATES_INFO=$($BACKEND gates -b "$artifacts_path/$ARTIFACT_NAME/target/program.json") - MAIN_FUNCTION_INFO=$(echo $GATES_INFO | jq -r '.functions[0] | .name = "main"') - echo "{\"package_name\": \"$ARTIFACT_NAME\", \"functions\": [$MAIN_FUNCTION_INFO]" >> gates_report.json + MAIN_FUNCTION_INFO=$(echo $GATES_INFO | jq -r ".functions[0] | {package_name: "\"$ARTIFACT_NAME\"", functions: [{name: \"main\", acir_opcodes, opcodes: .acir_opcodes, circuit_size}], unconstrained_functions: []}") + echo -n $MAIN_FUNCTION_INFO >> gates_report.json if (($ITER == $NUM_ARTIFACTS)); then - echo "}" >> gates_report.json + echo "" >> gates_report.json else - echo "}, " >> gates_report.json + echo "," >> gates_report.json fi ITER=$(( $ITER + 1 )) diff --git a/noir/noir-repo/test_programs/gates_report_brillig.sh b/noir/noir-repo/test_programs/gates_report_brillig.sh index d3f6344dbf4..3f2a4542735 100755 --- a/noir/noir-repo/test_programs/gates_report_brillig.sh +++ b/noir/noir-repo/test_programs/gates_report_brillig.sh @@ -28,6 +28,6 @@ done echo "]" >> Nargo.toml -nargo info --force-brillig --json > gates_report_brillig.json +nargo info --silence-warnings --force-brillig --json --inliner-aggressiveness $1 | jq -r ".programs[].functions = []" > gates_report_brillig.json rm Nargo.toml diff --git a/noir/noir-repo/test_programs/gates_report_brillig_execution.sh b/noir/noir-repo/test_programs/gates_report_brillig_execution.sh index b3f4a8bda98..c6a904f8b93 100755 --- a/noir/noir-repo/test_programs/gates_report_brillig_execution.sh +++ b/noir/noir-repo/test_programs/gates_report_brillig_execution.sh @@ -3,10 +3,10 @@ set -e # These tests are incompatible with gas reporting excluded_dirs=( - "workspace" - "workspace_default_member" - "double_verify_nested_proof" - "overlapping_dep_and_mod" + "workspace" + "workspace_default_member" + "double_verify_nested_proof" + "overlapping_dep_and_mod" "comptime_println" # bit sizes for bigint operation doesn't match up. "bigint" @@ -33,11 +33,15 @@ for dir in $test_dirs; do continue fi + if [[ ! -f "${base_path}/${dir}/Nargo.toml" ]]; then + continue + fi + echo " \"execution_success/$dir\"," >> Nargo.toml done echo "]" >> Nargo.toml -nargo info --profile-execution --json > gates_report_brillig_execution.json +nargo info --silence-warnings --profile-execution --json --inliner-aggressiveness $1 | jq -r ".programs[].functions = []" > gates_report_brillig_execution.json -rm Nargo.toml \ No newline at end of file +rm Nargo.toml diff --git a/noir/noir-repo/test_programs/memory_report.sh b/noir/noir-repo/test_programs/memory_report.sh index 4d03726d374..4ba7579b390 100755 --- a/noir/noir-repo/test_programs/memory_report.sh +++ b/noir/noir-repo/test_programs/memory_report.sh @@ -4,6 +4,8 @@ set -e sudo apt-get install heaptrack NARGO="nargo" +PARSE_MEMORY=$(realpath "$(dirname "$0")/parse_memory.sh") + # Tests to be profiled for memory report tests_to_profile=("keccak256" "workspace" "regression_4709" "ram_blowup_regression") @@ -12,13 +14,14 @@ current_dir=$(pwd) base_path="$current_dir/execution_success" # If there is an argument that means we want to generate a report for only the current directory -if [ "$#" -ne 0 ]; then +if [ "$1" == "1" ]; then base_path="$current_dir" tests_to_profile=(".") fi FIRST="1" +FLAGS=${FLAGS:- ""} echo "{\"memory_reports\": [ " > memory_report.json for test_name in ${tests_to_profile[@]}; do @@ -31,11 +34,16 @@ for test_name in ${tests_to_profile[@]}; do echo " ," >> $current_dir"/memory_report.json" fi - if [ "$#" -ne 0 ]; then + if [ "$1" == "1" ]; then test_name=$(basename $current_dir) fi - heaptrack --output $current_dir/$test_name"_heap" $NARGO compile --force + COMMAND="compile --force --silence-warnings $FLAGS" + if [ "$2" == "1" ]; then + COMMAND="execute --silence-warnings" + fi + + heaptrack --output $current_dir/$test_name"_heap" $NARGO $COMMAND if test -f $current_dir/$test_name"_heap.gz"; then heaptrack --analyze $current_dir/$test_name"_heap.gz" > $current_dir/$test_name"_heap_analysis.txt" @@ -48,7 +56,8 @@ for test_name in ${tests_to_profile[@]}; do len=${#consumption}-30 peak=${consumption:30:len} rm $current_dir/$test_name"_heap_analysis.txt" - echo -e " {\n \"artifact_name\":\"$test_name\",\n \"peak_memory\":\"$peak\"\n }" >> $current_dir"/memory_report.json" + peak_memory=$($PARSE_MEMORY $peak) + echo -e " {\n \"artifact_name\":\"$test_name\",\n \"peak_memory\":\"$peak_memory\"\n }" >> $current_dir"/memory_report.json" done echo "]}" >> $current_dir"/memory_report.json" diff --git a/noir/noir-repo/test_programs/execution_success/brillig_oracle/Nargo.toml b/noir/noir-repo/test_programs/noir_test_success/brillig_oracle/Nargo.toml similarity index 100% rename from noir/noir-repo/test_programs/execution_success/brillig_oracle/Nargo.toml rename to noir/noir-repo/test_programs/noir_test_success/brillig_oracle/Nargo.toml diff --git a/noir/noir-repo/test_programs/execution_success/brillig_oracle/src/main.nr b/noir/noir-repo/test_programs/noir_test_success/brillig_oracle/src/main.nr similarity index 95% rename from noir/noir-repo/test_programs/execution_success/brillig_oracle/src/main.nr rename to noir/noir-repo/test_programs/noir_test_success/brillig_oracle/src/main.nr index 8f5b2fa7566..77dbeef9aa1 100644 --- a/noir/noir-repo/test_programs/execution_success/brillig_oracle/src/main.nr +++ b/noir/noir-repo/test_programs/noir_test_success/brillig_oracle/src/main.nr @@ -1,8 +1,14 @@ use std::slice; use std::test::OracleMock; +#[test] +fn test_main() { + main(10); +} + // Tests oracle usage in brillig/unconstrained functions fn main(_x: Field) { + /// Safety: testing context unsafe { let size = 20; // TODO: Add a method along the lines of `(0..size).to_array()`. @@ -45,4 +51,3 @@ unconstrained fn get_number_sequence_wrapper(size: Field) { assert(slice[i] == reversed_slice[19 - i]); } } - diff --git a/noir/noir-repo/test_programs/noir_test_success/comptime_blackbox/src/main.nr b/noir/noir-repo/test_programs/noir_test_success/comptime_blackbox/src/main.nr index c3784e73b09..859790f7d2d 100644 --- a/noir/noir-repo/test_programs/noir_test_success/comptime_blackbox/src/main.nr +++ b/noir/noir-repo/test_programs/noir_test_success/comptime_blackbox/src/main.nr @@ -121,10 +121,12 @@ fn test_sha256() { #[test] fn test_embedded_curve_ops() { let (sum, mul) = comptime { - let g1 = EmbeddedCurvePoint { x: 1, y: 17631683881184975370165255887551781615748388533673675138860, is_infinite: false }; let s1 = EmbeddedCurveScalar { lo: 1, hi: 0 }; - let sum = g1 + g1; - let mul = multi_scalar_mul([g1, g1], [s1, s1]); + let s2 = EmbeddedCurveScalar { lo: 2, hi: 0 }; + let g1 = EmbeddedCurvePoint { x: 1, y: 17631683881184975370165255887551781615748388533673675138860, is_infinite: false }; + let g2 = multi_scalar_mul([g1], [s2]); + let sum = g1 + g2; + let mul = multi_scalar_mul([g1, g2], [s1, s1]); (sum, mul) }; assert_eq(sum, mul); diff --git a/noir/noir-repo/test_programs/noir_test_success/comptime_expr/src/main.nr b/noir/noir-repo/test_programs/noir_test_success/comptime_expr/src/main.nr index 709180879a0..6efbc212cbe 100644 --- a/noir/noir-repo/test_programs/noir_test_success/comptime_expr/src/main.nr +++ b/noir/noir-repo/test_programs/noir_test_success/comptime_expr/src/main.nr @@ -1,11 +1,10 @@ mod tests { - use std::meta::op::UnaryOp; use std::meta::op::BinaryOp; + use std::meta::op::UnaryOp; #[test] fn test_expr_as_array() { - comptime - { + comptime { let expr = quote { [1, 2, 4] }.as_expr().unwrap(); let elems = expr.as_array().unwrap(); assert_eq(elems.len(), 3); @@ -17,8 +16,7 @@ mod tests { #[test] fn test_expr_modify_for_array() { - comptime - { + comptime { let expr = quote { [1, 2, 4] }.as_expr().unwrap(); let expr = expr.modify(times_two); let elems = expr.as_array().unwrap(); @@ -31,8 +29,7 @@ mod tests { #[test] fn test_expr_as_assert() { - comptime - { + comptime { let expr = quote { assert(true) }.as_expr().unwrap(); let (predicate, msg) = expr.as_assert().unwrap(); assert_eq(predicate.as_bool().unwrap(), true); @@ -47,8 +44,7 @@ mod tests { #[test] fn test_expr_modify_for_assert() { - comptime - { + comptime { let expr = quote { assert(1) }.as_expr().unwrap(); let expr = expr.modify(times_two); let (predicate, msg) = expr.as_assert().unwrap(); @@ -65,8 +61,7 @@ mod tests { #[test] fn test_expr_as_assert_eq() { - comptime - { + comptime { let expr = quote { assert_eq(true, false) }.as_expr().unwrap(); let (lhs, rhs, msg) = expr.as_assert_eq().unwrap(); assert_eq(lhs.as_bool().unwrap(), true); @@ -83,8 +78,7 @@ mod tests { #[test] fn test_expr_modify_for_assert_eq() { - comptime - { + comptime { let expr = quote { assert_eq(1, 2) }.as_expr().unwrap(); let expr = expr.modify(times_two); let (lhs, rhs, msg) = expr.as_assert_eq().unwrap(); @@ -103,8 +97,7 @@ mod tests { #[test] fn test_expr_as_assign() { - comptime - { + comptime { let expr = quote { { a = 1; } }.as_expr().unwrap(); let exprs = expr.as_block().unwrap(); let (_lhs, rhs) = exprs[0].as_assign().unwrap(); @@ -114,8 +107,7 @@ mod tests { #[test] fn test_expr_modify_for_assign() { - comptime - { + comptime { let expr = quote { { a = 1; } }.as_expr().unwrap(); let expr = expr.modify(times_two); let exprs = expr.as_block().unwrap(); @@ -126,8 +118,7 @@ mod tests { #[test] fn test_expr_as_block() { - comptime - { + comptime { let expr = quote { { 1; 4; 23 } }.as_expr().unwrap(); let exprs = expr.as_block().unwrap(); assert_eq(exprs.len(), 3); @@ -143,8 +134,7 @@ mod tests { #[test] fn test_expr_modify_for_block() { - comptime - { + comptime { let expr = quote { { 1; 4; 23 } }.as_expr().unwrap(); let expr = expr.modify(times_two); let exprs = expr.as_block().unwrap(); @@ -161,8 +151,7 @@ mod tests { #[test] fn test_expr_as_method_call() { - comptime - { + comptime { let expr = quote { foo.bar::(3, 4) }.as_expr().unwrap(); let (_object, name, generics, arguments) = expr.as_method_call().unwrap(); @@ -179,8 +168,7 @@ mod tests { #[test] fn test_expr_modify_for_method_call() { - comptime - { + comptime { let expr = quote { foo.bar(3, 4) }.as_expr().unwrap(); let expr = expr.modify(times_two); @@ -198,8 +186,7 @@ mod tests { #[test] fn test_expr_as_integer() { - comptime - { + comptime { let expr = quote { 1 }.as_expr().unwrap(); assert_eq((1, false), expr.as_integer().unwrap()); @@ -210,8 +197,7 @@ mod tests { #[test] fn test_expr_modify_for_integer() { - comptime - { + comptime { let expr = quote { 1 }.as_expr().unwrap(); let expr = expr.modify(times_two); @@ -221,8 +207,7 @@ mod tests { #[test] fn test_expr_as_binary_op() { - comptime - { + comptime { assert(get_binary_op(quote { x + y }).is_add()); assert(get_binary_op(quote { x - y }).is_subtract()); assert(get_binary_op(quote { x * y }).is_multiply()); @@ -244,8 +229,7 @@ mod tests { #[test] fn test_expr_modify_for_binary_op() { - comptime - { + comptime { let expr = quote { 3 + 4 }.as_expr().unwrap(); let expr = expr.modify(times_two); @@ -258,8 +242,7 @@ mod tests { #[test] fn test_expr_as_bool() { - comptime - { + comptime { let expr = quote { false }.as_expr().unwrap(); assert(expr.as_bool().unwrap() == false); @@ -270,8 +253,7 @@ mod tests { #[test] fn test_expr_as_cast() { - comptime - { + comptime { let expr = quote { 1 as Field }.as_expr().unwrap(); let (expr, typ) = expr.as_cast().unwrap(); assert_eq(expr.as_integer().unwrap(), (1, false)); @@ -281,8 +263,7 @@ mod tests { #[test] fn test_expr_modify_for_cast() { - comptime - { + comptime { let expr = quote { 1 as Field }.as_expr().unwrap(); let expr = expr.modify(times_two); let (expr, typ) = expr.as_cast().unwrap(); @@ -293,8 +274,7 @@ mod tests { #[test] fn test_expr_as_comptime() { - comptime - { + comptime { let expr = quote { comptime { 1; 4; 23 } }.as_expr().unwrap(); let exprs = expr.as_comptime().unwrap(); assert_eq(exprs.len(), 3); @@ -303,8 +283,7 @@ mod tests { #[test] fn test_expr_modify_for_comptime() { - comptime - { + comptime { let expr = quote { comptime { 1; 4; 23 } }.as_expr().unwrap(); let expr = expr.modify(times_two); let exprs = expr.as_comptime().unwrap(); @@ -315,8 +294,7 @@ mod tests { #[test] fn test_expr_as_comptime_as_statement() { - comptime - { + comptime { let expr = quote { { comptime { 1; 4; 23 } } }.as_expr().unwrap(); let exprs = expr.as_block().unwrap(); assert_eq(exprs.len(), 1); @@ -328,8 +306,7 @@ mod tests { #[test] fn test_expr_as_constructor() { - comptime - { + comptime { let expr = quote { Foo { a: 1, b: 2 } }.as_expr().unwrap(); let (_typ, fields) = expr.as_constructor().unwrap(); assert_eq(fields.len(), 2); @@ -342,8 +319,7 @@ mod tests { #[test] fn test_expr_modify_for_constructor() { - comptime - { + comptime { let expr = quote { foo::bar::Baz:: { a: 1, b: 2 } }.as_expr().unwrap(); let expr = expr.modify(times_two); let (_typ, fields) = expr.as_constructor().unwrap(); @@ -360,8 +336,7 @@ mod tests { // docs:start:as_expr_example #[test] fn test_expr_as_function_call() { - comptime - { + comptime { let expr = quote { foo(42) }.as_expr().unwrap(); let (_function, args) = expr.as_function_call().unwrap(); assert_eq(args.len(), 1); @@ -372,8 +347,7 @@ mod tests { #[test] fn test_expr_modify_for_function_call() { - comptime - { + comptime { let expr = quote { foo(42) }.as_expr().unwrap(); let expr = expr.modify(times_two); let (_function, args) = expr.as_function_call().unwrap(); @@ -384,8 +358,7 @@ mod tests { #[test] fn test_expr_as_if() { - comptime - { + comptime { let expr = quote { if 1 { 2 } }.as_expr().unwrap(); let (_condition, _consequence, alternative) = expr.as_if().unwrap(); assert(alternative.is_none()); @@ -398,8 +371,7 @@ mod tests { #[test] fn test_expr_modify_for_if() { - comptime - { + comptime { let expr = quote { if 1 { 2 } }.as_expr().unwrap(); let expr = expr.modify(times_two); let (condition, consequence, alternative) = expr.as_if().unwrap(); @@ -421,8 +393,7 @@ mod tests { #[test] fn test_expr_as_index() { - comptime - { + comptime { let expr = quote { foo[bar] }.as_expr().unwrap(); assert(expr.as_index().is_some()); } @@ -430,8 +401,7 @@ mod tests { #[test] fn test_expr_modify_for_index() { - comptime - { + comptime { let expr = quote { 1[2] }.as_expr().unwrap(); let expr = expr.modify(times_two); let (object, index) = expr.as_index().unwrap(); @@ -442,8 +412,7 @@ mod tests { #[test] fn test_expr_as_member_access() { - comptime - { + comptime { let expr = quote { foo.bar }.as_expr().unwrap(); let (_, name) = expr.as_member_access().unwrap(); assert_eq(name, quote { bar }); @@ -452,8 +421,7 @@ mod tests { #[test] fn test_expr_modify_for_member_access() { - comptime - { + comptime { let expr = quote { 1.bar }.as_expr().unwrap(); let expr = expr.modify(times_two); let (expr, name) = expr.as_member_access().unwrap(); @@ -464,8 +432,7 @@ mod tests { #[test] fn test_expr_as_member_access_with_an_lvalue() { - comptime - { + comptime { let expr = quote { { foo.bar = 1; } }.as_expr().unwrap(); let exprs = expr.as_block().unwrap(); let (lhs, _rhs) = exprs[0].as_assign().unwrap(); @@ -476,8 +443,7 @@ mod tests { #[test] fn test_expr_as_repeated_element_array() { - comptime - { + comptime { let expr = quote { [1; 3] }.as_expr().unwrap(); let (expr, length) = expr.as_repeated_element_array().unwrap(); assert_eq(expr.as_integer().unwrap(), (1, false)); @@ -487,8 +453,7 @@ mod tests { #[test] fn test_expr_modify_for_repeated_element_array() { - comptime - { + comptime { let expr = quote { [1; 3] }.as_expr().unwrap(); let expr = expr.modify(times_two); let (expr, length) = expr.as_repeated_element_array().unwrap(); @@ -499,8 +464,7 @@ mod tests { #[test] fn test_expr_as_repeated_element_slice() { - comptime - { + comptime { let expr = quote { &[1; 3] }.as_expr().unwrap(); let (expr, length) = expr.as_repeated_element_slice().unwrap(); assert_eq(expr.as_integer().unwrap(), (1, false)); @@ -510,8 +474,7 @@ mod tests { #[test] fn test_expr_modify_for_repeated_element_slice() { - comptime - { + comptime { let expr = quote { &[1; 3] }.as_expr().unwrap(); let expr = expr.modify(times_two); let (expr, length) = expr.as_repeated_element_slice().unwrap(); @@ -522,8 +485,7 @@ mod tests { #[test] fn test_expr_as_slice() { - comptime - { + comptime { let expr = quote { &[1, 3, 5] }.as_expr().unwrap(); let elems = expr.as_slice().unwrap(); assert_eq(elems.len(), 3); @@ -535,8 +497,7 @@ mod tests { #[test] fn test_expr_modify_for_slice() { - comptime - { + comptime { let expr = quote { &[1, 3, 5] }.as_expr().unwrap(); let expr = expr.modify(times_two); let elems = expr.as_slice().unwrap(); @@ -549,8 +510,7 @@ mod tests { #[test] fn test_expr_as_tuple() { - comptime - { + comptime { let expr = quote { (1, 2) }.as_expr().unwrap(); let tuple_exprs = expr.as_tuple().unwrap(); assert_eq(tuple_exprs.len(), 2); @@ -559,8 +519,7 @@ mod tests { #[test] fn test_expr_modify_for_tuple() { - comptime - { + comptime { let expr = quote { (1, 2) }.as_expr().unwrap(); let expr = expr.modify(times_two); let tuple_exprs = expr.as_tuple().unwrap(); @@ -572,8 +531,7 @@ mod tests { #[test] fn test_expr_as_unary_op() { - comptime - { + comptime { assert(get_unary_op(quote { -x }).is_minus()); assert(get_unary_op(quote { !x }).is_not()); assert(get_unary_op(quote { &mut x }).is_mutable_reference()); @@ -583,8 +541,7 @@ mod tests { #[test] fn test_expr_modify_for_unary_op() { - comptime - { + comptime { let expr = quote { -(1) }.as_expr().unwrap(); let expr = expr.modify(times_two); let (op, expr) = expr.as_unary_op().unwrap(); @@ -595,9 +552,13 @@ mod tests { #[test] fn test_expr_as_unsafe() { - comptime - { - let expr = quote { unsafe { 1; 4; 23 } }.as_expr().unwrap(); + comptime { + let expr = quote { + /// Safety: test + unsafe { 1; 4; 23 } + } + .as_expr() + .unwrap(); let exprs = expr.as_unsafe().unwrap(); assert_eq(exprs.len(), 3); } @@ -605,9 +566,13 @@ mod tests { #[test] fn test_expr_modify_for_unsafe() { - comptime - { - let expr = quote { unsafe { 1; 4; 23 } }.as_expr().unwrap(); + comptime { + let expr = quote { + /// Safety: test + unsafe { 1; 4; 23 } + } + .as_expr() + .unwrap(); let expr = expr.modify(times_two); let exprs = expr.as_unsafe().unwrap(); assert_eq(exprs.len(), 3); @@ -617,8 +582,7 @@ mod tests { #[test] fn test_expr_is_break() { - comptime - { + comptime { let expr = quote { { break; } }.as_expr().unwrap(); let exprs = expr.as_block().unwrap(); assert(exprs[0].is_break()); @@ -627,8 +591,7 @@ mod tests { #[test] fn test_expr_is_continue() { - comptime - { + comptime { let expr = quote { { continue; } }.as_expr().unwrap(); let exprs = expr.as_block().unwrap(); assert(exprs[0].is_continue()); @@ -637,8 +600,7 @@ mod tests { #[test] fn test_expr_as_lambda() { - comptime - { + comptime { let expr = quote { |x: Field| -> Field { 1 } }.as_expr().unwrap(); let (params, return_type, body) = expr.as_lambda().unwrap(); assert_eq(params.len(), 1); @@ -657,15 +619,17 @@ mod tests { #[test] fn test_expr_modify_lambda() { - comptime - { + comptime { let expr = quote { |x: Field| -> Field { 1 } }.as_expr().unwrap(); let expr = expr.modify(times_two); let (params, return_type, body) = expr.as_lambda().unwrap(); assert_eq(params.len(), 1); assert(params[0].1.unwrap().is_field()); assert(return_type.unwrap().is_field()); - assert_eq(body.as_block().unwrap()[0].as_block().unwrap()[0].as_integer().unwrap(), (2, false)); + assert_eq( + body.as_block().unwrap()[0].as_block().unwrap()[0].as_integer().unwrap(), + (2, false), + ); let expr = quote { |x| { 1 } }.as_expr().unwrap(); let expr = expr.modify(times_two); @@ -673,14 +637,16 @@ mod tests { assert_eq(params.len(), 1); assert(params[0].1.is_none()); assert(return_type.is_none()); - assert_eq(body.as_block().unwrap()[0].as_block().unwrap()[0].as_integer().unwrap(), (2, false)); + assert_eq( + body.as_block().unwrap()[0].as_block().unwrap()[0].as_integer().unwrap(), + (2, false), + ); } } #[test] fn test_expr_as_let() { - comptime - { + comptime { let expr = quote { let x: Field = 1; }.as_expr().unwrap(); let (_pattern, typ, expr) = expr.as_let().unwrap(); assert(typ.unwrap().is_field()); @@ -690,8 +656,7 @@ mod tests { #[test] fn test_expr_modify_for_let() { - comptime - { + comptime { let expr = quote { let x : Field = 1; }.as_expr().unwrap(); let expr = expr.modify(times_two); let (_pattern, typ, expr) = expr.as_let().unwrap(); @@ -702,8 +667,7 @@ mod tests { #[test] fn test_expr_modify_for_let_without_type() { - comptime - { + comptime { let expr = quote { let x = 1; }.as_expr().unwrap(); let expr = expr.modify(times_two); let (_pattern, typ, expr) = expr.as_let().unwrap(); @@ -714,8 +678,7 @@ mod tests { #[test] fn test_expr_as_for_statement() { - comptime - { + comptime { let expr = quote { for x in 2 { 3 } }.as_expr().unwrap(); let (index, array, body) = expr.as_for().unwrap(); assert_eq(index, quote { x }); @@ -726,21 +689,22 @@ mod tests { #[test] fn test_expr_modify_for_statement() { - comptime - { + comptime { let expr = quote { for x in 2 { 3 } }.as_expr().unwrap(); let expr = expr.modify(times_two); let (index, array, body) = expr.as_for().unwrap(); assert_eq(index, quote { x }); assert_eq(array.as_integer().unwrap(), (4, false)); - assert_eq(body.as_block().unwrap()[0].as_block().unwrap()[0].as_integer().unwrap(), (6, false)); + assert_eq( + body.as_block().unwrap()[0].as_block().unwrap()[0].as_integer().unwrap(), + (6, false), + ); } } #[test] fn test_expr_as_for_range_statement() { - comptime - { + comptime { let expr = quote { for x in 2..3 { 4 } }.as_expr().unwrap(); let (index, from, to, body) = expr.as_for_range().unwrap(); assert_eq(index, quote { x }); @@ -752,22 +716,23 @@ mod tests { #[test] fn test_expr_modify_for_range_statement() { - comptime - { + comptime { let expr = quote { for x in 2..3 { 4 } }.as_expr().unwrap(); let expr = expr.modify(times_two); let (index, from, to, body) = expr.as_for_range().unwrap(); assert_eq(index, quote { x }); assert_eq(from.as_integer().unwrap(), (4, false)); assert_eq(to.as_integer().unwrap(), (6, false)); - assert_eq(body.as_block().unwrap()[0].as_block().unwrap()[0].as_integer().unwrap(), (8, false)); + assert_eq( + body.as_block().unwrap()[0].as_block().unwrap()[0].as_integer().unwrap(), + (8, false), + ); } } #[test] fn test_automatically_unwraps_parenthesized_expression() { - comptime - { + comptime { let expr = quote { ((if 1 { 2 })) }.as_expr().unwrap(); assert(expr.as_if().is_some()); } @@ -775,8 +740,7 @@ mod tests { #[test] fn test_resolve_to_function_definition() { - comptime - { + comptime { let expr = quote { times_two }.as_expr().unwrap(); let func = expr.resolve(Option::none()).as_function_definition().unwrap(); assert_eq(func.name(), quote { times_two }); @@ -797,13 +761,10 @@ mod tests { } comptime fn times_two(expr: Expr) -> Option { - expr.as_integer().and_then( - |integer: (Field, bool)| { - let (value, _) = integer; + expr.as_integer().and_then(|(value, _)| { let value = value * 2; quote { $value }.as_expr() - } - ) + }) } } diff --git a/noir/noir-repo/test_programs/noir_test_success/global_eval/Nargo.toml b/noir/noir-repo/test_programs/noir_test_success/global_eval/Nargo.toml new file mode 100644 index 00000000000..1dfe3a9660a --- /dev/null +++ b/noir/noir-repo/test_programs/noir_test_success/global_eval/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "global_eval" +type = "bin" +authors = [""] +compiler_version = ">=0.27.0" + +[dependencies] diff --git a/noir/noir-repo/test_programs/noir_test_success/global_eval/src/main.nr b/noir/noir-repo/test_programs/noir_test_success/global_eval/src/main.nr new file mode 100644 index 00000000000..6ec366c4cd6 --- /dev/null +++ b/noir/noir-repo/test_programs/noir_test_success/global_eval/src/main.nr @@ -0,0 +1,20 @@ +use std::uint128::U128; + +// These definitions require `to_be_bits` and `to_le_bits` to be supported at comptime. +global BITS_BE_13: [u1; 4] = (13 as Field).to_be_bits(); +global BITS_LE_13: [u1; 4] = (13 as Field).to_le_bits(); + +// Examples from #6691 which use the above behind the scenes. +global POW64_A: Field = 2.pow_32(64); +global POW64_B: Field = (U128::one() << 64).to_integer(); + +#[test] +fn test_be_and_le_bits() { + assert_eq(BITS_BE_13, [1,1,0,1]); + assert_eq(BITS_LE_13, [1,0,1,1]); +} + +#[test] +fn test_pow64() { + assert_eq(POW64_A, POW64_B); +} diff --git a/noir/noir-repo/test_programs/noir_test_success/mock_oracle/src/main.nr b/noir/noir-repo/test_programs/noir_test_success/mock_oracle/src/main.nr index 1b427043e91..aaf2c87ddb0 100644 --- a/noir/noir-repo/test_programs/noir_test_success/mock_oracle/src/main.nr +++ b/noir/noir-repo/test_programs/noir_test_success/mock_oracle/src/main.nr @@ -34,6 +34,7 @@ unconstrained fn struct_field(point: Point, array: [Field; 4]) -> Field { #[test(should_fail)] fn test_mock_no_returns() { + /// Safety: testing context unsafe { OracleMock::mock("void_field"); void_field(); // Some return value must be set @@ -42,6 +43,7 @@ fn test_mock_no_returns() { #[test] fn test_mock() { + /// Safety: testing context unsafe { OracleMock::mock("void_field").returns(10); assert_eq(void_field(), 10); @@ -50,6 +52,7 @@ fn test_mock() { #[test] fn test_multiple_mock() { + /// Safety: testing context unsafe { let first_mock = OracleMock::mock("void_field").returns(10); OracleMock::mock("void_field").returns(42); @@ -64,6 +67,7 @@ fn test_multiple_mock() { #[test] fn test_multiple_mock_times() { + /// Safety: testing context unsafe { OracleMock::mock("void_field").returns(10).times(2); OracleMock::mock("void_field").returns(42); @@ -76,6 +80,7 @@ fn test_multiple_mock_times() { #[test] fn test_mock_with_params() { + /// Safety: testing context unsafe { OracleMock::mock("field_field").with_params((5,)).returns(10); assert_eq(field_field(5), 10); @@ -84,6 +89,7 @@ fn test_mock_with_params() { #[test] fn test_multiple_mock_with_params() { + /// Safety: testing context unsafe { OracleMock::mock("field_field").with_params((5,)).returns(10); OracleMock::mock("field_field").with_params((7,)).returns(14); @@ -95,6 +101,7 @@ fn test_multiple_mock_with_params() { #[test] fn test_mock_last_params() { + /// Safety: testing context unsafe { let mock = OracleMock::mock("field_field").returns(10); assert_eq(field_field(5), 10); @@ -105,6 +112,7 @@ fn test_mock_last_params() { #[test] fn test_mock_last_params_many_calls() { + /// Safety: testing context unsafe { let mock = OracleMock::mock("field_field").returns(10); assert_eq(field_field(5), 10); @@ -117,11 +125,11 @@ fn test_mock_last_params_many_calls() { #[test] fn test_mock_struct_field() { // Combination of simpler test cases - let array = [1, 2, 3, 4]; let another_array = [4, 3, 2, 1]; let point = Point { x: 14, y: 27 }; + /// Safety: testing context unsafe { OracleMock::mock("struct_field").returns(42).times(2); let timeless_mock = OracleMock::mock("struct_field").returns(0); diff --git a/noir/noir-repo/test_programs/noir_test_success/out_of_bounds_alignment/src/main.nr b/noir/noir-repo/test_programs/noir_test_success/out_of_bounds_alignment/src/main.nr index fb90e3d96c5..05bc2a5ce13 100644 --- a/noir/noir-repo/test_programs/noir_test_success/out_of_bounds_alignment/src/main.nr +++ b/noir/noir-repo/test_programs/noir_test_success/out_of_bounds_alignment/src/main.nr @@ -2,7 +2,10 @@ fn out_of_bounds(arr_1: [Field; 50]) -> Field { arr_1[50 + 1] } -unconstrained fn out_of_bounds_unconstrained_wrapper(arr_1: [Field; 50], arr_2: [Field; 50]) -> Field { +unconstrained fn out_of_bounds_unconstrained_wrapper( + arr_1: [Field; 50], + arr_2: [Field; 50], +) -> Field { out_of_bounds(arr_1) } @@ -13,6 +16,7 @@ fn test_acir() { #[test(should_fail)] fn test_brillig() { + /// Safety: testing context unsafe { assert_eq(out_of_bounds_unconstrained_wrapper([0; 50], [0; 50]), 0); } diff --git a/noir/noir-repo/test_programs/parse_memory.sh b/noir/noir-repo/test_programs/parse_memory.sh new file mode 100755 index 00000000000..8e125b13b23 --- /dev/null +++ b/noir/noir-repo/test_programs/parse_memory.sh @@ -0,0 +1,21 @@ +#!/usr/bin/env bash +set -e + +# This script accepts a string representing the amount of memory allocated as outputted by `heaptrack` +# and standardizes it to be in terms of megabytes as `heaptrack` will report different units depending on the duration. + +DIGITS='([0-9]+(\.[0-9]+)?)' +KILOBYTES_REGEX=^${DIGITS}K$ +MEGABYTES_REGEX=^${DIGITS}M$ +GIGABYTES_REGEX=^${DIGITS}G$ + +if [[ $1 =~ $KILOBYTES_REGEX ]]; then + echo ${BASH_REMATCH[1]} 1000 | awk '{printf "%.3f\n", $1/$2}' +elif [[ $1 =~ $MEGABYTES_REGEX ]]; then + echo ${BASH_REMATCH[1]} | awk '{printf "%.3f\n", $1}' +elif [[ $1 =~ $GIGABYTES_REGEX ]]; then + echo ${BASH_REMATCH[1]} 1000 | awk '{printf "%.3f\n", $1*$2}' +else + echo "Could not parse memory: unrecognized format" 1>&2 + exit 1 +fi diff --git a/noir/noir-repo/test_programs/parse_time.sh b/noir/noir-repo/test_programs/parse_time.sh new file mode 100755 index 00000000000..2ff5c259cd2 --- /dev/null +++ b/noir/noir-repo/test_programs/parse_time.sh @@ -0,0 +1,21 @@ +#!/usr/bin/env bash +set -e + +# This script accepts a string representing the time spent within a span as outputted by `tracing` +# and standardizes it to be in terms of seconds as `tracing` will report different units depending on the duration. + +DIGITS='([0-9]+(\.[0-9]+)?)' +MICROSECONDS_REGEX=^${DIGITS}µs$ +MILLISECONDS_REGEX=^${DIGITS}ms$ +SECONDS_REGEX=^${DIGITS}s$ + +if [[ $1 =~ $MICROSECONDS_REGEX ]]; then + echo ${BASH_REMATCH[1]} 1000000 | awk '{printf "%.3f\n", $1/$2}' +elif [[ $1 =~ $MILLISECONDS_REGEX ]]; then + echo ${BASH_REMATCH[1]} 1000 | awk '{printf "%.3f\n", $1/$2}' +elif [[ $1 =~ $SECONDS_REGEX ]]; then + echo ${BASH_REMATCH[1]} | awk '{printf "%.3f\n", $1}' +else + echo "Could not parse time: unrecognized format" 1>&2 + exit 1 +fi \ No newline at end of file diff --git a/noir/noir-repo/tooling/acvm_cli/src/cli/execute_cmd.rs b/noir/noir-repo/tooling/acvm_cli/src/cli/execute_cmd.rs index d4473eb3eef..e5d48073ca8 100644 --- a/noir/noir-repo/tooling/acvm_cli/src/cli/execute_cmd.rs +++ b/noir/noir-repo/tooling/acvm_cli/src/cli/execute_cmd.rs @@ -9,7 +9,7 @@ use nargo::PrintOutput; use crate::cli::fs::inputs::{read_bytecode_from_file, read_inputs_from_file}; use crate::errors::CliError; -use nargo::{foreign_calls::DefaultForeignCallExecutor, ops::execute_program}; +use nargo::{foreign_calls::DefaultForeignCallBuilder, ops::execute_program}; use super::fs::witness::{create_output_witness_string, save_witness_to_dir}; @@ -35,12 +35,19 @@ pub(crate) struct ExecuteCommand { /// Set to print output witness to stdout #[clap(long, short, action)] print: bool, + + /// Use pedantic ACVM solving, i.e. double-check some black-box function + /// assumptions when solving. + /// This is disabled by default. + #[clap(long, default_value = "false")] + pedantic_solving: bool, } fn run_command(args: ExecuteCommand) -> Result { let bytecode = read_bytecode_from_file(&args.working_directory, &args.bytecode)?; let circuit_inputs = read_inputs_from_file(&args.working_directory, &args.input_witness)?; - let output_witness = execute_program_from_witness(circuit_inputs, &bytecode)?; + let output_witness = + execute_program_from_witness(circuit_inputs, &bytecode, args.pedantic_solving)?; assert_eq!(output_witness.length(), 1, "ACVM CLI only supports a witness stack of size 1"); let output_witness_string = create_output_witness_string( &output_witness.peek().expect("Should have a witness stack item").witness, @@ -67,14 +74,20 @@ pub(crate) fn run(args: ExecuteCommand) -> Result { pub(crate) fn execute_program_from_witness( inputs_map: WitnessMap, bytecode: &[u8], + pedantic_solving: bool, ) -> Result, CliError> { let program: Program = Program::deserialize_program(bytecode) .map_err(|_| CliError::CircuitDeserializationError())?; execute_program( &program, inputs_map, - &Bn254BlackBoxSolver, - &mut DefaultForeignCallExecutor::new(PrintOutput::Stdout, None, None, None), + &Bn254BlackBoxSolver(pedantic_solving), + &mut DefaultForeignCallBuilder { + output: PrintOutput::Stdout, + enable_mocks: false, + ..Default::default() + } + .build(), ) .map_err(CliError::CircuitExecutionError) } diff --git a/noir/noir-repo/tooling/acvm_cli/src/cli/mod.rs b/noir/noir-repo/tooling/acvm_cli/src/cli/mod.rs index a610b08ab77..f31e123d0cd 100644 --- a/noir/noir-repo/tooling/acvm_cli/src/cli/mod.rs +++ b/noir/noir-repo/tooling/acvm_cli/src/cli/mod.rs @@ -22,7 +22,6 @@ enum ACVMCommand { Execute(execute_cmd::ExecuteCommand), } -#[cfg(not(feature = "codegen-docs"))] pub(crate) fn start_cli() -> eyre::Result<()> { let ACVMCli { command } = ACVMCli::parse(); @@ -32,10 +31,3 @@ pub(crate) fn start_cli() -> eyre::Result<()> { Ok(()) } - -#[cfg(feature = "codegen-docs")] -pub(crate) fn start_cli() -> eyre::Result<()> { - let markdown: String = clap_markdown::help_markdown::(); - println!("{markdown}"); - Ok(()) -} diff --git a/noir/noir-repo/tooling/debugger/build.rs b/noir/noir-repo/tooling/debugger/build.rs index ebdf2036894..63924de8336 100644 --- a/noir/noir-repo/tooling/debugger/build.rs +++ b/noir/noir-repo/tooling/debugger/build.rs @@ -2,7 +2,6 @@ use std::collections::HashSet; use std::fs::File; use std::io::Write; use std::path::{Path, PathBuf}; -use std::{env, fs}; const GIT_COMMIT: &&str = &"GIT_COMMIT"; @@ -14,7 +13,7 @@ fn main() { build_data::no_debug_rebuilds(); } - let out_dir = env::var("OUT_DIR").unwrap(); + let out_dir = std::env::var("OUT_DIR").unwrap(); let destination = Path::new(&out_dir).join("debug.rs"); let mut test_file = File::create(destination).unwrap(); @@ -39,8 +38,8 @@ fn generate_debugger_tests(test_file: &mut File, test_data_dir: &Path) { let test_data_dir = test_data_dir.join(test_sub_dir); let test_case_dirs = - fs::read_dir(test_data_dir).unwrap().flatten().filter(|c| c.path().is_dir()); - let ignored_tests_contents = fs::read_to_string("ignored-tests.txt").unwrap(); + std::fs::read_dir(test_data_dir).unwrap().flatten().filter(|c| c.path().is_dir()); + let ignored_tests_contents = std::fs::read_to_string("ignored-tests.txt").unwrap(); let ignored_tests = ignored_tests_contents.lines().collect::>(); for test_dir in test_case_dirs { diff --git a/noir/noir-repo/tooling/debugger/src/context.rs b/noir/noir-repo/tooling/debugger/src/context.rs index 77c186fe707..9741749df64 100644 --- a/noir/noir-repo/tooling/debugger/src/context.rs +++ b/noir/noir-repo/tooling/debugger/src/context.rs @@ -233,7 +233,7 @@ pub struct ExecutionFrame<'a, B: BlackBoxFunctionSolver> { } pub(super) struct DebugContext<'a, B: BlackBoxFunctionSolver> { - acvm: ACVM<'a, FieldElement, B>, + pub(crate) acvm: ACVM<'a, FieldElement, B>, current_circuit_id: u32, brillig_solver: Option>, @@ -971,6 +971,7 @@ mod tests { #[test] fn test_resolve_foreign_calls_stepping_into_brillig() { + let solver = StubbedBlackBoxSolver::default(); let fe_1 = FieldElement::one(); let w_x = Witness(1); @@ -1015,10 +1016,10 @@ mod tests { outputs: vec![], predicate: None, }]; - let brillig_funcs = &vec![brillig_bytecode]; + let brillig_funcs = &[brillig_bytecode]; let current_witness_index = 2; let circuit = Circuit { current_witness_index, opcodes, ..Circuit::default() }; - let circuits = &vec![circuit]; + let circuits = &[circuit]; let debug_symbols = vec![]; let file_map = BTreeMap::new(); @@ -1031,7 +1032,7 @@ mod tests { debug_artifact, )); let mut context = DebugContext::new( - &StubbedBlackBoxSolver, + &solver, circuits, debug_artifact, initial_witness, @@ -1116,6 +1117,7 @@ mod tests { #[test] fn test_break_brillig_block_while_stepping_acir_opcodes() { + let solver = StubbedBlackBoxSolver::default(); let fe_0 = FieldElement::zero(); let fe_1 = FieldElement::one(); let w_x = Witness(1); @@ -1185,7 +1187,7 @@ mod tests { ]; let current_witness_index = 3; let circuit = Circuit { current_witness_index, opcodes, ..Circuit::default() }; - let circuits = &vec![circuit]; + let circuits = &[circuit]; let debug_symbols = vec![]; let file_map = BTreeMap::new(); @@ -1197,9 +1199,9 @@ mod tests { PrintOutput::Stdout, debug_artifact, )); - let brillig_funcs = &vec![brillig_bytecode]; + let brillig_funcs = &[brillig_bytecode]; let mut context = DebugContext::new( - &StubbedBlackBoxSolver, + &solver, circuits, debug_artifact, initial_witness, @@ -1240,6 +1242,7 @@ mod tests { #[test] fn test_address_debug_location_mapping() { + let solver = StubbedBlackBoxSolver::default(); let brillig_one = BrilligBytecode { bytecode: vec![BrilligOpcode::Return, BrilligOpcode::Return] }; let brillig_two = BrilligBytecode { @@ -1283,10 +1286,10 @@ mod tests { }; let circuits = vec![circuit_one, circuit_two]; let debug_artifact = DebugArtifact { debug_symbols: vec![], file_map: BTreeMap::new() }; - let brillig_funcs = &vec![brillig_one, brillig_two]; + let brillig_funcs = &[brillig_one, brillig_two]; let context = DebugContext::new( - &StubbedBlackBoxSolver, + &solver, &circuits, &debug_artifact, WitnessMap::new(), diff --git a/noir/noir-repo/tooling/debugger/src/foreign_calls.rs b/noir/noir-repo/tooling/debugger/src/foreign_calls.rs index 899ba892d8f..b92e22844ea 100644 --- a/noir/noir-repo/tooling/debugger/src/foreign_calls.rs +++ b/noir/noir-repo/tooling/debugger/src/foreign_calls.rs @@ -4,12 +4,13 @@ use acvm::{ AcirField, FieldElement, }; use nargo::{ - foreign_calls::{DefaultForeignCallExecutor, ForeignCallExecutor}, + foreign_calls::{ + layers::Layer, DefaultForeignCallBuilder, ForeignCallError, ForeignCallExecutor, + }, PrintOutput, }; use noirc_artifacts::debug::{DebugArtifact, DebugVars, StackFrame}; use noirc_errors::debug_info::{DebugFnId, DebugVarId}; -use noirc_printable_type::ForeignCallError; pub(crate) enum DebugForeignCall { VarAssign, @@ -44,23 +45,31 @@ pub trait DebugForeignCallExecutor: ForeignCallExecutor { fn current_stack_frame(&self) -> Option>; } -pub struct DefaultDebugForeignCallExecutor<'a> { - executor: DefaultForeignCallExecutor<'a, FieldElement>, +#[derive(Default)] +pub struct DefaultDebugForeignCallExecutor { pub debug_vars: DebugVars, } -impl<'a> DefaultDebugForeignCallExecutor<'a> { - pub fn new(output: PrintOutput<'a>) -> Self { - Self { - executor: DefaultForeignCallExecutor::new(output, None, None, None), - debug_vars: DebugVars::default(), - } +impl DefaultDebugForeignCallExecutor { + fn make( + output: PrintOutput<'_>, + ex: DefaultDebugForeignCallExecutor, + ) -> impl DebugForeignCallExecutor + '_ { + DefaultForeignCallBuilder::default().with_output(output).build().add_layer(ex) + } + + #[allow(clippy::new_ret_no_self, dead_code)] + pub fn new(output: PrintOutput<'_>) -> impl DebugForeignCallExecutor + '_ { + Self::make(output, Self::default()) } - pub fn from_artifact(output: PrintOutput<'a>, artifact: &DebugArtifact) -> Self { - let mut ex = Self::new(output); + pub fn from_artifact<'a>( + output: PrintOutput<'a>, + artifact: &DebugArtifact, + ) -> impl DebugForeignCallExecutor + 'a { + let mut ex = Self::default(); ex.load_artifact(artifact); - ex + Self::make(output, ex) } pub fn load_artifact(&mut self, artifact: &DebugArtifact) { @@ -73,7 +82,7 @@ impl<'a> DefaultDebugForeignCallExecutor<'a> { } } -impl DebugForeignCallExecutor for DefaultDebugForeignCallExecutor<'_> { +impl DebugForeignCallExecutor for DefaultDebugForeignCallExecutor { fn get_variables(&self) -> Vec> { self.debug_vars.get_variables() } @@ -91,7 +100,7 @@ fn debug_fn_id(value: &FieldElement) -> DebugFnId { DebugFnId(value.to_u128() as u32) } -impl ForeignCallExecutor for DefaultDebugForeignCallExecutor<'_> { +impl ForeignCallExecutor for DefaultDebugForeignCallExecutor { fn execute( &mut self, foreign_call: &ForeignCallWaitInfo, @@ -166,7 +175,21 @@ impl ForeignCallExecutor for DefaultDebugForeignCallExecutor<'_> { self.debug_vars.pop_fn(); Ok(ForeignCallResult::default()) } - None => self.executor.execute(foreign_call), + None => Err(ForeignCallError::NoHandler(foreign_call_name.to_string())), } } } + +impl DebugForeignCallExecutor for Layer +where + H: DebugForeignCallExecutor, + I: ForeignCallExecutor, +{ + fn get_variables(&self) -> Vec> { + self.handler().get_variables() + } + + fn current_stack_frame(&self) -> Option> { + self.handler().current_stack_frame() + } +} diff --git a/noir/noir-repo/tooling/fuzzer/src/dictionary/mod.rs b/noir/noir-repo/tooling/fuzzer/src/dictionary/mod.rs index 172edfa54c2..7fe1c4cd602 100644 --- a/noir/noir-repo/tooling/fuzzer/src/dictionary/mod.rs +++ b/noir/noir-repo/tooling/fuzzer/src/dictionary/mod.rs @@ -33,9 +33,11 @@ pub(super) fn build_dictionary_from_program(program: &Program) constants } +/// Collect `Field` values used in the opcodes of an ACIR circuit. fn build_dictionary_from_circuit(circuit: &Circuit) -> HashSet { let mut constants: HashSet = HashSet::new(); + /// Pull out all the fields from an expression. fn insert_expr(dictionary: &mut HashSet, expr: &Expression) { let quad_coefficients = expr.mul_terms.iter().map(|(k, _, _)| *k); let linear_coefficients = expr.linear_combinations.iter().map(|(k, _)| *k); @@ -104,6 +106,7 @@ fn build_dictionary_from_circuit(circuit: &Circuit) -> HashSet< constants } +/// Collect `Field` values used in the opcodes of a Brillig function. fn build_dictionary_from_unconstrained_function( function: &BrilligBytecode, ) -> HashSet { diff --git a/noir/noir-repo/tooling/fuzzer/src/lib.rs b/noir/noir-repo/tooling/fuzzer/src/lib.rs index 28a43279c95..324be323fc2 100644 --- a/noir/noir-repo/tooling/fuzzer/src/lib.rs +++ b/noir/noir-repo/tooling/fuzzer/src/lib.rs @@ -22,7 +22,7 @@ use types::{CaseOutcome, CounterExampleOutcome, FuzzOutcome, FuzzTestResult}; use noirc_artifacts::program::ProgramArtifact; -/// An executor for Noir programs which which provides fuzzing support using [`proptest`]. +/// An executor for Noir programs which provides fuzzing support using [`proptest`]. /// /// After instantiation, calling `fuzz` will proceed to hammer the program with /// inputs, until it finds a counterexample. The provided [`TestRunner`] contains all the @@ -38,14 +38,14 @@ pub struct FuzzedExecutor { runner: TestRunner, } -impl< - E: Fn( - &Program, - WitnessMap, - ) -> Result, String>, - > FuzzedExecutor +impl FuzzedExecutor +where + E: Fn( + &Program, + WitnessMap, + ) -> Result, String>, { - /// Instantiates a fuzzed executor given a testrunner + /// Instantiates a fuzzed executor given a [TestRunner]. pub fn new(program: ProgramArtifact, executor: E, runner: TestRunner) -> Self { Self { program, executor, runner } } @@ -53,7 +53,7 @@ impl< /// Fuzzes the provided program. pub fn fuzz(&self) -> FuzzTestResult { let dictionary = build_dictionary_from_program(&self.program.bytecode); - let strategy = strategies::arb_input_map(&self.program.abi, dictionary); + let strategy = strategies::arb_input_map(&self.program.abi, &dictionary); let run_result: Result<(), TestError> = self.runner.clone().run(&strategy, |input_map| { diff --git a/noir/noir-repo/tooling/fuzzer/src/strategies/int.rs b/noir/noir-repo/tooling/fuzzer/src/strategies/int.rs index d11cafcfae5..22dded7c7b7 100644 --- a/noir/noir-repo/tooling/fuzzer/src/strategies/int.rs +++ b/noir/noir-repo/tooling/fuzzer/src/strategies/int.rs @@ -4,6 +4,8 @@ use proptest::{ }; use rand::Rng; +type BinarySearch = proptest::num::i128::BinarySearch; + /// Strategy for signed ints (up to i128). /// The strategy combines 2 different strategies, each assigned a specific weight: /// 1. Generate purely random value in a range. This will first choose bit size uniformly (up `bits` @@ -27,6 +29,7 @@ impl IntStrategy { Self { bits, edge_weight: 10usize, random_weight: 50usize } } + /// Generate random values near MIN or the MAX value. fn generate_edge_tree(&self, runner: &mut TestRunner) -> NewTree { let rng = runner.rng(); @@ -40,16 +43,16 @@ impl IntStrategy { 3 => self.type_max() - offset, _ => unreachable!(), }; - Ok(proptest::num::i128::BinarySearch::new(start)) + Ok(BinarySearch::new(start)) } fn generate_random_tree(&self, runner: &mut TestRunner) -> NewTree { let rng = runner.rng(); - let start: i128 = rng.gen_range(self.type_min()..=self.type_max()); - Ok(proptest::num::i128::BinarySearch::new(start)) + Ok(BinarySearch::new(start)) } + /// Maximum allowed positive number. fn type_max(&self) -> i128 { if self.bits < 128 { (1i128 << (self.bits - 1)) - 1 @@ -58,6 +61,7 @@ impl IntStrategy { } } + /// Minimum allowed negative number. fn type_min(&self) -> i128 { if self.bits < 128 { -(1i128 << (self.bits - 1)) @@ -68,7 +72,7 @@ impl IntStrategy { } impl Strategy for IntStrategy { - type Tree = proptest::num::i128::BinarySearch; + type Tree = BinarySearch; type Value = i128; fn new_tree(&self, runner: &mut TestRunner) -> NewTree { diff --git a/noir/noir-repo/tooling/fuzzer/src/strategies/mod.rs b/noir/noir-repo/tooling/fuzzer/src/strategies/mod.rs index 46187a28d5b..99c7ca56f2e 100644 --- a/noir/noir-repo/tooling/fuzzer/src/strategies/mod.rs +++ b/noir/noir-repo/tooling/fuzzer/src/strategies/mod.rs @@ -11,22 +11,28 @@ use uint::UintStrategy; mod int; mod uint; +/// Create a strategy for generating random values for an [AbiType]. +/// +/// Uses the `dictionary` for unsigned integer types. pub(super) fn arb_value_from_abi_type( abi_type: &AbiType, - dictionary: HashSet, + dictionary: &HashSet, ) -> SBoxedStrategy { match abi_type { AbiType::Field => vec(any::(), 32) .prop_map(|bytes| InputValue::Field(FieldElement::from_be_bytes_reduce(&bytes))) .sboxed(), AbiType::Integer { width, sign } if sign == &Sign::Unsigned => { - UintStrategy::new(*width as usize, dictionary) + // We've restricted the type system to only allow u64s as the maximum integer type. + let width = (*width).min(64); + UintStrategy::new(width as usize, dictionary) .prop_map(|uint| InputValue::Field(uint.into())) .sboxed() } AbiType::Integer { width, .. } => { - let shift = 2i128.pow(*width); - IntStrategy::new(*width as usize) + let width = (*width).min(64); + let shift = 2i128.pow(width); + IntStrategy::new(width as usize) .prop_map(move |mut int| { if int < 0 { int += shift @@ -38,7 +44,6 @@ pub(super) fn arb_value_from_abi_type( AbiType::Boolean => { any::().prop_map(|val| InputValue::Field(FieldElement::from(val))).sboxed() } - AbiType::String { length } => { // Strings only allow ASCII characters as each character must be able to be represented by a single byte. let string_regex = format!("[[:ascii:]]{{{length}}}"); @@ -53,12 +58,11 @@ pub(super) fn arb_value_from_abi_type( elements.prop_map(InputValue::Vec).sboxed() } - AbiType::Struct { fields, .. } => { let fields: Vec> = fields .iter() .map(|(name, typ)| { - (Just(name.clone()), arb_value_from_abi_type(typ, dictionary.clone())).sboxed() + (Just(name.clone()), arb_value_from_abi_type(typ, dictionary)).sboxed() }) .collect(); @@ -69,25 +73,25 @@ pub(super) fn arb_value_from_abi_type( }) .sboxed() } - AbiType::Tuple { fields } => { let fields: Vec<_> = - fields.iter().map(|typ| arb_value_from_abi_type(typ, dictionary.clone())).collect(); + fields.iter().map(|typ| arb_value_from_abi_type(typ, dictionary)).collect(); fields.prop_map(InputValue::Vec).sboxed() } } } +/// Given the [Abi] description of a [ProgramArtifact], generate random [InputValue]s for each circuit parameter. +/// +/// Use the `dictionary` to draw values from for numeric types. pub(super) fn arb_input_map( abi: &Abi, - dictionary: HashSet, + dictionary: &HashSet, ) -> BoxedStrategy { let values: Vec<_> = abi .parameters .iter() - .map(|param| { - (Just(param.name.clone()), arb_value_from_abi_type(¶m.typ, dictionary.clone())) - }) + .map(|param| (Just(param.name.clone()), arb_value_from_abi_type(¶m.typ, dictionary))) .collect(); values diff --git a/noir/noir-repo/tooling/fuzzer/src/strategies/uint.rs b/noir/noir-repo/tooling/fuzzer/src/strategies/uint.rs index 94610dbc829..402e6559752 100644 --- a/noir/noir-repo/tooling/fuzzer/src/strategies/uint.rs +++ b/noir/noir-repo/tooling/fuzzer/src/strategies/uint.rs @@ -7,6 +7,8 @@ use proptest::{ }; use rand::Rng; +type BinarySearch = proptest::num::u128::BinarySearch; + /// Value tree for unsigned ints (up to u128). /// The strategy combines 2 different strategies, each assigned a specific weight: /// 1. Generate purely random value in a range. This will first choose bit size uniformly (up `bits` @@ -14,7 +16,7 @@ use rand::Rng; /// 2. Generate a random value around the edges (+/- 3 around 0 and max possible value) #[derive(Debug)] pub struct UintStrategy { - /// Bit size of uint (e.g. 128) + /// Bit size of uint (e.g. 64) bits: usize, /// A set of fixtures to be generated fixtures: Vec, @@ -31,25 +33,29 @@ impl UintStrategy { /// # Arguments /// * `bits` - Size of uint in bits /// * `fixtures` - Set of `FieldElements` representing values which the fuzzer weight towards testing. - pub fn new(bits: usize, fixtures: HashSet) -> Self { + pub fn new(bits: usize, fixtures: &HashSet) -> Self { Self { bits, - fixtures: fixtures.into_iter().collect(), + // We can only consider the fixtures which fit into the bit width. + fixtures: fixtures.iter().filter(|f| f.num_bits() <= bits as u32).copied().collect(), edge_weight: 10usize, fixtures_weight: 40usize, random_weight: 50usize, } } + /// Generate random numbers starting from near 0 or the maximum of the range. fn generate_edge_tree(&self, runner: &mut TestRunner) -> NewTree { let rng = runner.rng(); // Choose if we want values around 0 or max let is_min = rng.gen_bool(0.5); let offset = rng.gen_range(0..4); let start = if is_min { offset } else { self.type_max().saturating_sub(offset) }; - Ok(proptest::num::u128::BinarySearch::new(start)) + Ok(BinarySearch::new(start)) } + /// Pick a random `FieldElement` from the `fixtures` as a starting point for + /// generating random numbers. fn generate_fixtures_tree(&self, runner: &mut TestRunner) -> NewTree { // generate random cases if there's no fixtures if self.fixtures.is_empty() { @@ -58,21 +64,19 @@ impl UintStrategy { // Generate value tree from fixture. let fixture = &self.fixtures[runner.rng().gen_range(0..self.fixtures.len())]; - if fixture.num_bits() <= self.bits as u32 { - return Ok(proptest::num::u128::BinarySearch::new(fixture.to_u128())); - } - // If fixture is not a valid type, generate random value. - self.generate_random_tree(runner) + Ok(BinarySearch::new(fixture.to_u128())) } + /// Generate random values between 0 and the MAX with the given bit width. fn generate_random_tree(&self, runner: &mut TestRunner) -> NewTree { let rng = runner.rng(); let start = rng.gen_range(0..=self.type_max()); - Ok(proptest::num::u128::BinarySearch::new(start)) + Ok(BinarySearch::new(start)) } + /// Maximum integer that fits in the given bit width. fn type_max(&self) -> u128 { if self.bits < 128 { (1 << self.bits) - 1 @@ -83,8 +87,10 @@ impl UintStrategy { } impl Strategy for UintStrategy { - type Tree = proptest::num::u128::BinarySearch; + type Tree = BinarySearch; type Value = u128; + + /// Pick randomly from the 3 available strategies for generating unsigned integers. fn new_tree(&self, runner: &mut TestRunner) -> NewTree { let total_weight = self.random_weight + self.fixtures_weight + self.edge_weight; let bias = runner.rng().gen_range(0..total_weight); diff --git a/noir/noir-repo/tooling/lsp/Cargo.toml b/noir/noir-repo/tooling/lsp/Cargo.toml index 04c8edf7ea9..65b59552b32 100644 --- a/noir/noir-repo/tooling/lsp/Cargo.toml +++ b/noir/noir-repo/tooling/lsp/Cargo.toml @@ -3,7 +3,7 @@ name = "noir_lsp" description = "Language server for Noir" version.workspace = true authors.workspace = true -edition.workspace = true# +edition.workspace = true rust-version.workspace = true license.workspace = true @@ -16,7 +16,7 @@ workspace = true acvm.workspace = true codespan-lsp.workspace = true lsp-types.workspace = true -nargo.workspace = true +nargo = { workspace = true, features = ["rpc"] } nargo_fmt.workspace = true nargo_toml.workspace = true noirc_driver.workspace = true diff --git a/noir/noir-repo/tooling/lsp/src/lib.rs b/noir/noir-repo/tooling/lsp/src/lib.rs index a152cc17c7b..97a1c7ad27b 100644 --- a/noir/noir-repo/tooling/lsp/src/lib.rs +++ b/noir/noir-repo/tooling/lsp/src/lib.rs @@ -452,7 +452,7 @@ fn prepare_package_from_source_string() { "#; let client = ClientSocket::new_closed(); - let mut state = LspState::new(&client, acvm::blackbox_solver::StubbedBlackBoxSolver); + let mut state = LspState::new(&client, acvm::blackbox_solver::StubbedBlackBoxSolver::default()); let (mut context, crate_id) = prepare_source(source.to_string(), &mut state); let _check_result = noirc_driver::check_crate(&mut context, crate_id, &Default::default()); diff --git a/noir/noir-repo/tooling/lsp/src/modules.rs b/noir/noir-repo/tooling/lsp/src/modules.rs index cadf71b2eec..b023f3886c3 100644 --- a/noir/noir-repo/tooling/lsp/src/modules.rs +++ b/noir/noir-repo/tooling/lsp/src/modules.rs @@ -41,9 +41,7 @@ pub(crate) fn relative_module_full_path( interner, ); } else { - let Some(parent_module) = get_parent_module(interner, module_def_id) else { - return None; - }; + let parent_module = get_parent_module(interner, module_def_id)?; full_path = relative_module_id_path( parent_module, diff --git a/noir/noir-repo/tooling/lsp/src/requests/code_action.rs b/noir/noir-repo/tooling/lsp/src/requests/code_action.rs index 38cc6bddf64..24ed327393d 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/code_action.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/code_action.rs @@ -33,6 +33,7 @@ use super::{process_request, to_lsp_location}; mod fill_struct_fields; mod implement_missing_members; mod import_or_qualify; +mod import_trait; mod remove_bang_from_call; mod remove_unused_import; mod tests; @@ -285,6 +286,8 @@ impl<'a> Visitor for CodeActionFinder<'a> { self.remove_bang_from_call(method_call.method_name.span()); } + self.import_trait_in_method_call(method_call); + true } } diff --git a/noir/noir-repo/tooling/lsp/src/requests/code_action/import_trait.rs b/noir/noir-repo/tooling/lsp/src/requests/code_action/import_trait.rs new file mode 100644 index 00000000000..dd500c334ab --- /dev/null +++ b/noir/noir-repo/tooling/lsp/src/requests/code_action/import_trait.rs @@ -0,0 +1,411 @@ +use std::collections::HashSet; + +use noirc_errors::Location; +use noirc_frontend::{ + ast::MethodCallExpression, + hir::def_map::ModuleDefId, + node_interner::{ReferenceId, TraitId}, +}; + +use crate::{ + modules::{relative_module_full_path, relative_module_id_path}, + requests::TraitReexport, + use_segment_positions::{ + use_completion_item_additional_text_edits, UseCompletionItemAdditionTextEditsRequest, + }, + visibility::module_def_id_is_visible, +}; + +use super::CodeActionFinder; + +impl<'a> CodeActionFinder<'a> { + pub(super) fn import_trait_in_method_call(&mut self, method_call: &MethodCallExpression) { + // First see if the method name already points to a function. + let name_location = Location::new(method_call.method_name.span(), self.file); + if let Some(ReferenceId::Function(func_id)) = self.interner.find_referenced(name_location) { + // If yes, it could be that the compiler is issuing a warning because there's + // only one possible trait that the method could be coming from, but it's not imported + let func_meta = self.interner.function_meta(&func_id); + let Some(trait_impl_id) = func_meta.trait_impl else { + return; + }; + + let trait_impl = self.interner.get_trait_implementation(trait_impl_id); + let trait_id = trait_impl.borrow().trait_id; + self.import_trait(trait_id); + return; + } + + // Find out the type of the object + let object_location = Location::new(method_call.object.span, self.file); + let Some(typ) = self.interner.type_at_location(object_location) else { + return; + }; + + let trait_methods = + self.interner.lookup_trait_methods(&typ, &method_call.method_name.0.contents, true); + let trait_ids: HashSet<_> = trait_methods.iter().map(|(_, trait_id)| *trait_id).collect(); + + for trait_id in trait_ids { + self.import_trait(trait_id); + } + } + + fn import_trait(&mut self, trait_id: TraitId) { + // First check if the trait is visible + let trait_ = self.interner.get_trait(trait_id); + let visibility = trait_.visibility; + let module_def_id = ModuleDefId::TraitId(trait_id); + let mut trait_reexport = None; + + if !module_def_id_is_visible( + module_def_id, + self.module_id, + visibility, + None, + self.interner, + self.def_maps, + ) { + // If it's not, try to find a visible reexport of the trait + // that is visible from the current module + let Some((visible_module_id, name, _)) = + self.interner.get_trait_reexports(trait_id).iter().find( + |(module_id, _, visibility)| { + module_def_id_is_visible( + module_def_id, + self.module_id, + *visibility, + Some(*module_id), + self.interner, + self.def_maps, + ) + }, + ) + else { + return; + }; + trait_reexport = Some(TraitReexport { module_id: visible_module_id, name }); + } + + let trait_name = if let Some(trait_reexport) = &trait_reexport { + trait_reexport.name + } else { + &trait_.name + }; + + // Check if the trait is currently imported. If yes, no need to suggest anything + let module_data = + &self.def_maps[&self.module_id.krate].modules()[self.module_id.local_id.0]; + if !module_data.scope().find_name(trait_name).is_none() { + return; + } + + let module_def_id = ModuleDefId::TraitId(trait_id); + let current_module_parent_id = self.module_id.parent(self.def_maps); + let module_full_path = if let Some(trait_reexport) = &trait_reexport { + relative_module_id_path( + *trait_reexport.module_id, + &self.module_id, + current_module_parent_id, + self.interner, + ) + } else { + let Some(path) = relative_module_full_path( + module_def_id, + self.module_id, + current_module_parent_id, + self.interner, + ) else { + return; + }; + path + }; + + let full_path = format!("{}::{}", module_full_path, trait_name); + + let title = format!("Import {}", full_path); + + let text_edits = use_completion_item_additional_text_edits( + UseCompletionItemAdditionTextEditsRequest { + full_path: &full_path, + files: self.files, + file: self.file, + lines: &self.lines, + nesting: self.nesting, + auto_import_line: self.auto_import_line, + }, + &self.use_segment_positions, + ); + + let code_action = self.new_quick_fix_multiple_edits(title, text_edits); + self.code_actions.push(code_action); + } +} + +#[cfg(test)] +mod tests { + use tokio::test; + + use crate::requests::code_action::tests::assert_code_action; + + #[test] + async fn test_import_trait_in_method_call_when_one_option_but_not_in_scope() { + let title = "Import moo::Foo"; + + let src = r#"mod moo { + pub trait Foo { + fn foobar(self); + } + + impl Foo for Field { + fn foobar(self) {} + } +} + +fn main() { + let x: Field = 1; + x.foo>||||| CodeActionResponse { ) .await .expect("Could not execute on_code_action_request") - .unwrap() + .expect("Expected to get a CodeActionResponse, got None") } pub(crate) async fn assert_code_action(title: &str, src: &str, expected: &str) { diff --git a/noir/noir-repo/tooling/lsp/src/requests/completion.rs b/noir/noir-repo/tooling/lsp/src/requests/completion.rs index 3762ba9cf8d..a845fd4496f 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/completion.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/completion.rs @@ -31,7 +31,7 @@ use noirc_frontend::{ }, }, hir_def::traits::Trait, - node_interner::{NodeInterner, ReferenceId, StructId}, + node_interner::{FuncId, NodeInterner, ReferenceId, StructId}, parser::{Item, ItemKind, ParsedSubModule}, token::{MetaAttribute, Token, Tokens}, Kind, ParsedModule, StructType, Type, TypeBinding, @@ -40,10 +40,11 @@ use sort_text::underscore_sort_text; use crate::{ requests::to_lsp_location, trait_impl_method_stub_generator::TraitImplMethodStubGenerator, - use_segment_positions::UseSegmentPositions, utils, LspState, + use_segment_positions::UseSegmentPositions, utils, visibility::module_def_id_is_visible, + LspState, }; -use super::process_request; +use super::{process_request, TraitReexport}; mod auto_import; mod builtins; @@ -120,6 +121,8 @@ struct NodeFinder<'a> { use_segment_positions: UseSegmentPositions, self_type: Option, in_comptime: bool, + /// The function we are in, if any + func_id: Option, } impl<'a> NodeFinder<'a> { @@ -165,6 +168,7 @@ impl<'a> NodeFinder<'a> { use_segment_positions: UseSegmentPositions::default(), self_type: None, in_comptime: false, + func_id: None, } } @@ -242,7 +246,7 @@ impl<'a> NodeFinder<'a> { let mut idents: Vec = Vec::new(); // Find in which ident we are in, and in which part of it - // (it could be that we are completting in the middle of an ident) + // (it could be that we are completing in the middle of an ident) for segment in &path.segments { let ident = &segment.ident; @@ -618,7 +622,7 @@ impl<'a> NodeFinder<'a> { | Type::Forall(_, _) | Type::Constant(..) | Type::Quoted(_) - | Type::InfixExpr(_, _, _) + | Type::InfixExpr(..) | Type::Error => (), } @@ -639,6 +643,13 @@ impl<'a> NodeFinder<'a> { function_completion_kind: FunctionCompletionKind, self_prefix: bool, ) { + self.complete_trait_constraints_methods( + typ, + prefix, + function_kind, + function_completion_kind, + ); + let Some(methods_by_name) = self.interner.get_type_methods(typ) else { return; }; @@ -648,45 +659,82 @@ impl<'a> NodeFinder<'a> { let has_self_param = matches!(function_kind, FunctionKind::SelfType(..)); for (name, methods) in methods_by_name { - let Some(func_id) = - methods.find_matching_method(typ, has_self_param, self.interner).or_else(|| { - // Also try to find a method assuming typ is `&mut typ`: - // we want to suggest methods that take `&mut self` even though a variable might not - // be mutable, so a user can know they need to mark it as mutable. - let typ = Type::MutableReference(Box::new(typ.clone())); - methods.find_matching_method(&typ, has_self_param, self.interner) - }) - else { + if !name_matches(name, prefix) { continue; - }; - - if let Some(struct_id) = struct_id { - let modifiers = self.interner.function_modifiers(&func_id); - let visibility = modifiers.visibility; - if !struct_member_is_visible(struct_id, visibility, self.module_id, self.def_maps) { - continue; - } } - if is_primitive - && !method_call_is_visible( - typ, - func_id, - self.module_id, - self.interner, - self.def_maps, - ) + for (func_id, trait_id) in + methods.find_matching_methods(typ, has_self_param, self.interner) { - continue; - } + if let Some(struct_id) = struct_id { + let modifiers = self.interner.function_modifiers(&func_id); + let visibility = modifiers.visibility; + if !struct_member_is_visible( + struct_id, + visibility, + self.module_id, + self.def_maps, + ) { + continue; + } + } + + let mut trait_reexport = None; + + if let Some(trait_id) = trait_id { + let modifiers = self.interner.function_modifiers(&func_id); + let visibility = modifiers.visibility; + let module_def_id = ModuleDefId::TraitId(trait_id); + if !module_def_id_is_visible( + module_def_id, + self.module_id, + visibility, + None, // defining module + self.interner, + self.def_maps, + ) { + // Try to find a visible reexport of the trait + // that is visible from the current module + let Some((visible_module_id, name, _)) = + self.interner.get_trait_reexports(trait_id).iter().find( + |(module_id, _, visibility)| { + module_def_id_is_visible( + module_def_id, + self.module_id, + *visibility, + Some(*module_id), + self.interner, + self.def_maps, + ) + }, + ) + else { + continue; + }; + + trait_reexport = Some(TraitReexport { module_id: visible_module_id, name }); + } + } + + if is_primitive + && !method_call_is_visible( + typ, + func_id, + self.module_id, + self.interner, + self.def_maps, + ) + { + continue; + } - if name_matches(name, prefix) { let completion_items = self.function_completion_items( name, func_id, function_completion_kind, function_kind, None, // attribute first type + trait_id.map(|id| (id, trait_reexport.as_ref())), self_prefix, ); if !completion_items.is_empty() { @@ -697,6 +745,31 @@ impl<'a> NodeFinder<'a> { } } + fn complete_trait_constraints_methods( + &mut self, + typ: &Type, + prefix: &str, + function_kind: FunctionKind, + function_completion_kind: FunctionCompletionKind, + ) { + let Some(func_id) = self.func_id else { + return; + }; + + let func_meta = self.interner.function_meta(&func_id); + for constraint in &func_meta.trait_constraints { + if *typ == constraint.typ { + let trait_ = self.interner.get_trait(constraint.trait_bound.trait_id); + self.complete_trait_methods( + trait_, + prefix, + function_kind, + function_completion_kind, + ); + } + } + } + fn complete_trait_methods( &mut self, trait_: &Trait, @@ -714,6 +787,7 @@ impl<'a> NodeFinder<'a> { function_completion_kind, function_kind, None, // attribute first type + None, // trait_id (we are suggesting methods for `Trait::>|<` so no need to auto-import it) self_prefix, ); if !completion_items.is_empty() { @@ -1125,8 +1199,17 @@ impl<'a> Visitor for NodeFinder<'a> { let old_in_comptime = self.in_comptime; self.in_comptime = noir_function.def.is_comptime; + if let Some(ReferenceId::Function(func_id)) = self + .interner + .reference_at_location(Location::new(noir_function.name_ident().span(), self.file)) + { + self.func_id = Some(func_id); + } + noir_function.def.body.accept(Some(span), self); + self.func_id = None; + self.in_comptime = old_in_comptime; self.type_parameters = old_type_parameters; self.self_type = None; @@ -1207,7 +1290,7 @@ impl<'a> Visitor for NodeFinder<'a> { fn visit_trait_item_function( &mut self, - _name: &Ident, + name: &Ident, generics: &UnresolvedGenerics, parameters: &[(Ident, UnresolvedType)], return_type: &noirc_frontend::ast::FunctionReturnType, @@ -1232,7 +1315,16 @@ impl<'a> Visitor for NodeFinder<'a> { for (name, _) in parameters { self.local_variables.insert(name.to_string(), name.span()); } + + if let Some(ReferenceId::Function(func_id)) = + self.interner.reference_at_location(Location::new(name.span(), self.file)) + { + self.func_id = Some(func_id); + } + body.accept(None, self); + + self.func_id = None; }; self.type_parameters = old_type_parameters; diff --git a/noir/noir-repo/tooling/lsp/src/requests/completion/builtins.rs b/noir/noir-repo/tooling/lsp/src/requests/completion/builtins.rs index b4c7d8b6e01..10267d4719b 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/completion/builtins.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/completion/builtins.rs @@ -91,6 +91,9 @@ impl<'a> NodeFinder<'a> { AttributeTarget::Struct => { self.suggest_one_argument_attributes(prefix, &["abi"]); } + AttributeTarget::Enum => { + self.suggest_one_argument_attributes(prefix, &["abi"]); + } AttributeTarget::Function => { let no_arguments_attributes = &[ "contract_library_method", @@ -156,6 +159,7 @@ pub(super) fn keyword_builtin_type(keyword: &Keyword) -> Option<&'static str> { match keyword { Keyword::Bool => Some("bool"), Keyword::CtString => Some("CtString"), + Keyword::EnumDefinition => Some("EnumDefinition"), Keyword::Expr => Some("Expr"), Keyword::Field => Some("Field"), Keyword::FunctionDefinition => Some("FunctionDefinition"), @@ -182,6 +186,7 @@ pub(super) fn keyword_builtin_type(keyword: &Keyword) -> Option<&'static str> { | Keyword::Crate | Keyword::Dep | Keyword::Else + | Keyword::Enum | Keyword::Fn | Keyword::For | Keyword::FormatString @@ -190,6 +195,8 @@ pub(super) fn keyword_builtin_type(keyword: &Keyword) -> Option<&'static str> { | Keyword::Impl | Keyword::In | Keyword::Let + | Keyword::Loop + | Keyword::Match | Keyword::Mod | Keyword::Mut | Keyword::Pub @@ -243,6 +250,8 @@ pub(super) fn keyword_builtin_function(keyword: &Keyword) -> Option Option NodeFinder<'a> { @@ -56,6 +63,7 @@ impl<'a> NodeFinder<'a> { match target { AttributeTarget::Module => Some(Type::Quoted(QuotedType::Module)), AttributeTarget::Struct => Some(Type::Quoted(QuotedType::StructDefinition)), + AttributeTarget::Enum => Some(Type::Quoted(QuotedType::EnumDefinition)), AttributeTarget::Trait => Some(Type::Quoted(QuotedType::TraitDefinition)), AttributeTarget::Function => Some(Type::Quoted(QuotedType::FunctionDefinition)), AttributeTarget::Let => { @@ -75,6 +83,7 @@ impl<'a> NodeFinder<'a> { function_completion_kind, function_kind, attribute_first_type.as_ref(), + None, // trait_id false, // self_prefix ), ModuleDefId::TypeId(struct_id) => vec![self.struct_completion_item(name, struct_id)], @@ -144,6 +153,7 @@ impl<'a> NodeFinder<'a> { self.completion_item_with_doc_comments(ReferenceId::Global(global_id), completion_item) } + #[allow(clippy::too_many_arguments)] pub(super) fn function_completion_items( &self, name: &String, @@ -151,6 +161,7 @@ impl<'a> NodeFinder<'a> { function_completion_kind: FunctionCompletionKind, function_kind: FunctionKind, attribute_first_type: Option<&Type>, + trait_info: Option<(TraitId, Option<&TraitReexport>)>, self_prefix: bool, ) -> Vec { let func_meta = self.interner.function_meta(&func_id); @@ -223,6 +234,7 @@ impl<'a> NodeFinder<'a> { function_completion_kind, function_kind, attribute_first_type, + trait_info, self_prefix, is_macro_call, ) @@ -265,6 +277,7 @@ impl<'a> NodeFinder<'a> { function_completion_kind: FunctionCompletionKind, function_kind: FunctionKind, attribute_first_type: Option<&Type>, + trait_info: Option<(TraitId, Option<&TraitReexport>)>, self_prefix: bool, is_macro_call: bool, ) -> CompletionItem { @@ -325,7 +338,7 @@ impl<'a> NodeFinder<'a> { completion_item }; - let completion_item = match function_completion_kind { + let mut completion_item = match function_completion_kind { FunctionCompletionKind::Name => completion_item, FunctionCompletionKind::NameAndParameters => { if has_arguments { @@ -336,9 +349,69 @@ impl<'a> NodeFinder<'a> { } }; + self.auto_import_trait_if_trait_method(func_id, trait_info, &mut completion_item); + self.completion_item_with_doc_comments(ReferenceId::Function(func_id), completion_item) } + fn auto_import_trait_if_trait_method( + &self, + func_id: FuncId, + trait_info: Option<(TraitId, Option<&TraitReexport>)>, + completion_item: &mut CompletionItem, + ) -> Option<()> { + // If this is a trait method, check if the trait is in scope + let (trait_id, trait_reexport) = trait_info?; + + let trait_name = if let Some(trait_reexport) = trait_reexport { + trait_reexport.name + } else { + let trait_ = self.interner.get_trait(trait_id); + &trait_.name + }; + + let module_data = + &self.def_maps[&self.module_id.krate].modules()[self.module_id.local_id.0]; + if !module_data.scope().find_name(trait_name).is_none() { + return None; + } + + // If not, automatically import it + let current_module_parent_id = self.module_id.parent(self.def_maps); + let module_full_path = if let Some(reexport_data) = trait_reexport { + relative_module_id_path( + *reexport_data.module_id, + &self.module_id, + current_module_parent_id, + self.interner, + ) + } else { + relative_module_full_path( + ModuleDefId::FunctionId(func_id), + self.module_id, + current_module_parent_id, + self.interner, + )? + }; + let full_path = format!("{}::{}", module_full_path, trait_name); + let mut label_details = completion_item.label_details.clone().unwrap(); + label_details.detail = Some(format!("(use {})", full_path)); + completion_item.label_details = Some(label_details); + completion_item.additional_text_edits = Some(use_completion_item_additional_text_edits( + UseCompletionItemAdditionTextEditsRequest { + full_path: &full_path, + files: self.files, + file: self.file, + lines: &self.lines, + nesting: self.nesting, + auto_import_line: self.auto_import_line, + }, + &self.use_segment_positions, + )); + + None + } + fn compute_function_insert_text( &self, func_meta: &FuncMeta, diff --git a/noir/noir-repo/tooling/lsp/src/requests/completion/tests.rs b/noir/noir-repo/tooling/lsp/src/requests/completion/tests.rs index 97c7ad86d5a..8ff568e3c26 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/completion/tests.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/completion/tests.rs @@ -2861,4 +2861,220 @@ fn main() { assert_eq!(items.len(), 1); assert!(items[0].label == "bar_baz()"); } + + #[test] + async fn test_suggests_trait_method_from_where_clause_in_function() { + let src = r#" + trait Foo { + fn foo(self) -> i32; + } + + fn something(x: T) -> i32 + where + T: Foo, + { + x.fo>|< + } + "#; + let items = get_completions(src).await; + assert_eq!(items.len(), 1); + } + + #[test] + async fn test_does_not_suggest_trait_function_not_visible() { + let src = r#" + mod moo { + trait Foo { + fn foobar(); + } + + impl Foo for Field { + fn foobar() {} + } + } + + fn main() { + Field::fooba>|< + } + + "#; + assert_completion(src, vec![]).await; + } + + #[test] + async fn test_suggests_multiple_trait_methods() { + let src = r#" + mod moo { + pub trait Foo { + fn foobar(); + } + + impl Foo for Field { + fn foobar() {} + } + + pub trait Bar { + fn foobar(); + } + + impl Bar for Field { + fn foobar() {} + } + } + + fn main() { + Field::fooba>|< + } + + "#; + let items = get_completions(src).await; + assert_eq!(items.len(), 2); + } + + #[test] + async fn test_suggests_and_imports_trait_method_without_self() { + let src = r#" +mod moo { + pub trait Foo { + fn foobar(); + } + + impl Foo for Field { + fn foobar() {} + } +} + +fn main() { + Field::fooba>|< +} + "#; + let mut items = get_completions(src).await; + assert_eq!(items.len(), 1); + + let item = items.remove(0); + assert_eq!(item.label_details.unwrap().detail, Some("(use moo::Foo)".to_string())); + + let new_code = + apply_text_edits(&src.replace(">|<", ""), &item.additional_text_edits.unwrap()); + + let expected = r#"use moo::Foo; + +mod moo { + pub trait Foo { + fn foobar(); + } + + impl Foo for Field { + fn foobar() {} + } +} + +fn main() { + Field::fooba +} + "#; + assert_eq!(new_code, expected); + } + + #[test] + async fn test_suggests_and_imports_trait_method_with_self() { + let src = r#" +mod moo { + pub trait Foo { + fn foobar(self); + } + + impl Foo for Field { + fn foobar(self) {} + } +} + +fn main() { + let x: Field = 1; + x.fooba>|< +} + "#; + let mut items = get_completions(src).await; + assert_eq!(items.len(), 1); + + let item = items.remove(0); + assert_eq!(item.label_details.unwrap().detail, Some("(use moo::Foo)".to_string())); + + let new_code = + apply_text_edits(&src.replace(">|<", ""), &item.additional_text_edits.unwrap()); + + let expected = r#"use moo::Foo; + +mod moo { + pub trait Foo { + fn foobar(self); + } + + impl Foo for Field { + fn foobar(self) {} + } +} + +fn main() { + let x: Field = 1; + x.fooba +} + "#; + assert_eq!(new_code, expected); + } + + #[test] + async fn test_suggests_and_imports_trait_method_with_self_using_public_export() { + let src = r#" +mod moo { + mod nested { + pub trait Foo { + fn foobar(self); + } + + impl Foo for Field { + fn foobar(self) {} + } + } + + pub use nested::Foo as Bar; +} + +fn main() { + let x: Field = 1; + x.fooba>|< +} + "#; + let mut items = get_completions(src).await; + assert_eq!(items.len(), 1); + + let item = items.remove(0); + assert_eq!(item.label_details.unwrap().detail, Some("(use moo::Bar)".to_string())); + + let new_code = + apply_text_edits(&src.replace(">|<", ""), &item.additional_text_edits.unwrap()); + + let expected = r#"use moo::Bar; + +mod moo { + mod nested { + pub trait Foo { + fn foobar(self); + } + + impl Foo for Field { + fn foobar(self) {} + } + } + + pub use nested::Foo as Bar; +} + +fn main() { + let x: Field = 1; + x.fooba +} + "#; + assert_eq!(new_code, expected); + } } diff --git a/noir/noir-repo/tooling/lsp/src/requests/hover.rs b/noir/noir-repo/tooling/lsp/src/requests/hover.rs index bb1ea661719..5d8c50fa47b 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/hover.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/hover.rs @@ -5,16 +5,16 @@ use fm::{FileMap, PathString}; use lsp_types::{Hover, HoverContents, HoverParams, MarkupContent, MarkupKind}; use noirc_frontend::{ ast::{ItemVisibility, Visibility}, - elaborator::types::try_eval_array_length_id, hir::def_map::ModuleId, hir_def::{ expr::{HirArrayLiteral, HirExpression, HirLiteral}, + function::FuncMeta, stmt::HirPattern, traits::Trait, }, node_interner::{ DefinitionId, DefinitionKind, ExprId, FuncId, GlobalId, NodeInterner, ReferenceId, - StructId, TraitId, TypeAliasId, + StructId, TraitId, TraitImplKind, TypeAliasId, }, Generics, Shared, StructType, Type, TypeAlias, TypeBinding, TypeVariable, }; @@ -93,9 +93,7 @@ fn format_module(id: ModuleId, args: &ProcessRequestCallbackArgs) -> Option String { string.push('\n'); } + let mut print_comptime = definition.comptime; + let mut opt_value = None; + + // See if we can figure out what's the global's value + if let Some(stmt) = args.interner.get_global_let_statement(id) { + print_comptime = stmt.comptime; + opt_value = get_global_value(args.interner, stmt.expression); + } + string.push_str(" "); - if definition.comptime { + if print_comptime { string.push_str("comptime "); } if definition.mutable { @@ -217,12 +224,9 @@ fn format_global(id: GlobalId, args: &ProcessRequestCallbackArgs) -> String { string.push_str(": "); string.push_str(&format!("{}", typ)); - // See if we can figure out what's the global's value - if let Some(stmt) = args.interner.get_global_let_statement(id) { - if let Some(value) = get_global_value(args.interner, stmt.expression) { - string.push_str(" = "); - string.push_str(&value); - } + if let Some(value) = opt_value { + string.push_str(" = "); + string.push_str(&value); } string.push_str(&go_to_type_links(&typ, args.interner, args.files)); @@ -233,13 +237,6 @@ fn format_global(id: GlobalId, args: &ProcessRequestCallbackArgs) -> String { } fn get_global_value(interner: &NodeInterner, expr: ExprId) -> Option { - let span = interner.expr_span(&expr); - - // Globals as array lengths are extremely common, so we try that first. - if let Ok(result) = try_eval_array_length_id(interner, expr, span) { - return Some(result.to_string()); - } - match interner.expression(&expr) { HirExpression::Literal(literal) => match literal { HirLiteral::Array(hir_array_literal) => { @@ -300,6 +297,12 @@ fn get_exprs_global_value(interner: &NodeInterner, exprs: &[ExprId]) -> Option String { let func_meta = args.interner.function_meta(&id); + + // If this points to a trait method, see if we can figure out what's the concrete trait impl method + if let Some(func_id) = get_trait_impl_func_id(id, args, func_meta) { + return format_function(func_id, args); + } + let func_modifiers = args.interner.function_modifiers(&id); let func_name_definition_id = args.interner.definition(func_meta.name.id); @@ -396,7 +399,10 @@ fn format_function(id: FuncId, args: &ProcessRequestCallbackArgs) -> String { } string.push_str(" "); - if func_modifiers.visibility != ItemVisibility::Private { + if func_modifiers.visibility != ItemVisibility::Private + && func_meta.trait_id.is_none() + && func_meta.trait_impl.is_none() + { string.push_str(&func_modifiers.visibility.to_string()); string.push(' '); } @@ -407,8 +413,10 @@ fn format_function(id: FuncId, args: &ProcessRequestCallbackArgs) -> String { string.push_str("comptime "); } + let func_name = &func_name_definition_id.name; + string.push_str("fn "); - string.push_str(&func_name_definition_id.name); + string.push_str(func_name); format_generics(&func_meta.direct_generics, &mut string); string.push('('); let parameters = &func_meta.parameters; @@ -439,11 +447,49 @@ fn format_function(id: FuncId, args: &ProcessRequestCallbackArgs) -> String { string.push_str(&go_to_type_links(return_type, args.interner, args.files)); - append_doc_comments(args.interner, ReferenceId::Function(id), &mut string); + let had_doc_comments = + append_doc_comments(args.interner, ReferenceId::Function(id), &mut string); + if !had_doc_comments { + // If this function doesn't have doc comments, but it's a trait impl method, + // use the trait method doc comments. + if let Some(trait_impl_id) = func_meta.trait_impl { + let trait_impl = args.interner.get_trait_implementation(trait_impl_id); + let trait_impl = trait_impl.borrow(); + let trait_ = args.interner.get_trait(trait_impl.trait_id); + if let Some(func_id) = trait_.method_ids.get(func_name) { + append_doc_comments(args.interner, ReferenceId::Function(*func_id), &mut string); + } + } + } string } +fn get_trait_impl_func_id( + id: FuncId, + args: &ProcessRequestCallbackArgs, + func_meta: &FuncMeta, +) -> Option { + func_meta.trait_id?; + + let index = args.interner.find_location_index(args.location)?; + let expr_id = args.interner.get_expr_id_from_index(index)?; + let Some(TraitImplKind::Normal(trait_impl_id)) = + args.interner.get_selected_impl_for_expression(expr_id) + else { + return None; + }; + + let trait_impl = args.interner.get_trait_implementation(trait_impl_id); + let trait_impl = trait_impl.borrow(); + + let function_name = args.interner.function_name(&id); + let mut trait_impl_methods = trait_impl.methods.iter(); + let func_id = + trait_impl_methods.find(|func_id| args.interner.function_name(func_id) == function_name)?; + Some(*func_id) +} + fn format_alias(id: TypeAliasId, args: &ProcessRequestCallbackArgs) -> String { let type_alias = args.interner.get_type_alias(id); let type_alias = type_alias.borrow(); @@ -675,7 +721,7 @@ impl<'a> TypeLinksGatherer<'a> { self.gather_type_links(env); } Type::MutableReference(typ) => self.gather_type_links(typ), - Type::InfixExpr(lhs, _, rhs) => { + Type::InfixExpr(lhs, _, rhs, _) => { self.gather_type_links(lhs); self.gather_type_links(rhs); } @@ -748,13 +794,16 @@ fn format_link(name: String, location: lsp_types::Location) -> String { ) } -fn append_doc_comments(interner: &NodeInterner, id: ReferenceId, string: &mut String) { +fn append_doc_comments(interner: &NodeInterner, id: ReferenceId, string: &mut String) -> bool { if let Some(doc_comments) = interner.doc_comments(id) { string.push_str("\n\n---\n\n"); for comment in doc_comments { string.push_str(comment); string.push('\n'); } + true + } else { + false } } @@ -1108,7 +1157,15 @@ mod hover_tests { assert!(hover_text.starts_with( " two impl Bar for Foo - pub fn bar_stuff(self)" + fn bar_stuff(self)" )); } + + #[test] + async fn hover_on_trait_impl_method_uses_docs_from_trait_method() { + let hover_text = + get_hover_text("workspace", "two/src/lib.nr", Position { line: 92, character: 8 }) + .await; + assert!(hover_text.contains("Some docs")); + } } diff --git a/noir/noir-repo/tooling/lsp/src/requests/mod.rs b/noir/noir-repo/tooling/lsp/src/requests/mod.rs index 22bdda8d7d7..80f4a167a04 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/mod.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/mod.rs @@ -16,8 +16,10 @@ use lsp_types::{ }; use nargo_fmt::Config; +use noirc_frontend::ast::Ident; use noirc_frontend::graph::CrateId; -use noirc_frontend::hir::def_map::CrateDefMap; +use noirc_frontend::hir::def_map::{CrateDefMap, ModuleId}; +use noirc_frontend::parser::ParserError; use noirc_frontend::usage_tracker::UsageTracker; use noirc_frontend::{graph::Dependency, node_interner::NodeInterner}; use serde::{Deserialize, Serialize}; @@ -285,7 +287,8 @@ fn on_formatting_inner( if let Some(source) = state.input_files.get(&path) { let (module, errors) = noirc_frontend::parse_program(source); - if !errors.is_empty() { + let is_all_warnings = errors.iter().all(ParserError::is_warning); + if !is_all_warnings { return Ok(None); } @@ -617,6 +620,12 @@ pub(crate) fn find_all_references( .unwrap_or_default() } +/// Represents a trait reexported from a given module with a name. +pub(crate) struct TraitReexport<'a> { + pub(super) module_id: &'a ModuleId, + pub(super) name: &'a Ident, +} + #[cfg(test)] mod initialization { use acvm::blackbox_solver::StubbedBlackBoxSolver; @@ -631,7 +640,7 @@ mod initialization { #[test] async fn test_on_initialize() { let client = ClientSocket::new_closed(); - let mut state = LspState::new(&client, StubbedBlackBoxSolver); + let mut state = LspState::new(&client, StubbedBlackBoxSolver::default()); let params = InitializeParams::default(); let response = on_initialize(&mut state, params).await.unwrap(); assert!(matches!( diff --git a/noir/noir-repo/tooling/lsp/src/requests/test_run.rs b/noir/noir-repo/tooling/lsp/src/requests/test_run.rs index 72ae6695b82..bd53526298e 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/test_run.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/test_run.rs @@ -3,6 +3,7 @@ use std::future::{self, Future}; use crate::insert_all_files_for_workspace_into_file_manager; use async_lsp::{ErrorCode, ResponseError}; use nargo::{ + foreign_calls::DefaultForeignCallBuilder, ops::{run_test, TestStatus}, PrintOutput, }; @@ -88,10 +89,17 @@ fn on_test_run_request_inner( &mut context, &test_function, PrintOutput::Stdout, - None, - Some(workspace.root_dir.clone()), - Some(package.name.to_string()), &CompileOptions::default(), + |output, base| { + DefaultForeignCallBuilder { + output, + enable_mocks: true, + resolver_url: None, // NB without this the root and package don't do anything. + root_path: Some(workspace.root_dir.clone()), + package_name: Some(package.name.to_string()), + } + .build_with_base(base) + }, ); let result = match test_result { TestStatus::Pass => NargoTestRunResult { diff --git a/noir/noir-repo/tooling/lsp/src/solver.rs b/noir/noir-repo/tooling/lsp/src/solver.rs index a36e30a944e..df5c8eeb44f 100644 --- a/noir/noir-repo/tooling/lsp/src/solver.rs +++ b/noir/noir-repo/tooling/lsp/src/solver.rs @@ -6,6 +6,10 @@ use acvm::BlackBoxFunctionSolver; pub(super) struct WrapperSolver(pub(super) Box>); impl BlackBoxFunctionSolver for WrapperSolver { + fn pedantic_solving(&self) -> bool { + self.0.pedantic_solving() + } + fn multi_scalar_mul( &self, points: &[acvm::FieldElement], diff --git a/noir/noir-repo/tooling/lsp/src/test_utils.rs b/noir/noir-repo/tooling/lsp/src/test_utils.rs index c0505107842..c2c19b0efc7 100644 --- a/noir/noir-repo/tooling/lsp/src/test_utils.rs +++ b/noir/noir-repo/tooling/lsp/src/test_utils.rs @@ -5,7 +5,7 @@ use lsp_types::{Position, Range, Url}; pub(crate) async fn init_lsp_server(directory: &str) -> (LspState, Url) { let client = ClientSocket::new_closed(); - let mut state = LspState::new(&client, StubbedBlackBoxSolver); + let mut state = LspState::new(&client, StubbedBlackBoxSolver::default()); let root_path = std::env::current_dir() .unwrap() diff --git a/noir/noir-repo/tooling/lsp/src/trait_impl_method_stub_generator.rs b/noir/noir-repo/tooling/lsp/src/trait_impl_method_stub_generator.rs index 2ae0d526f3e..eb1709e34d0 100644 --- a/noir/noir-repo/tooling/lsp/src/trait_impl_method_stub_generator.rs +++ b/noir/noir-repo/tooling/lsp/src/trait_impl_method_stub_generator.rs @@ -361,7 +361,7 @@ impl<'a> TraitImplMethodStubGenerator<'a> { Type::Forall(_, _) => { panic!("Shouldn't get a Type::Forall"); } - Type::InfixExpr(left, op, right) => { + Type::InfixExpr(left, op, right, _) => { self.append_type(left); self.string.push(' '); self.string.push_str(&op.to_string()); diff --git a/noir/noir-repo/tooling/lsp/src/use_segment_positions.rs b/noir/noir-repo/tooling/lsp/src/use_segment_positions.rs index 246ff653245..2cd406ee773 100644 --- a/noir/noir-repo/tooling/lsp/src/use_segment_positions.rs +++ b/noir/noir-repo/tooling/lsp/src/use_segment_positions.rs @@ -193,10 +193,13 @@ impl UseSegmentPositions { } fn insert_use_segment_position(&mut self, segment: String, position: UseSegmentPosition) { - if self.use_segment_positions.get(&segment).is_none() { - self.use_segment_positions.insert(segment, position); - } else { - self.use_segment_positions.insert(segment, UseSegmentPosition::NoneOrMultiple); + match self.use_segment_positions.entry(segment) { + std::collections::hash_map::Entry::Vacant(e) => { + e.insert(position); + } + std::collections::hash_map::Entry::Occupied(mut e) => { + e.insert(UseSegmentPosition::NoneOrMultiple); + } } } } diff --git a/noir/noir-repo/tooling/lsp/test_programs/workspace/two/src/lib.nr b/noir/noir-repo/tooling/lsp/test_programs/workspace/two/src/lib.nr index 2dec902f327..d18a663b276 100644 --- a/noir/noir-repo/tooling/lsp/test_programs/workspace/two/src/lib.nr +++ b/noir/noir-repo/tooling/lsp/test_programs/workspace/two/src/lib.nr @@ -84,3 +84,12 @@ fn bar_stuff() { foo.bar_stuff(); } +trait TraitWithDocs { + /// Some docs + fn foo(); +} + +impl TraitWithDocs for Field { + fn foo() {} +} + diff --git a/noir/noir-repo/tooling/nargo/Cargo.toml b/noir/noir-repo/tooling/nargo/Cargo.toml index 1dbb9978b0b..4587638d693 100644 --- a/noir/noir-repo/tooling/nargo/Cargo.toml +++ b/noir/noir-repo/tooling/nargo/Cargo.toml @@ -21,20 +21,25 @@ noirc_errors.workspace = true noirc_frontend.workspace = true noirc_printable_type.workspace = true iter-extended.workspace = true +jsonrpsee.workspace = true +rayon.workspace = true thiserror.workspace = true tracing.workspace = true -rayon.workspace = true -jsonrpc.workspace = true -rand.workspace = true serde.workspace = true +serde_json.workspace = true walkdir = "2.5.0" +noir_fuzzer = { workspace = true } +proptest = { workspace = true } -[target.'cfg(not(target_arch = "wasm32"))'.dependencies] -noir_fuzzer.workspace = true -proptest.workspace = true +# Some dependencies are optional so we can compile to Wasm. +tokio = { workspace = true, optional = true } +rand = { workspace = true, optional = true } [dev-dependencies] -jsonrpc-http-server = "18.0" -jsonrpc-core-client = "18.0" -jsonrpc-derive = "18.0" -jsonrpc-core = "18.0" +jsonrpsee = { workspace = true, features = ["server"] } + +[features] +default = [] + +# Execution currently uses HTTP based Oracle resolvers; does not compile to Wasm. +rpc = ["jsonrpsee/http-client", "jsonrpsee/macros", "tokio/rt", "rand"] diff --git a/noir/noir-repo/tooling/nargo/src/errors.rs b/noir/noir-repo/tooling/nargo/src/errors.rs index 5256f28e36c..00c411bf7e4 100644 --- a/noir/noir-repo/tooling/nargo/src/errors.rs +++ b/noir/noir-repo/tooling/nargo/src/errors.rs @@ -16,9 +16,10 @@ use noirc_errors::{ pub use noirc_errors::Location; use noirc_driver::CrateName; -use noirc_printable_type::ForeignCallError; use thiserror::Error; +use crate::foreign_calls::ForeignCallError; + /// Errors covering situations where a package cannot be compiled. #[derive(Debug, Error)] pub enum CompileError { diff --git a/noir/noir-repo/tooling/nargo/src/foreign_calls/default.rs b/noir/noir-repo/tooling/nargo/src/foreign_calls/default.rs new file mode 100644 index 00000000000..19928e89563 --- /dev/null +++ b/noir/noir-repo/tooling/nargo/src/foreign_calls/default.rs @@ -0,0 +1,152 @@ +use acvm::AcirField; +use serde::{Deserialize, Serialize}; + +use crate::PrintOutput; + +use super::{ + layers::{self, Either, Layer, Layering}, + mocker::{DisabledMockForeignCallExecutor, MockForeignCallExecutor}, + print::PrintForeignCallExecutor, + ForeignCallExecutor, +}; + +#[cfg(feature = "rpc")] +use super::rpc::RPCForeignCallExecutor; + +/// A builder for [DefaultForeignCallLayers] where we can enable fields based on feature flags, +/// which is easier than providing different overrides for a `new` method. +pub struct DefaultForeignCallBuilder<'a> { + pub output: PrintOutput<'a>, + pub enable_mocks: bool, + + #[cfg(feature = "rpc")] + pub resolver_url: Option, + + #[cfg(feature = "rpc")] + pub root_path: Option, + + #[cfg(feature = "rpc")] + pub package_name: Option, +} + +impl<'a> Default for DefaultForeignCallBuilder<'a> { + fn default() -> Self { + Self { + output: PrintOutput::default(), + enable_mocks: true, + + #[cfg(feature = "rpc")] + resolver_url: None, + + #[cfg(feature = "rpc")] + root_path: None, + + #[cfg(feature = "rpc")] + package_name: None, + } + } +} + +impl<'a> DefaultForeignCallBuilder<'a> { + /// Override the output. + pub fn with_output(mut self, output: PrintOutput<'a>) -> Self { + self.output = output; + self + } + + /// Enable or disable mocks. + pub fn with_mocks(mut self, enabled: bool) -> Self { + self.enable_mocks = enabled; + self + } + + /// Compose the executor layers with [layers::Empty] as the default handler. + pub fn build(self) -> DefaultForeignCallLayers<'a, layers::Empty, F> + where + F: AcirField + Serialize + for<'de> Deserialize<'de> + 'a, + { + self.build_with_base(layers::Empty) + } + + /// Compose the executor layers with `base` as the default handler. + pub fn build_with_base(self, base: B) -> DefaultForeignCallLayers<'a, B, F> + where + F: AcirField + Serialize + for<'de> Deserialize<'de> + 'a, + B: ForeignCallExecutor + 'a, + { + let executor = { + #[cfg(feature = "rpc")] + { + use rand::Rng; + + base.add_layer(self.resolver_url.map(|resolver_url| { + let id = rand::thread_rng().gen(); + RPCForeignCallExecutor::new( + &resolver_url, + id, + self.root_path, + self.package_name, + ) + })) + } + #[cfg(not(feature = "rpc"))] + { + base + } + }; + + executor + .add_layer(if self.enable_mocks { + Either::Left(MockForeignCallExecutor::default()) + } else { + Either::Right(DisabledMockForeignCallExecutor) + }) + .add_layer(PrintForeignCallExecutor::new(self.output)) + } +} + +/// Facilitate static typing of layers on a base layer, so inner layers can be accessed. +#[cfg(feature = "rpc")] +pub type DefaultForeignCallLayers<'a, B, F> = Layer< + PrintForeignCallExecutor<'a>, + Layer< + Either, DisabledMockForeignCallExecutor>, + Layer, B>, + >, +>; +#[cfg(not(feature = "rpc"))] +pub type DefaultForeignCallLayers<'a, B, F> = Layer< + PrintForeignCallExecutor<'a>, + Layer, DisabledMockForeignCallExecutor>, B>, +>; + +/// Convenience constructor for code that used to create the executor this way. +#[cfg(feature = "rpc")] +pub struct DefaultForeignCallExecutor; + +/// Convenience constructors for the RPC case. Non-RPC versions are not provided +/// because once a crate opts into this within the workspace, everyone gets it +/// even if they don't want to. For the non-RPC case we can nudge people to +/// use `DefaultForeignCallBuilder` which is easier to keep flexible. +#[cfg(feature = "rpc")] +impl DefaultForeignCallExecutor { + #[allow(clippy::new_ret_no_self)] + pub fn new<'a, F>( + output: PrintOutput<'a>, + resolver_url: Option<&str>, + root_path: Option, + package_name: Option, + ) -> impl ForeignCallExecutor + 'a + where + F: AcirField + Serialize + for<'de> Deserialize<'de> + 'a, + { + DefaultForeignCallBuilder { + output, + enable_mocks: true, + resolver_url: resolver_url.map(|s| s.to_string()), + root_path, + package_name, + } + .build() + } +} diff --git a/noir/noir-repo/tooling/nargo/src/foreign_calls/layers.rs b/noir/noir-repo/tooling/nargo/src/foreign_calls/layers.rs new file mode 100644 index 00000000000..83145cacb44 --- /dev/null +++ b/noir/noir-repo/tooling/nargo/src/foreign_calls/layers.rs @@ -0,0 +1,162 @@ +use acvm::{acir::brillig::ForeignCallResult, pwg::ForeignCallWaitInfo, AcirField}; + +use super::{ForeignCallError, ForeignCallExecutor}; + +/// Returns an empty result when called. +/// +/// If all executors have no handler for the given foreign call then we cannot +/// return a correct response to the ACVM. The best we can do is to return an empty response, +/// this allows us to ignore any foreign calls which exist solely to pass information from inside +/// the circuit to the environment (e.g. custom logging) as the execution will still be able to progress. +/// +/// We optimistically return an empty response for all oracle calls as the ACVM will error +/// should a response have been required. +pub struct Empty; + +impl ForeignCallExecutor for Empty { + fn execute( + &mut self, + _foreign_call: &ForeignCallWaitInfo, + ) -> Result, ForeignCallError> { + Ok(ForeignCallResult::default()) + } +} + +/// Returns `NoHandler` for every call. +pub struct Unhandled; + +impl ForeignCallExecutor for Unhandled { + fn execute( + &mut self, + foreign_call: &ForeignCallWaitInfo, + ) -> Result, ForeignCallError> { + Err(ForeignCallError::NoHandler(foreign_call.function.clone())) + } +} + +/// Forwards to the inner executor if its own handler doesn't handle the call. +pub struct Layer { + pub handler: H, + pub inner: I, +} + +impl ForeignCallExecutor for Layer +where + H: ForeignCallExecutor, + I: ForeignCallExecutor, +{ + fn execute( + &mut self, + foreign_call: &ForeignCallWaitInfo, + ) -> Result, ForeignCallError> { + match self.handler.execute(foreign_call) { + Err(ForeignCallError::NoHandler(_)) => self.inner.execute(foreign_call), + handled => handled, + } + } +} + +impl Layer { + /// Create a layer from two handlers + pub fn new(handler: H, inner: I) -> Self { + Self { handler, inner } + } +} + +impl Layer { + /// Create a layer from a handler. + /// If the handler doesn't handle a call, a default empty response is returned. + pub fn or_empty(handler: H) -> Self { + Self { handler, inner: Empty } + } +} + +impl Layer { + /// Create a layer from a handler. + /// If the handler doesn't handle a call, `NoHandler` error is returned. + pub fn or_unhandled(handler: H) -> Self { + Self { handler, inner: Unhandled } + } +} + +impl Layer { + /// A base layer that doesn't handle anything. + pub fn unhandled() -> Self { + Self { handler: Unhandled, inner: Unhandled } + } +} + +impl Layer { + /// Add another layer on top of this one. + pub fn add_layer(self, handler: J) -> Layer { + Layer::new(handler, self) + } + + pub fn handler(&self) -> &H { + &self.handler + } + + pub fn inner(&self) -> &I { + &self.inner + } +} + +/// Compose handlers. +pub trait Layering { + /// Layer an executor on top of this one. + /// The `other` executor will be called first. + fn add_layer(self, other: L) -> Layer + where + Self: Sized + ForeignCallExecutor, + L: ForeignCallExecutor; +} + +impl Layering for T { + fn add_layer(self, other: L) -> Layer + where + T: Sized + ForeignCallExecutor, + L: ForeignCallExecutor, + { + Layer::new(other, self) + } +} + +/// A case where we can have either this or that type of handler. +pub enum Either { + Left(L), + Right(R), +} + +impl ForeignCallExecutor for Either +where + L: ForeignCallExecutor, + R: ForeignCallExecutor, +{ + fn execute( + &mut self, + foreign_call: &ForeignCallWaitInfo, + ) -> Result, ForeignCallError> { + match self { + Either::Left(left) => left.execute(foreign_call), + Either::Right(right) => right.execute(foreign_call), + } + } +} + +/// Support disabling a layer by making it optional. +/// This way we can still have a known static type for a composition, +/// because layers are always added, potentially wrapped in an `Option`. +impl ForeignCallExecutor for Option +where + H: ForeignCallExecutor, +{ + fn execute( + &mut self, + foreign_call: &ForeignCallWaitInfo, + ) -> Result, ForeignCallError> { + match self { + Some(handler) => handler.execute(foreign_call), + None => Err(ForeignCallError::NoHandler(foreign_call.function.clone())), + } + } +} diff --git a/noir/noir-repo/tooling/nargo/src/foreign_calls/mocker.rs b/noir/noir-repo/tooling/nargo/src/foreign_calls/mocker.rs index c93d16bbaf6..41fac610052 100644 --- a/noir/noir-repo/tooling/nargo/src/foreign_calls/mocker.rs +++ b/noir/noir-repo/tooling/nargo/src/foreign_calls/mocker.rs @@ -3,10 +3,9 @@ use acvm::{ pwg::ForeignCallWaitInfo, AcirField, }; -use noirc_printable_type::{decode_string_value, ForeignCallError}; -use serde::{Deserialize, Serialize}; +use noirc_abi::decode_string_value; -use super::{ForeignCall, ForeignCallExecutor}; +use super::{ForeignCall, ForeignCallError, ForeignCallExecutor}; /// This struct represents an oracle mock. It can be used for testing programs that use oracles. #[derive(Debug, PartialEq, Eq, Clone)] @@ -45,7 +44,7 @@ impl MockedCall { } #[derive(Debug, Default)] -pub(crate) struct MockForeignCallExecutor { +pub struct MockForeignCallExecutor { /// Mocks have unique ids used to identify them in Noir, allowing to update or remove them. last_mock_id: usize, /// The registered mocks @@ -78,8 +77,9 @@ impl MockForeignCallExecutor { } } -impl Deserialize<'a>> ForeignCallExecutor - for MockForeignCallExecutor +impl ForeignCallExecutor for MockForeignCallExecutor +where + F: AcirField, { fn execute( &mut self, @@ -174,3 +174,28 @@ impl Deserialize<'a>> ForeignCallExecutor } } } + +/// Handler that panics if any of the mock functions are called. +pub struct DisabledMockForeignCallExecutor; + +impl ForeignCallExecutor for DisabledMockForeignCallExecutor { + fn execute( + &mut self, + foreign_call: &ForeignCallWaitInfo, + ) -> Result, ForeignCallError> { + let foreign_call_name = foreign_call.function.as_str(); + if let Some( + ForeignCall::CreateMock + | ForeignCall::SetMockParams + | ForeignCall::GetMockLastParams + | ForeignCall::SetMockReturns + | ForeignCall::SetMockTimes + | ForeignCall::ClearMock, + ) = ForeignCall::lookup(foreign_call_name) + { + // Returning an error instead of panicking so this can be tested. + return Err(ForeignCallError::Disabled(foreign_call.function.to_string())); + } + Err(ForeignCallError::NoHandler(foreign_call.function.clone())) + } +} diff --git a/noir/noir-repo/tooling/nargo/src/foreign_calls/mod.rs b/noir/noir-repo/tooling/nargo/src/foreign_calls/mod.rs index 65ff051bcbf..f17a97cecd4 100644 --- a/noir/noir-repo/tooling/nargo/src/foreign_calls/mod.rs +++ b/noir/noir-repo/tooling/nargo/src/foreign_calls/mod.rs @@ -1,16 +1,16 @@ -use std::path::PathBuf; +use acvm::{acir::brillig::ForeignCallResult, pwg::ForeignCallWaitInfo}; +use thiserror::Error; -use acvm::{acir::brillig::ForeignCallResult, pwg::ForeignCallWaitInfo, AcirField}; -use mocker::MockForeignCallExecutor; -use noirc_printable_type::ForeignCallError; -use print::{PrintForeignCallExecutor, PrintOutput}; -use rand::Rng; -use rpc::RPCForeignCallExecutor; -use serde::{Deserialize, Serialize}; +pub mod layers; +pub mod mocker; +pub mod print; -pub(crate) mod mocker; -pub(crate) mod print; -pub(crate) mod rpc; +pub mod default; +#[cfg(feature = "rpc")] +pub mod rpc; +pub use default::DefaultForeignCallBuilder; +#[cfg(feature = "rpc")] +pub use default::DefaultForeignCallExecutor; pub trait ForeignCallExecutor { fn execute( @@ -64,77 +64,23 @@ impl ForeignCall { } } -#[derive(Debug, Default)] -pub struct DefaultForeignCallExecutor<'a, F> { - /// The executor for any [`ForeignCall::Print`] calls. - printer: PrintForeignCallExecutor<'a>, - mocker: MockForeignCallExecutor, - external: Option, -} +#[derive(Debug, Error)] +pub enum ForeignCallError { + #[error("Attempted to call disabled foreign call `{0}`")] + Disabled(String), -impl<'a, F: Default> DefaultForeignCallExecutor<'a, F> { - pub fn new( - output: PrintOutput<'a>, - resolver_url: Option<&str>, - root_path: Option, - package_name: Option, - ) -> Self { - let id = rand::thread_rng().gen(); - let printer = PrintForeignCallExecutor { output }; - let external_resolver = resolver_url.map(|resolver_url| { - RPCForeignCallExecutor::new(resolver_url, id, root_path, package_name) - }); - DefaultForeignCallExecutor { - printer, - mocker: MockForeignCallExecutor::default(), - external: external_resolver, - } - } -} + #[error("No handler could be found for foreign call `{0}`")] + NoHandler(String), -impl<'a, F: AcirField + Serialize + for<'b> Deserialize<'b>> ForeignCallExecutor - for DefaultForeignCallExecutor<'a, F> -{ - fn execute( - &mut self, - foreign_call: &ForeignCallWaitInfo, - ) -> Result, ForeignCallError> { - let foreign_call_name = foreign_call.function.as_str(); - match ForeignCall::lookup(foreign_call_name) { - Some(ForeignCall::Print) => self.printer.execute(foreign_call), - Some( - ForeignCall::CreateMock - | ForeignCall::SetMockParams - | ForeignCall::GetMockLastParams - | ForeignCall::SetMockReturns - | ForeignCall::SetMockTimes - | ForeignCall::ClearMock, - ) => self.mocker.execute(foreign_call), + #[error("Foreign call inputs needed for execution are missing")] + MissingForeignCallInputs, - None => { - // First check if there's any defined mock responses for this foreign call. - match self.mocker.execute(foreign_call) { - Err(ForeignCallError::NoHandler(_)) => (), - response_or_error => return response_or_error, - }; + #[error("Could not parse PrintableType argument. {0}")] + ParsingError(#[from] serde_json::Error), - if let Some(external_resolver) = &mut self.external { - // If the user has registered an external resolver then we forward any remaining oracle calls there. - match external_resolver.execute(foreign_call) { - Err(ForeignCallError::NoHandler(_)) => (), - response_or_error => return response_or_error, - }; - } + #[error("Failed calling external resolver. {0}")] + ExternalResolverError(#[from] jsonrpsee::core::client::Error), - // If all executors have no handler for the given foreign call then we cannot - // return a correct response to the ACVM. The best we can do is to return an empty response, - // this allows us to ignore any foreign calls which exist solely to pass information from inside - // the circuit to the environment (e.g. custom logging) as the execution will still be able to progress. - // - // We optimistically return an empty response for all oracle calls as the ACVM will error - // should a response have been required. - Ok(ForeignCallResult::default()) - } - } - } + #[error("Assert message resolved after an unsatisfied constrain. {0}")] + ResolvedAssertMessage(String), } diff --git a/noir/noir-repo/tooling/nargo/src/foreign_calls/print.rs b/noir/noir-repo/tooling/nargo/src/foreign_calls/print.rs index 8b2b5efd8b6..fb5621da942 100644 --- a/noir/noir-repo/tooling/nargo/src/foreign_calls/print.rs +++ b/noir/noir-repo/tooling/nargo/src/foreign_calls/print.rs @@ -1,7 +1,12 @@ -use acvm::{acir::brillig::ForeignCallResult, pwg::ForeignCallWaitInfo, AcirField}; -use noirc_printable_type::{ForeignCallError, PrintableValueDisplay}; +use acvm::{ + acir::brillig::{ForeignCallParam, ForeignCallResult}, + pwg::ForeignCallWaitInfo, + AcirField, +}; +use noirc_abi::{decode_printable_value, decode_string_value}; +use noirc_printable_type::{PrintableType, PrintableValueDisplay}; -use super::{ForeignCall, ForeignCallExecutor}; +use super::{ForeignCall, ForeignCallError, ForeignCallExecutor}; #[derive(Debug, Default)] pub enum PrintOutput<'a> { @@ -12,8 +17,14 @@ pub enum PrintOutput<'a> { } #[derive(Debug, Default)] -pub(crate) struct PrintForeignCallExecutor<'a> { - pub(crate) output: PrintOutput<'a>, +pub struct PrintForeignCallExecutor<'a> { + output: PrintOutput<'a>, +} + +impl<'a> PrintForeignCallExecutor<'a> { + pub fn new(output: PrintOutput<'a>) -> Self { + Self { output } + } } impl ForeignCallExecutor for PrintForeignCallExecutor<'_> { @@ -32,7 +43,8 @@ impl ForeignCallExecutor for PrintForeignCallExecutor<'_> { .ok_or(ForeignCallError::MissingForeignCallInputs)? .1; - let display_values: PrintableValueDisplay = foreign_call_inputs.try_into()?; + let display_values: PrintableValueDisplay = + try_from_params(foreign_call_inputs)?; let display_string = format!("{display_values}{}", if skip_newline { "" } else { "\n" }); @@ -50,3 +62,72 @@ impl ForeignCallExecutor for PrintForeignCallExecutor<'_> { } } } + +fn try_from_params( + foreign_call_inputs: &[ForeignCallParam], +) -> Result, ForeignCallError> { + let (is_fmt_str, foreign_call_inputs) = + foreign_call_inputs.split_last().ok_or(ForeignCallError::MissingForeignCallInputs)?; + + if is_fmt_str.unwrap_field().is_one() { + convert_fmt_string_inputs(foreign_call_inputs) + } else { + convert_string_inputs(foreign_call_inputs) + } +} + +fn convert_string_inputs( + foreign_call_inputs: &[ForeignCallParam], +) -> Result, ForeignCallError> { + // Fetch the PrintableType from the foreign call input + // The remaining input values should hold what is to be printed + let (printable_type_as_values, input_values) = + foreign_call_inputs.split_last().ok_or(ForeignCallError::MissingForeignCallInputs)?; + let printable_type = fetch_printable_type(printable_type_as_values)?; + + // We must use a flat map here as each value in a struct will be in a separate input value + let mut input_values_as_fields = input_values.iter().flat_map(|param| param.fields()); + + let value = decode_printable_value(&mut input_values_as_fields, &printable_type); + + Ok(PrintableValueDisplay::Plain(value, printable_type)) +} + +fn convert_fmt_string_inputs( + foreign_call_inputs: &[ForeignCallParam], +) -> Result, ForeignCallError> { + let (message, input_and_printable_types) = + foreign_call_inputs.split_first().ok_or(ForeignCallError::MissingForeignCallInputs)?; + + let message_as_fields = message.fields(); + let message_as_string = decode_string_value(&message_as_fields); + + let (num_values, input_and_printable_types) = input_and_printable_types + .split_first() + .ok_or(ForeignCallError::MissingForeignCallInputs)?; + + let mut output = Vec::new(); + let num_values = num_values.unwrap_field().to_u128() as usize; + + let types_start_at = input_and_printable_types.len() - num_values; + let mut input_iter = + input_and_printable_types[0..types_start_at].iter().flat_map(|param| param.fields()); + for printable_type in input_and_printable_types.iter().skip(types_start_at) { + let printable_type = fetch_printable_type(printable_type)?; + let value = decode_printable_value(&mut input_iter, &printable_type); + + output.push((value, printable_type)); + } + + Ok(PrintableValueDisplay::FmtString(message_as_string, output)) +} + +fn fetch_printable_type( + printable_type: &ForeignCallParam, +) -> Result { + let printable_type_as_fields = printable_type.fields(); + let printable_type_as_string = decode_string_value(&printable_type_as_fields); + let printable_type: PrintableType = serde_json::from_str(&printable_type_as_string)?; + + Ok(printable_type) +} diff --git a/noir/noir-repo/tooling/nargo/src/foreign_calls/rpc.rs b/noir/noir-repo/tooling/nargo/src/foreign_calls/rpc.rs index 0653eb1c7e3..89a748b6c45 100644 --- a/noir/noir-repo/tooling/nargo/src/foreign_calls/rpc.rs +++ b/noir/noir-repo/tooling/nargo/src/foreign_calls/rpc.rs @@ -1,25 +1,31 @@ use std::path::PathBuf; use acvm::{acir::brillig::ForeignCallResult, pwg::ForeignCallWaitInfo, AcirField}; -use jsonrpc::{arg as build_json_rpc_arg, minreq_http::Builder, Client}; -use noirc_printable_type::ForeignCallError; +use jsonrpsee::{ + core::client::ClientT, + http_client::{HttpClient, HttpClientBuilder}, + rpc_params, +}; use serde::{Deserialize, Serialize}; -use super::ForeignCallExecutor; +use super::{ForeignCallError, ForeignCallExecutor}; #[derive(Debug)] -pub(crate) struct RPCForeignCallExecutor { +pub struct RPCForeignCallExecutor { /// A randomly generated id for this `DefaultForeignCallExecutor`. /// /// This is used so that a single `external_resolver` can distinguish between requests from multiple /// instantiations of `DefaultForeignCallExecutor`. id: u64, /// JSON RPC client to resolve foreign calls - external_resolver: Client, + external_resolver: HttpClient, /// Root path to the program or workspace in execution. root_path: Option, /// Name of the package in execution package_name: Option, + /// Runtime to execute asynchronous tasks on. + /// See [bridging](https://tokio.rs/tokio/topics/bridging). + runtime: tokio::runtime::Runtime, } #[derive(Debug, Serialize, Deserialize)] @@ -31,59 +37,76 @@ struct ResolveForeignCallRequest { /// performed in parallel. session_id: u64, - #[serde(flatten)] /// The foreign call which the external RPC server is to provide a response for. + #[serde(flatten)] function_call: ForeignCallWaitInfo, - #[serde(skip_serializing_if = "Option::is_none")] /// Root path to the program or workspace in execution. - root_path: Option, #[serde(skip_serializing_if = "Option::is_none")] + root_path: Option, + /// Name of the package in execution + #[serde(skip_serializing_if = "Option::is_none")] package_name: Option, } +type ResolveForeignCallResult = Result, ForeignCallError>; + impl RPCForeignCallExecutor { - pub(crate) fn new( + pub fn new( resolver_url: &str, id: u64, root_path: Option, package_name: Option, ) -> Self { - let mut transport_builder = - Builder::new().url(resolver_url).expect("Invalid oracle resolver URL"); + let mut client_builder = HttpClientBuilder::new(); if let Some(Ok(timeout)) = std::env::var("NARGO_FOREIGN_CALL_TIMEOUT").ok().map(|timeout| timeout.parse()) { let timeout_duration = std::time::Duration::from_millis(timeout); - transport_builder = transport_builder.timeout(timeout_duration); + client_builder = client_builder.request_timeout(timeout_duration); }; - let oracle_resolver = Client::with_transport(transport_builder.build()); - RPCForeignCallExecutor { external_resolver: oracle_resolver, id, root_path, package_name } + let oracle_resolver = + client_builder.build(resolver_url).expect("Invalid oracle resolver URL"); + + // Opcodes are executed in the `ProgramExecutor::execute_circuit` one by one in a loop, + // we don't need a concurrent thread pool. + let runtime = tokio::runtime::Builder::new_current_thread() + .enable_time() + .enable_io() + .build() + .expect("failed to build tokio runtime"); + + RPCForeignCallExecutor { + external_resolver: oracle_resolver, + id, + root_path, + package_name, + runtime, + } } } -impl Deserialize<'a>> ForeignCallExecutor - for RPCForeignCallExecutor +impl ForeignCallExecutor for RPCForeignCallExecutor +where + F: AcirField + Serialize + for<'a> Deserialize<'a>, { - fn execute( - &mut self, - foreign_call: &ForeignCallWaitInfo, - ) -> Result, ForeignCallError> { - let encoded_params = vec![build_json_rpc_arg(ResolveForeignCallRequest { + /// Execute an async call blocking the current thread. + /// This method cannot be called from inside a `tokio` runtime, for that to work + /// we need to offload the execution into a different thread; see the tests. + fn execute(&mut self, foreign_call: &ForeignCallWaitInfo) -> ResolveForeignCallResult { + let encoded_params = rpc_params!(ResolveForeignCallRequest { session_id: self.id, function_call: foreign_call.clone(), root_path: self.root_path.clone().map(|path| path.to_str().unwrap().to_string()), package_name: self.package_name.clone(), - })]; - - let req = self.external_resolver.build_request("resolve_foreign_call", &encoded_params); - - let response = self.external_resolver.send_request(req)?; + }); - let parsed_response: ForeignCallResult = response.result()?; + let parsed_response = self.runtime.block_on(async { + self.external_resolver.request("resolve_foreign_call", encoded_params).await + })?; Ok(parsed_response) } @@ -95,20 +118,23 @@ mod tests { acir::brillig::ForeignCallParam, brillig_vm::brillig::ForeignCallResult, pwg::ForeignCallWaitInfo, FieldElement, }; - use jsonrpc_core::Result as RpcResult; - use jsonrpc_derive::rpc; - use jsonrpc_http_server::{Server, ServerBuilder}; - - use super::{ForeignCallExecutor, RPCForeignCallExecutor, ResolveForeignCallRequest}; + use jsonrpsee::proc_macros::rpc; + use jsonrpsee::server::Server; + use jsonrpsee::types::ErrorObjectOwned; + use tokio::sync::{mpsc, oneshot}; + + use super::{ + ForeignCallExecutor, RPCForeignCallExecutor, ResolveForeignCallRequest, + ResolveForeignCallResult, + }; - #[allow(unreachable_pub)] - #[rpc] - pub trait OracleResolver { - #[rpc(name = "resolve_foreign_call")] + #[rpc(server)] + trait OracleResolver { + #[method(name = "resolve_foreign_call")] fn resolve_foreign_call( &self, req: ResolveForeignCallRequest, - ) -> RpcResult>; + ) -> Result, ErrorObjectOwned>; } struct OracleResolverImpl; @@ -129,99 +155,129 @@ mod tests { } } - impl OracleResolver for OracleResolverImpl { + impl OracleResolverServer for OracleResolverImpl { fn resolve_foreign_call( &self, req: ResolveForeignCallRequest, - ) -> RpcResult> { + ) -> Result, ErrorObjectOwned> { let response = match req.function_call.function.as_str() { "sum" => self.sum(req.function_call.inputs[0].clone()), "echo" => self.echo(req.function_call.inputs[0].clone()), "id" => FieldElement::from(req.session_id as u128).into(), - _ => panic!("unexpected foreign call"), }; Ok(response) } } - fn build_oracle_server() -> (Server, String) { - let mut io = jsonrpc_core::IoHandler::new(); - io.extend_with(OracleResolverImpl.to_delegate()); + /// The test client send its request and a response channel. + type RPCForeignCallClientRequest = ( + ForeignCallWaitInfo, + oneshot::Sender>, + ); - // Choosing port 0 results in a random port being assigned. - let server = ServerBuilder::new(io) - .start_http(&"127.0.0.1:0".parse().expect("Invalid address")) - .expect("Could not start server"); + /// Async client used in the tests. + #[derive(Clone)] + struct RPCForeignCallClient { + tx: mpsc::UnboundedSender, + } + + impl RPCForeignCallExecutor { + /// Spawn and run the executor in the background until all clients are closed. + fn run(mut self) -> RPCForeignCallClient { + let (tx, mut rx) = mpsc::unbounded_channel::(); + let handle = tokio::task::spawn_blocking(move || { + while let Some((req, tx)) = rx.blocking_recv() { + let res = self.execute(&req); + let _ = tx.send(res); + } + }); + // The task will finish when the client goes out of scope. + drop(handle); + RPCForeignCallClient { tx } + } + } + + impl RPCForeignCallClient { + /// Asynchronously execute a foreign call. + async fn execute( + &self, + req: &ForeignCallWaitInfo, + ) -> ResolveForeignCallResult { + let (tx, rx) = oneshot::channel(); + self.tx.send((req.clone(), tx)).expect("failed to send to executor"); + rx.await.expect("failed to receive from executor") + } + } - let url = format!("http://{}", server.address()); - (server, url) + /// Start running the Oracle server or a random port, returning the listen URL. + async fn build_oracle_server() -> std::io::Result { + // Choosing port 0 results in a random port being assigned. + let server = Server::builder().build("127.0.0.1:0").await?; + let addr = server.local_addr()?; + let handle = server.start(OracleResolverImpl.into_rpc()); + let url = format!("http://{}", addr); + // In this test we don't care about doing shutdown so let's it run forever. + tokio::spawn(handle.stopped()); + Ok(url) } - #[test] - fn test_oracle_resolver_echo() { - let (server, url) = build_oracle_server(); + #[tokio::test] + async fn test_oracle_resolver_echo() { + let url = build_oracle_server().await.unwrap(); - let mut executor = RPCForeignCallExecutor::new(&url, 1, None, None); + let executor = RPCForeignCallExecutor::new(&url, 1, None, None).run(); let foreign_call: ForeignCallWaitInfo = ForeignCallWaitInfo { function: "echo".to_string(), inputs: vec![ForeignCallParam::Single(1_u128.into())], }; - let result = executor.execute(&foreign_call); + let result = executor.execute(&foreign_call).await; assert_eq!(result.unwrap(), ForeignCallResult { values: foreign_call.inputs }); - - server.close(); } - #[test] - fn test_oracle_resolver_sum() { - let (server, url) = build_oracle_server(); + #[tokio::test] + async fn test_oracle_resolver_sum() { + let url = build_oracle_server().await.unwrap(); - let mut executor = RPCForeignCallExecutor::new(&url, 2, None, None); + let executor = RPCForeignCallExecutor::new(&url, 2, None, None).run(); let foreign_call: ForeignCallWaitInfo = ForeignCallWaitInfo { function: "sum".to_string(), inputs: vec![ForeignCallParam::Array(vec![1_usize.into(), 2_usize.into()])], }; - let result = executor.execute(&foreign_call); + let result = executor.execute(&foreign_call).await; assert_eq!(result.unwrap(), FieldElement::from(3_usize).into()); - - server.close(); } - #[test] - fn foreign_call_executor_id_is_persistent() { - let (server, url) = build_oracle_server(); + #[tokio::test] + async fn foreign_call_executor_id_is_persistent() { + let url = build_oracle_server().await.unwrap(); - let mut executor = RPCForeignCallExecutor::new(&url, 3, None, None); + let executor = RPCForeignCallExecutor::new(&url, 3, None, None).run(); let foreign_call: ForeignCallWaitInfo = ForeignCallWaitInfo { function: "id".to_string(), inputs: Vec::new() }; - let result_1 = executor.execute(&foreign_call).unwrap(); - let result_2 = executor.execute(&foreign_call).unwrap(); + let result_1 = executor.execute(&foreign_call).await.unwrap(); + let result_2 = executor.execute(&foreign_call).await.unwrap(); assert_eq!(result_1, result_2); - - server.close(); } - #[test] - fn oracle_resolver_rpc_can_distinguish_executors() { - let (server, url) = build_oracle_server(); + #[tokio::test] + async fn oracle_resolver_rpc_can_distinguish_executors() { + let url = build_oracle_server().await.unwrap(); - let mut executor_1 = RPCForeignCallExecutor::new(&url, 4, None, None); - let mut executor_2 = RPCForeignCallExecutor::new(&url, 5, None, None); + let executor_1 = RPCForeignCallExecutor::new(&url, 4, None, None).run(); + let executor_2 = RPCForeignCallExecutor::new(&url, 5, None, None).run(); let foreign_call: ForeignCallWaitInfo = ForeignCallWaitInfo { function: "id".to_string(), inputs: Vec::new() }; - let result_1 = executor_1.execute(&foreign_call).unwrap(); - let result_2 = executor_2.execute(&foreign_call).unwrap(); + let result_1 = executor_1.execute(&foreign_call).await.unwrap(); + let result_2 = executor_2.execute(&foreign_call).await.unwrap(); assert_ne!(result_1, result_2); - - server.close(); } } diff --git a/noir/noir-repo/tooling/nargo/src/lib.rs b/noir/noir-repo/tooling/nargo/src/lib.rs index ee7b2e4809a..30f25356e41 100644 --- a/noir/noir-repo/tooling/nargo/src/lib.rs +++ b/noir/noir-repo/tooling/nargo/src/lib.rs @@ -14,6 +14,9 @@ pub mod ops; pub mod package; pub mod workspace; +pub use self::errors::NargoError; +pub use self::foreign_calls::print::PrintOutput; + use std::{ collections::{BTreeMap, HashMap, HashSet}, path::PathBuf, @@ -29,9 +32,6 @@ use package::{Dependency, Package}; use rayon::prelude::*; use walkdir::WalkDir; -pub use self::errors::NargoError; -pub use self::foreign_calls::print::PrintOutput; - pub fn prepare_dependencies( context: &mut Context, parent_crate: CrateId, @@ -97,7 +97,7 @@ fn insert_all_files_for_package_into_file_manager( .parent() .unwrap_or_else(|| panic!("The entry path is expected to be a single file within a directory and so should have a parent {:?}", package.entry_path)); - for entry in WalkDir::new(entry_path_parent) { + for entry in WalkDir::new(entry_path_parent).sort_by_file_name() { let Ok(entry) = entry else { continue; }; @@ -172,6 +172,7 @@ pub fn parse_all(file_manager: &FileManager) -> ParsedFiles { .collect() } +#[tracing::instrument(level = "trace", skip_all)] pub fn prepare_package<'file_manager, 'parsed_files>( file_manager: &'file_manager FileManager, parsed_files: &'parsed_files ParsedFiles, diff --git a/noir/noir-repo/tooling/nargo/src/ops/check.rs b/noir/noir-repo/tooling/nargo/src/ops/check.rs index 707353ccdad..f22def8bd91 100644 --- a/noir/noir-repo/tooling/nargo/src/ops/check.rs +++ b/noir/noir-repo/tooling/nargo/src/ops/check.rs @@ -3,6 +3,7 @@ use noirc_driver::{CompiledProgram, ErrorsAndWarnings}; use noirc_errors::{CustomDiagnostic, FileDiagnostic}; /// Run each function through a circuit simulator to check that they are solvable. +#[tracing::instrument(level = "trace", skip_all)] pub fn check_program(compiled_program: &CompiledProgram) -> Result<(), ErrorsAndWarnings> { for (i, circuit) in compiled_program.program.functions.iter().enumerate() { let mut simulator = CircuitSimulator::default(); diff --git a/noir/noir-repo/tooling/nargo/src/ops/mod.rs b/noir/noir-repo/tooling/nargo/src/ops/mod.rs index 04efeb5a9ec..7a52a829be3 100644 --- a/noir/noir-repo/tooling/nargo/src/ops/mod.rs +++ b/noir/noir-repo/tooling/nargo/src/ops/mod.rs @@ -3,10 +3,10 @@ pub use self::compile::{ collect_errors, compile_contract, compile_program, compile_program_with_debug_instrumenter, compile_workspace, report_errors, }; -pub use self::execute::{execute_program, execute_program_with_profiling}; pub use self::optimize::{optimize_contract, optimize_program}; pub use self::transform::{transform_contract, transform_program}; +pub use self::execute::{execute_program, execute_program_with_profiling}; pub use self::test::{run_test, TestStatus}; mod check; diff --git a/noir/noir-repo/tooling/nargo/src/ops/test.rs b/noir/noir-repo/tooling/nargo/src/ops/test.rs index 1306150518d..a2f94cd61eb 100644 --- a/noir/noir-repo/tooling/nargo/src/ops/test.rs +++ b/noir/noir-repo/tooling/nargo/src/ops/test.rs @@ -1,5 +1,3 @@ -use std::path::PathBuf; - use acvm::{ acir::{ brillig::ForeignCallResult, @@ -12,18 +10,10 @@ use noirc_abi::Abi; use noirc_driver::{compile_no_check, CompileError, CompileOptions, DEFAULT_EXPRESSION_WIDTH}; use noirc_errors::{debug_info::DebugInfo, FileDiagnostic}; use noirc_frontend::hir::{def_map::TestFunction, Context}; -use noirc_printable_type::ForeignCallError; -use rand::Rng; -use serde::{Deserialize, Serialize}; use crate::{ errors::try_to_diagnose_runtime_error, - foreign_calls::{ - mocker::MockForeignCallExecutor, - print::{PrintForeignCallExecutor, PrintOutput}, - rpc::RPCForeignCallExecutor, - ForeignCall, ForeignCallExecutor, - }, + foreign_calls::{layers, print::PrintOutput, ForeignCallError, ForeignCallExecutor}, NargoError, }; @@ -43,17 +33,19 @@ impl TestStatus { } } -#[allow(clippy::too_many_arguments)] -pub fn run_test>( +pub fn run_test<'a, B, F, E>( blackbox_solver: &B, context: &mut Context, test_function: &TestFunction, - output: PrintOutput<'_>, - foreign_call_resolver_url: Option<&str>, - root_path: Option, - package_name: Option, + output: PrintOutput<'a>, config: &CompileOptions, -) -> TestStatus { + build_foreign_call_executor: F, +) -> TestStatus +where + B: BlackBoxFunctionSolver, + F: Fn(PrintOutput<'a>, layers::Unhandled) -> E + 'a, + E: ForeignCallExecutor, +{ let test_function_has_no_arguments = context .def_interner .function_meta(&test_function.get_id()) @@ -70,12 +62,9 @@ pub fn run_test>( if test_function_has_no_arguments { // Run the backend to ensure the PWG evaluates functions like std::hash::pedersen, // otherwise constraints involving these expressions will not error. - let mut foreign_call_executor = TestForeignCallExecutor::new( - output, - foreign_call_resolver_url, - root_path, - package_name, - ); + // Use a base layer that doesn't handle anything, which we handle in the `execute` below. + let inner_executor = build_foreign_call_executor(output, layers::Unhandled); + let mut foreign_call_executor = TestForeignCallExecutor::new(inner_executor); let circuit_execution = execute_program( &compiled_program.program, @@ -107,69 +96,58 @@ pub fn run_test>( status } } else { - #[cfg(target_arch = "wasm32")] - { - // We currently don't support fuzz testing on wasm32 as the u128 strategies do not exist on this platform. - TestStatus::Fail { - message: "Fuzz tests are not supported on wasm32".to_string(), - error_diagnostic: None, + use acvm::acir::circuit::Program; + use noir_fuzzer::FuzzedExecutor; + use proptest::test_runner::Config; + use proptest::test_runner::TestRunner; + + let runner = + TestRunner::new(Config { failure_persistence: None, ..Config::default() }); + + let abi = compiled_program.abi.clone(); + let debug = compiled_program.debug.clone(); + + let executor = |program: &Program, + initial_witness: WitnessMap| + -> Result, String> { + // Use a base layer that doesn't handle anything, which we handle in the `execute` below. + let inner_executor = + build_foreign_call_executor(PrintOutput::None, layers::Unhandled); + + let mut foreign_call_executor = TestForeignCallExecutor::new(inner_executor); + + let circuit_execution = execute_program( + program, + initial_witness, + blackbox_solver, + &mut foreign_call_executor, + ); + + // Check if a failure was actually expected. + let status = test_status_program_compile_pass( + test_function, + &abi, + &debug, + &circuit_execution, + ); + + if let TestStatus::Fail { message, error_diagnostic: _ } = status { + Err(message) + } else { + // The fuzzer doesn't care about the actual result. + Ok(WitnessStack::default()) } - } - - #[cfg(not(target_arch = "wasm32"))] - { - use acvm::acir::circuit::Program; - use noir_fuzzer::FuzzedExecutor; - use proptest::test_runner::Config; - use proptest::test_runner::TestRunner; - - let runner = - TestRunner::new(Config { failure_persistence: None, ..Config::default() }); - - let abi = compiled_program.abi.clone(); - let debug = compiled_program.debug.clone(); - - let executor = - |program: &Program, - initial_witness: WitnessMap| - -> Result, String> { - let circuit_execution = execute_program( - program, - initial_witness, - blackbox_solver, - &mut TestForeignCallExecutor::::new( - PrintOutput::None, - foreign_call_resolver_url, - root_path.clone(), - package_name.clone(), - ), - ); - - let status = test_status_program_compile_pass( - test_function, - &abi, - &debug, - &circuit_execution, - ); - - if let TestStatus::Fail { message, error_diagnostic: _ } = status { - Err(message) - } else { - // The fuzzer doesn't care about the actual result. - Ok(WitnessStack::default()) - } - }; + }; - let fuzzer = FuzzedExecutor::new(compiled_program.into(), executor, runner); + let fuzzer = FuzzedExecutor::new(compiled_program.into(), executor, runner); - let result = fuzzer.fuzz(); - if result.success { - TestStatus::Pass - } else { - TestStatus::Fail { - message: result.reason.unwrap_or_default(), - error_diagnostic: None, - } + let result = fuzzer.fuzz(); + if result.success { + TestStatus::Pass + } else { + TestStatus::Fail { + message: result.reason.unwrap_or_default(), + error_diagnostic: None, } } } @@ -278,38 +256,21 @@ fn check_expected_failure_message( } /// A specialized foreign call executor which tracks whether it has encountered any unknown foreign calls -struct TestForeignCallExecutor<'a, F> { - /// The executor for any [`ForeignCall::Print`] calls. - printer: PrintForeignCallExecutor<'a>, - mocker: MockForeignCallExecutor, - external: Option, - +struct TestForeignCallExecutor { + executor: E, encountered_unknown_foreign_call: bool, } -impl<'a, F: Default> TestForeignCallExecutor<'a, F> { - fn new( - output: PrintOutput<'a>, - resolver_url: Option<&str>, - root_path: Option, - package_name: Option, - ) -> Self { - let id = rand::thread_rng().gen(); - let printer = PrintForeignCallExecutor { output }; - let external_resolver = resolver_url.map(|resolver_url| { - RPCForeignCallExecutor::new(resolver_url, id, root_path, package_name) - }); - TestForeignCallExecutor { - printer, - mocker: MockForeignCallExecutor::default(), - external: external_resolver, - encountered_unknown_foreign_call: false, - } +impl TestForeignCallExecutor { + fn new(executor: E) -> Self { + Self { executor, encountered_unknown_foreign_call: false } } } -impl<'a, F: AcirField + Serialize + for<'b> Deserialize<'b>> ForeignCallExecutor - for TestForeignCallExecutor<'a, F> +impl ForeignCallExecutor for TestForeignCallExecutor +where + F: AcirField, + E: ForeignCallExecutor, { fn execute( &mut self, @@ -317,46 +278,14 @@ impl<'a, F: AcirField + Serialize + for<'b> Deserialize<'b>> ForeignCallExecutor ) -> Result, ForeignCallError> { // If the circuit has reached a new foreign call opcode then it can't have failed from any previous unknown foreign calls. self.encountered_unknown_foreign_call = false; - - let foreign_call_name = foreign_call.function.as_str(); - match ForeignCall::lookup(foreign_call_name) { - Some(ForeignCall::Print) => self.printer.execute(foreign_call), - - Some( - ForeignCall::CreateMock - | ForeignCall::SetMockParams - | ForeignCall::GetMockLastParams - | ForeignCall::SetMockReturns - | ForeignCall::SetMockTimes - | ForeignCall::ClearMock, - ) => self.mocker.execute(foreign_call), - - None => { - // First check if there's any defined mock responses for this foreign call. - match self.mocker.execute(foreign_call) { - Err(ForeignCallError::NoHandler(_)) => (), - response_or_error => return response_or_error, - }; - - if let Some(external_resolver) = &mut self.external { - // If the user has registered an external resolver then we forward any remaining oracle calls there. - match external_resolver.execute(foreign_call) { - Err(ForeignCallError::NoHandler(_)) => (), - response_or_error => return response_or_error, - }; - } - + match self.executor.execute(foreign_call) { + Err(ForeignCallError::NoHandler(_)) => { self.encountered_unknown_foreign_call = true; - - // If all executors have no handler for the given foreign call then we cannot - // return a correct response to the ACVM. The best we can do is to return an empty response, - // this allows us to ignore any foreign calls which exist solely to pass information from inside - // the circuit to the environment (e.g. custom logging) as the execution will still be able to progress. - // - // We optimistically return an empty response for all oracle calls as the ACVM will error - // should a response have been required. - Ok(ForeignCallResult::default()) + // If the inner executor cannot handle this foreign call, then it's very likely that this is a custom + // foreign call. We then return an empty response in case the foreign call doesn't need return values. + layers::Empty.execute(foreign_call) } + other => other, } } } diff --git a/noir/noir-repo/tooling/nargo_cli/Cargo.toml b/noir/noir-repo/tooling/nargo_cli/Cargo.toml index 5603b7f4fca..001306bb162 100644 --- a/noir/noir-repo/tooling/nargo_cli/Cargo.toml +++ b/noir/noir-repo/tooling/nargo_cli/Cargo.toml @@ -27,7 +27,13 @@ clap.workspace = true fm.workspace = true fxhash.workspace = true iter-extended.workspace = true -nargo.workspace = true +# This is the only crate that really needs the RPC feature, +# but enabling it here implicitly enables it for the whole +# workspace. A crate could opt out using `path` dependency, +# but it's only `noir_wasm` which couldn't compile with it, +# and that is a different target, and for that the feature +# aren't unified with this one. +nargo = { workspace = true, features = ["rpc"] } nargo_fmt.workspace = true nargo_toml.workspace = true noir_lsp.workspace = true @@ -65,9 +71,11 @@ notify-debouncer-full = "0.3.1" termion = "3.0.0" # Logs +tracing.workspace = true tracing-subscriber.workspace = true tracing-appender = "0.2.3" clap_complete = "4.5.36" +fs2 = "0.4.3" [target.'cfg(not(unix))'.dependencies] tokio-util = { version = "0.7.8", features = ["compat"] } @@ -79,7 +87,6 @@ dirs.workspace = true assert_cmd = "2.0.8" assert_fs = "1.0.10" predicates = "2.1.5" -file-lock = "2.1.11" fm.workspace = true criterion.workspace = true pprof.workspace = true @@ -88,11 +95,14 @@ proptest.workspace = true sha2.workspace = true sha3.workspace = true iai = "0.1.1" -test-binary = "3.0.2" test-case.workspace = true lazy_static.workspace = true light-poseidon = "0.2.0" +ark-bn254-v04 = { package = "ark-bn254", version = "^0.4.0", default-features = false, features = [ + "curve", +] } +ark-ff-v04 = { package = "ark-ff", version = "^0.4.0", default-features = false } [[bench]] name = "criterion" diff --git a/noir/noir-repo/tooling/nargo_cli/benches/criterion.rs b/noir/noir-repo/tooling/nargo_cli/benches/criterion.rs index 9bc50f87d8e..e43e498ea06 100644 --- a/noir/noir-repo/tooling/nargo_cli/benches/criterion.rs +++ b/noir/noir-repo/tooling/nargo_cli/benches/criterion.rs @@ -3,7 +3,6 @@ use acvm::{acir::native_types::WitnessMap, FieldElement}; use assert_cmd::prelude::{CommandCargoExt, OutputAssertExt}; use criterion::{criterion_group, criterion_main, Criterion}; -use nargo::PrintOutput; use noirc_abi::{ input_parser::{Format, InputValue}, Abi, InputMap, @@ -116,7 +115,7 @@ fn criterion_test_execution(c: &mut Criterion, test_program_dir: &Path, force_br let artifacts = RefCell::new(None); let mut foreign_call_executor = - nargo::foreign_calls::DefaultForeignCallExecutor::new(PrintOutput::None, None, None, None); + nargo::foreign_calls::DefaultForeignCallBuilder::default().build(); c.bench_function(&benchmark_name, |b| { b.iter_batched( @@ -142,10 +141,11 @@ fn criterion_test_execution(c: &mut Criterion, test_program_dir: &Path, force_br let artifacts = artifacts.as_ref().expect("setup compiled them"); for (program, initial_witness) in artifacts { + let solver = bn254_blackbox_solver::Bn254BlackBoxSolver::default(); let _witness_stack = black_box(nargo::ops::execute_program( black_box(&program.program), black_box(initial_witness.clone()), - &bn254_blackbox_solver::Bn254BlackBoxSolver, + &solver, &mut foreign_call_executor, )) .expect("failed to execute program"); diff --git a/noir/noir-repo/tooling/nargo_cli/build.rs b/noir/noir-repo/tooling/nargo_cli/build.rs index 8db2c1786d8..3b1aff88755 100644 --- a/noir/noir-repo/tooling/nargo_cli/build.rs +++ b/noir/noir-repo/tooling/nargo_cli/build.rs @@ -7,7 +7,7 @@ const GIT_COMMIT: &&str = &"GIT_COMMIT"; fn main() { // Only use build_data if the environment variable isn't set. - if std::env::var(GIT_COMMIT).is_err() { + if env::var(GIT_COMMIT).is_err() { build_data::set_GIT_COMMIT(); build_data::set_GIT_DIRTY(); build_data::no_debug_rebuilds(); @@ -19,9 +19,9 @@ fn main() { // Try to find the directory that Cargo sets when it is running; otherwise fallback to assuming the CWD // is the root of the repository and append the crate path - let root_dir = match std::env::var("CARGO_MANIFEST_DIR") { + let root_dir = match env::var("CARGO_MANIFEST_DIR") { Ok(dir) => PathBuf::from(dir).parent().unwrap().parent().unwrap().to_path_buf(), - Err(_) => std::env::current_dir().unwrap(), + Err(_) => env::current_dir().unwrap(), }; let test_dir = root_dir.join("test_programs"); @@ -177,37 +177,19 @@ fn generate_test_cases( } let test_cases = test_cases.join("\n"); - // We need to isolate test cases in the same group, otherwise they overwrite each other's artifacts. - // On CI we use `cargo nextest`, which runs tests in different processes; for this we use a file lock. - // Locally we might be using `cargo test`, which run tests in the same process; in this case the file lock - // wouldn't work, becuase the process itself has the lock, and it looks like it can have N instances without - // any problems; for this reason we also use a `Mutex`. - let mutex_name = format! {"TEST_MUTEX_{}", test_name.to_uppercase()}; write!( test_file, r#" -lazy_static::lazy_static! {{ - /// Prevent concurrent tests in the matrix from overwriting the compilation artifacts in {test_dir} - static ref {mutex_name}: std::sync::Mutex<()> = std::sync::Mutex::new(()); -}} - {test_cases} fn test_{test_name}(force_brillig: ForceBrillig, inliner_aggressiveness: Inliner) {{ let test_program_dir = PathBuf::from("{test_dir}"); - // Ignore poisoning errors if some of the matrix cases failed. - let mutex_guard = {mutex_name}.lock().unwrap_or_else(|e| e.into_inner()); - - let file_guard = file_lock::FileLock::lock( - test_program_dir.join("Nargo.toml"), - true, - file_lock::FileOptions::new().read(true).write(true).append(true) - ).expect("failed to lock Nargo.toml"); - let mut nargo = Command::cargo_bin("nargo").unwrap(); nargo.arg("--program-dir").arg(test_program_dir); nargo.arg("{test_command}").arg("--force"); nargo.arg("--inliner-aggressiveness").arg(inliner_aggressiveness.0.to_string()); + // Check whether the test case is non-deterministic + nargo.arg("--check-non-determinism"); if force_brillig.0 {{ nargo.arg("--force-brillig"); @@ -218,9 +200,6 @@ fn test_{test_name}(force_brillig: ForceBrillig, inliner_aggressiveness: Inliner }} {test_content} - - drop(file_guard); - drop(mutex_guard); }} "# ) @@ -380,7 +359,7 @@ fn generate_compile_success_empty_tests(test_file: &mut File, test_data_dir: &Pa panic!("JSON was not well-formatted {:?}\n\n{:?}", e, std::str::from_utf8(&output.stdout)) }}); let num_opcodes = &json["programs"][0]["functions"][0]["opcodes"]; - assert_eq!(num_opcodes.as_u64().expect("number of opcodes should fit in a u64"), 0); + assert_eq!(num_opcodes.as_u64().expect("number of opcodes should fit in a u64"), 0, "expected the number of opcodes to be 0"); "#; generate_test_cases( diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/compile_cmd.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/compile_cmd.rs index 2ecf6959a94..0af05703c9a 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/compile_cmd.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/compile_cmd.rs @@ -216,6 +216,20 @@ fn compile_programs( cached_program, )?; + if compile_options.check_non_determinism { + let (program_two, _) = compile_program( + file_manager, + parsed_files, + workspace, + package, + compile_options, + load_cached_program(package), + )?; + if fxhash::hash64(&program) != fxhash::hash64(&program_two) { + panic!("Non deterministic result compiling {}", package.name); + } + } + // Choose the target width for the final, backend specific transformation. let target_width = get_target_width(package.expression_width, compile_options.expression_width); @@ -319,7 +333,6 @@ mod tests { use nargo::ops::compile_program; use nargo_toml::PackageSelection; use noirc_driver::{CompileOptions, CrateName}; - use rayon::prelude::*; use crate::cli::compile_cmd::{get_target_width, parse_workspace, read_workspace}; @@ -388,7 +401,7 @@ mod tests { assert!(!test_workspaces.is_empty(), "should find some test workspaces"); - test_workspaces.par_iter().for_each(|workspace| { + test_workspaces.iter().for_each(|workspace| { let (file_manager, parsed_files) = parse_workspace(workspace); let binary_packages = workspace.into_iter().filter(|package| package.is_binary()); diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/dap_cmd.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/dap_cmd.rs index a84e961cfe7..7fbf685688a 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/dap_cmd.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/dap_cmd.rs @@ -50,6 +50,12 @@ pub(crate) struct DapCommand { #[clap(long)] preflight_skip_instrumentation: bool, + + /// Use pedantic ACVM solving, i.e. double-check some black-box function + /// assumptions when solving. + /// This is disabled by default. + #[arg(long, default_value = "false")] + pedantic_solving: bool, } fn parse_expression_width(input: &str) -> Result { @@ -137,6 +143,7 @@ fn load_and_compile_project( fn loop_uninitialized_dap( mut server: Server, expression_width: ExpressionWidth, + pedantic_solving: bool, ) -> Result<(), DapError> { loop { let req = match server.poll_request()? { @@ -197,7 +204,7 @@ fn loop_uninitialized_dap( noir_debugger::run_dap_loop( server, - &Bn254BlackBoxSolver, + &Bn254BlackBoxSolver(pedantic_solving), compiled_program, initial_witness, )?; @@ -269,5 +276,6 @@ pub(crate) fn run(args: DapCommand, _config: NargoConfig) -> Result<(), CliError let input = BufReader::new(std::io::stdin()); let server = Server::new(input, output); - loop_uninitialized_dap(server, args.expression_width).map_err(CliError::DapError) + loop_uninitialized_dap(server, args.expression_width, args.pedantic_solving) + .map_err(CliError::DapError) } diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/debug_cmd.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/debug_cmd.rs index f4dd607a53e..2ed30639d32 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/debug_cmd.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/debug_cmd.rs @@ -85,7 +85,14 @@ pub(crate) fn run(args: DebugCommand, config: NargoConfig) -> Result<(), CliErro let compiled_program = nargo::ops::transform_program(compiled_program, target_width); - run_async(package, compiled_program, &args.prover_name, &args.witness_name, target_dir) + run_async( + package, + compiled_program, + &args.prover_name, + &args.witness_name, + target_dir, + args.compile_options.pedantic_solving, + ) } pub(crate) fn compile_bin_package_for_debugging( @@ -172,6 +179,7 @@ fn run_async( prover_name: &str, witness_name: &Option, target_dir: &PathBuf, + pedantic_solving: bool, ) -> Result<(), CliError> { use tokio::runtime::Builder; let runtime = Builder::new_current_thread().enable_all().build().unwrap(); @@ -179,7 +187,7 @@ fn run_async( runtime.block_on(async { println!("[{}] Starting debugger", package.name); let (return_value, witness_stack) = - debug_program_and_decode(program, package, prover_name)?; + debug_program_and_decode(program, package, prover_name, pedantic_solving)?; if let Some(solved_witness_stack) = witness_stack { println!("[{}] Circuit witness successfully solved", package.name); @@ -206,12 +214,13 @@ fn debug_program_and_decode( program: CompiledProgram, package: &Package, prover_name: &str, + pedantic_solving: bool, ) -> Result<(Option, Option>), CliError> { // Parse the initial witness values from Prover.toml let (inputs_map, _) = read_inputs_from_file(&package.root_dir, prover_name, Format::Toml, &program.abi)?; let program_abi = program.abi.clone(); - let witness_stack = debug_program(program, &inputs_map)?; + let witness_stack = debug_program(program, &inputs_map, pedantic_solving)?; match witness_stack { Some(witness_stack) => { @@ -229,9 +238,14 @@ fn debug_program_and_decode( pub(crate) fn debug_program( compiled_program: CompiledProgram, inputs_map: &InputMap, + pedantic_solving: bool, ) -> Result>, CliError> { let initial_witness = compiled_program.abi.encode(inputs_map, None)?; - noir_debugger::run_repl_session(&Bn254BlackBoxSolver, compiled_program, initial_witness) - .map_err(CliError::from) + noir_debugger::run_repl_session( + &Bn254BlackBoxSolver(pedantic_solving), + compiled_program, + initial_witness, + ) + .map_err(CliError::from) } diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/execute_cmd.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/execute_cmd.rs index 49a23a7ea62..cb471995be5 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/execute_cmd.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/execute_cmd.rs @@ -7,7 +7,7 @@ use clap::Args; use nargo::constants::PROVER_INPUT_FILE; use nargo::errors::try_to_diagnose_runtime_error; -use nargo::foreign_calls::DefaultForeignCallExecutor; +use nargo::foreign_calls::DefaultForeignCallBuilder; use nargo::package::Package; use nargo::PrintOutput; use nargo_toml::{get_package_manifest, resolve_workspace_from_toml}; @@ -73,6 +73,7 @@ pub(crate) fn run(args: ExecuteCommand, config: NargoConfig) -> Result<(), CliEr args.oracle_resolver.as_deref(), Some(workspace.root_dir.clone()), Some(package.name.to_string()), + args.compile_options.pedantic_solving, )?; println!("[{}] Circuit witness successfully solved", package.name); @@ -108,12 +109,19 @@ fn execute_program_and_decode( foreign_call_resolver_url: Option<&str>, root_path: Option, package_name: Option, + pedantic_solving: bool, ) -> Result { // Parse the initial witness values from Prover.toml let (inputs_map, expected_return) = read_inputs_from_file(&package.root_dir, prover_name, Format::Toml, &program.abi)?; - let witness_stack = - execute_program(&program, &inputs_map, foreign_call_resolver_url, root_path, package_name)?; + let witness_stack = execute_program( + &program, + &inputs_map, + foreign_call_resolver_url, + root_path, + package_name, + pedantic_solving, + )?; // Get the entry point witness for the ABI let main_witness = &witness_stack.peek().expect("Should have at least one witness on the stack").witness; @@ -134,19 +142,22 @@ pub(crate) fn execute_program( foreign_call_resolver_url: Option<&str>, root_path: Option, package_name: Option, + pedantic_solving: bool, ) -> Result, CliError> { let initial_witness = compiled_program.abi.encode(inputs_map, None)?; let solved_witness_stack_err = nargo::ops::execute_program( &compiled_program.program, initial_witness, - &Bn254BlackBoxSolver, - &mut DefaultForeignCallExecutor::new( - PrintOutput::Stdout, - foreign_call_resolver_url, + &Bn254BlackBoxSolver(pedantic_solving), + &mut DefaultForeignCallBuilder { + output: PrintOutput::Stdout, + enable_mocks: false, + resolver_url: foreign_call_resolver_url.map(|s| s.to_string()), root_path, package_name, - ), + } + .build(), ); match solved_witness_stack_err { Ok(solved_witness_stack) => Ok(solved_witness_stack), @@ -162,7 +173,7 @@ pub(crate) fn execute_program( diagnostic.report(&debug_artifact, false); } - Err(crate::errors::CliError::NargoError(err)) + Err(CliError::NargoError(err)) } } } diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/fs/program.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/fs/program.rs index 0b30d23db2b..87783e4573a 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/fs/program.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/fs/program.rs @@ -36,6 +36,7 @@ fn save_build_artifact_to_file, T: ?Sized + serde::Serialize>( circuit_path } +#[tracing::instrument(level = "trace", skip_all)] pub(crate) fn read_program_from_file>( circuit_path: P, ) -> Result { diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/info_cmd.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/info_cmd.rs index ee8ff32922e..8d0fc257e1c 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/info_cmd.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/info_cmd.rs @@ -3,8 +3,7 @@ use bn254_blackbox_solver::Bn254BlackBoxSolver; use clap::Args; use iter_extended::vecmap; use nargo::{ - constants::PROVER_INPUT_FILE, foreign_calls::DefaultForeignCallExecutor, package::Package, - PrintOutput, + constants::PROVER_INPUT_FILE, foreign_calls::DefaultForeignCallBuilder, package::Package, }; use nargo_toml::{get_package_manifest, resolve_workspace_from_toml}; use noirc_abi::input_parser::Format; @@ -84,6 +83,7 @@ pub(crate) fn run(mut args: InfoCommand, config: NargoConfig) -> Result<(), CliE binary_packages, &args.prover_name, args.compile_options.expression_width, + args.compile_options.pedantic_solving, )? } else { binary_packages @@ -239,6 +239,7 @@ fn profile_brillig_execution( binary_packages: Vec<(Package, ProgramArtifact)>, prover_name: &str, expression_width: Option, + pedantic_solving: bool, ) -> Result, CliError> { let mut program_info = Vec::new(); for (package, program_artifact) in binary_packages.iter() { @@ -254,9 +255,16 @@ fn profile_brillig_execution( let (_, profiling_samples) = nargo::ops::execute_program_with_profiling( &program_artifact.bytecode, initial_witness, - &Bn254BlackBoxSolver, - &mut DefaultForeignCallExecutor::new(PrintOutput::None, None, None, None), - )?; + &Bn254BlackBoxSolver(pedantic_solving), + &mut DefaultForeignCallBuilder::default().build(), + ) + .map_err(|e| { + CliError::Generic(format!( + "failed to execute '{}': {}", + package.root_dir.to_string_lossy(), + e + )) + })?; let expression_width = get_target_width(package.expression_width, expression_width); diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/lsp_cmd.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/lsp_cmd.rs index bfaa913b33a..dc5d0995c06 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/lsp_cmd.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/lsp_cmd.rs @@ -25,7 +25,8 @@ pub(crate) fn run(_args: LspCommand, _config: NargoConfig) -> Result<(), CliErro runtime.block_on(async { let (server, _) = async_lsp::MainLoop::new_server(|client| { - let router = NargoLspService::new(&client, Bn254BlackBoxSolver); + let pedantic_solving = true; + let router = NargoLspService::new(&client, Bn254BlackBoxSolver(pedantic_solving)); ServiceBuilder::new() .layer(TracingLayer::default()) diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/mod.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/mod.rs index cc72092daa1..0b725afcf4e 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/mod.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/mod.rs @@ -116,7 +116,13 @@ enum NargoCommand { } #[cfg(not(feature = "codegen-docs"))] +#[tracing::instrument(level = "trace")] pub(crate) fn start_cli() -> eyre::Result<()> { + use std::fs::File; + + use fs2::FileExt as _; + use nargo_toml::get_package_manifest; + let NargoCli { command, mut config } = NargoCli::parse(); // If the provided `program_dir` is relative, make it absolute by joining it to the current directory. @@ -129,6 +135,28 @@ pub(crate) fn start_cli() -> eyre::Result<()> { config.program_dir = program_dir; } + let lock_file = match needs_lock(&command) { + Some(exclusive) => { + let toml_path = get_package_manifest(&config.program_dir)?; + let file = File::open(toml_path).expect("Expected Nargo.toml to exist"); + if exclusive { + if file.try_lock_exclusive().is_err() { + eprintln!("Waiting for lock on Nargo.toml..."); + } + + file.lock_exclusive().expect("Failed to lock Nargo.toml"); + } else { + if file.try_lock_shared().is_err() { + eprintln!("Waiting for lock on Nargo.toml..."); + } + + file.lock_shared().expect("Failed to lock Nargo.toml"); + } + Some(file) + } + None => None, + }; + match command { NargoCommand::New(args) => new_cmd::run(args, config), NargoCommand::Init(args) => init_cmd::run(args, config), @@ -145,6 +173,10 @@ pub(crate) fn start_cli() -> eyre::Result<()> { NargoCommand::GenerateCompletionScript(args) => generate_completion_script_cmd::run(args), }?; + if let Some(lock_file) = lock_file { + lock_file.unlock().expect("Failed to unlock Nargo.toml"); + } + Ok(()) } @@ -192,6 +224,23 @@ fn command_dir(cmd: &NargoCommand, program_dir: &Path) -> Result Ok(Some(nargo_toml::find_root(program_dir, workspace)?)) } +fn needs_lock(cmd: &NargoCommand) -> Option { + match cmd { + NargoCommand::Check(..) + | NargoCommand::Compile(..) + | NargoCommand::Execute(..) + | NargoCommand::Export(..) + | NargoCommand::Info(..) => Some(true), + NargoCommand::Debug(..) | NargoCommand::Test(..) => Some(false), + NargoCommand::Fmt(..) + | NargoCommand::New(..) + | NargoCommand::Init(..) + | NargoCommand::Lsp(..) + | NargoCommand::Dap(..) + | NargoCommand::GenerateCompletionScript(..) => None, + } +} + #[cfg(test)] mod tests { use clap::Parser; diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/test_cmd.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/test_cmd.rs index 1fd4ed2d873..9bf3ae9fedf 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/test_cmd.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/test_cmd.rs @@ -14,8 +14,9 @@ use clap::Args; use fm::FileManager; use formatters::{Formatter, JsonFormatter, PrettyFormatter, TerseFormatter}; use nargo::{ - insert_all_files_for_workspace_into_file_manager, ops::TestStatus, package::Package, parse_all, - prepare_package, workspace::Workspace, PrintOutput, + foreign_calls::DefaultForeignCallBuilder, insert_all_files_for_workspace_into_file_manager, + ops::TestStatus, package::Package, parse_all, prepare_package, workspace::Workspace, + PrintOutput, }; use nargo_toml::{get_package_manifest, resolve_workspace_from_toml}; use noirc_driver::{check_crate, CompileOptions, NOIR_ARTIFACT_VERSION_STRING}; @@ -494,10 +495,17 @@ impl<'a> TestRunner<'a> { &mut context, test_function, PrintOutput::String(&mut output_string), - foreign_call_resolver_url, - root_path, - Some(package_name), &self.args.compile_options, + |output, base| { + DefaultForeignCallBuilder { + output, + enable_mocks: true, + resolver_url: foreign_call_resolver_url.map(|s| s.to_string()), + root_path: root_path.clone(), + package_name: Some(package_name.clone()), + } + .build_with_base(base) + }, ); (test_status, output_string) } diff --git a/noir/noir-repo/tooling/nargo_cli/src/cli/test_cmd/formatters.rs b/noir/noir-repo/tooling/nargo_cli/src/cli/test_cmd/formatters.rs index 1b9b2d50378..75cf14ba120 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/cli/test_cmd/formatters.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/cli/test_cmd/formatters.rs @@ -417,6 +417,7 @@ impl Formatter for JsonFormatter { let mut json = Map::new(); json.insert("type".to_string(), json!("test")); json.insert("name".to_string(), json!(&test_result.name)); + json.insert("suite".to_string(), json!(&test_result.package_name)); json.insert("exec_time".to_string(), json!(test_result.time_to_run.as_secs_f64())); let mut stdout = String::new(); diff --git a/noir/noir-repo/tooling/nargo_cli/src/main.rs b/noir/noir-repo/tooling/nargo_cli/src/main.rs index a407d467ced..3ea167b7ffa 100644 --- a/noir/noir-repo/tooling/nargo_cli/src/main.rs +++ b/noir/noir-repo/tooling/nargo_cli/src/main.rs @@ -20,22 +20,7 @@ use tracing_subscriber::{fmt::format::FmtSpan, EnvFilter}; const PANIC_MESSAGE: &str = "This is a bug. We may have already fixed this in newer versions of Nargo so try searching for similar issues at https://github.com/noir-lang/noir/issues/.\nIf there isn't an open issue for this bug, consider opening one at https://github.com/noir-lang/noir/issues/new?labels=bug&template=bug_report.yml"; fn main() { - // Setup tracing - if let Ok(log_dir) = env::var("NARGO_LOG_DIR") { - let debug_file = rolling::daily(log_dir, "nargo-log"); - tracing_subscriber::fmt() - .with_span_events(FmtSpan::ENTER | FmtSpan::CLOSE) - .with_writer(debug_file) - .with_ansi(false) - .with_env_filter(EnvFilter::from_default_env()) - .init(); - } else { - tracing_subscriber::fmt() - .with_span_events(FmtSpan::ENTER | FmtSpan::CLOSE) - .with_ansi(true) - .with_env_filter(EnvFilter::from_env("NOIR_LOG")) - .init(); - } + setup_tracing(); // Register a panic hook to display more readable panic messages to end-users let (panic_hook, _) = @@ -47,3 +32,16 @@ fn main() { std::process::exit(1); } } + +fn setup_tracing() { + let subscriber = tracing_subscriber::fmt() + .with_span_events(FmtSpan::ENTER | FmtSpan::CLOSE) + .with_env_filter(EnvFilter::from_env("NOIR_LOG")); + + if let Ok(log_dir) = env::var("NARGO_LOG_DIR") { + let debug_file = rolling::daily(log_dir, "nargo-log"); + subscriber.with_writer(debug_file).with_ansi(false).json().init(); + } else { + subscriber.with_ansi(true).init(); + } +} diff --git a/noir/noir-repo/tooling/nargo_cli/tests/execute.rs b/noir/noir-repo/tooling/nargo_cli/tests/execute.rs index 561520c57a9..77d77cfd902 100644 --- a/noir/noir-repo/tooling/nargo_cli/tests/execute.rs +++ b/noir/noir-repo/tooling/nargo_cli/tests/execute.rs @@ -12,8 +12,6 @@ mod tests { use super::*; - test_binary::build_test_binary_once!(mock_backend, "../backend_interface/test-binaries"); - // Utilities to keep the test matrix labels more intuitive. #[derive(Debug, Clone, Copy)] struct ForceBrillig(pub bool); diff --git a/noir/noir-repo/tooling/nargo_cli/tests/stdlib-props.rs b/noir/noir-repo/tooling/nargo_cli/tests/stdlib-props.rs index a19408bd5fd..780b34bf0c3 100644 --- a/noir/noir-repo/tooling/nargo_cli/tests/stdlib-props.rs +++ b/noir/noir-repo/tooling/nargo_cli/tests/stdlib-props.rs @@ -2,9 +2,7 @@ use std::{cell::RefCell, collections::BTreeMap, path::Path}; use acvm::{acir::native_types::WitnessStack, AcirField, FieldElement}; use iter_extended::vecmap; -use nargo::{ - foreign_calls::DefaultForeignCallExecutor, ops::execute_program, parse_all, PrintOutput, -}; +use nargo::{foreign_calls::DefaultForeignCallBuilder, ops::execute_program, parse_all}; use noirc_abi::input_parser::InputValue; use noirc_driver::{ compile_main, file_manager_with_stdlib, prepare_crate, CompilationResult, CompileOptions, @@ -80,9 +78,9 @@ fn run_snippet_proptest( Err(e) => panic!("failed to compile program; brillig = {force_brillig}:\n{source}\n{e:?}"), }; - let blackbox_solver = bn254_blackbox_solver::Bn254BlackBoxSolver; - let foreign_call_executor = - RefCell::new(DefaultForeignCallExecutor::new(PrintOutput::None, None, None, None)); + let pedandic_solving = true; + let blackbox_solver = bn254_blackbox_solver::Bn254BlackBoxSolver(pedandic_solving); + let foreign_call_executor = RefCell::new(DefaultForeignCallBuilder::default().build()); // Generate multiple input/output proptest!(ProptestConfig::with_cases(100), |(io in strategy)| { @@ -202,7 +200,7 @@ fn fuzz_keccak256_equivalence() { }}" ) }, - |data| sha3::Keccak256::digest(data).try_into().unwrap(), + |data| sha3::Keccak256::digest(data).into(), ); } @@ -219,7 +217,7 @@ fn fuzz_keccak256_equivalence_over_135() { }}" ) }, - |data| sha3::Keccak256::digest(data).try_into().unwrap(), + |data| sha3::Keccak256::digest(data).into(), ); } @@ -235,7 +233,7 @@ fn fuzz_sha256_equivalence() { }}" ) }, - |data| sha2::Sha256::digest(data).try_into().unwrap(), + |data| sha2::Sha256::digest(data).into(), ); } @@ -251,7 +249,7 @@ fn fuzz_sha512_equivalence() { }}" ) }, - |data| sha2::Sha512::digest(data).try_into().unwrap(), + |data| sha2::Sha512::digest(data).into(), ); } @@ -292,13 +290,17 @@ fn fuzz_poseidon2_equivalence() { #[test] fn fuzz_poseidon_equivalence() { + use ark_ff_v04::{BigInteger, PrimeField}; use light_poseidon::{Poseidon, PoseidonHasher}; let poseidon_hash = |inputs: &[FieldElement]| { - let mut poseidon = Poseidon::::new_circom(inputs.len()).unwrap(); - let frs: Vec = inputs.iter().map(|f| f.into_repr()).collect::>(); + let mut poseidon = Poseidon::::new_circom(inputs.len()).unwrap(); + let frs: Vec = inputs + .iter() + .map(|f| ark_bn254_v04::Fr::from_be_bytes_mod_order(&f.to_be_bytes())) + .collect::>(); let hash = poseidon.hash(&frs).expect("failed to hash"); - FieldElement::from_repr(hash) + FieldElement::from_be_bytes_reduce(&hash.into_bigint().to_bytes_be()) }; // Noir has hashes up to length 16, but the reference library won't work with more than 12. diff --git a/noir/noir-repo/tooling/nargo_cli/tests/stdlib-tests.rs b/noir/noir-repo/tooling/nargo_cli/tests/stdlib-tests.rs index 29b871814b8..048de33f24c 100644 --- a/noir/noir-repo/tooling/nargo_cli/tests/stdlib-tests.rs +++ b/noir/noir-repo/tooling/nargo_cli/tests/stdlib-tests.rs @@ -2,6 +2,7 @@ #![allow(clippy::items_after_test_module)] use clap::Parser; use fm::FileManager; +use nargo::foreign_calls::DefaultForeignCallBuilder; use nargo::PrintOutput; use noirc_driver::{check_crate, file_manager_with_stdlib, CompileOptions}; use noirc_frontend::hir::FunctionNameMatch; @@ -83,15 +84,16 @@ fn run_stdlib_tests(force_brillig: bool, inliner_aggressiveness: i64) { let test_report: Vec<(String, TestStatus)> = test_functions .into_iter() .map(|(test_name, test_function)| { + let pedantic_solving = true; let status = run_test( - &bn254_blackbox_solver::Bn254BlackBoxSolver, + &bn254_blackbox_solver::Bn254BlackBoxSolver(pedantic_solving), &mut context, &test_function, PrintOutput::Stdout, - None, - Some(dummy_package.root_dir.clone()), - Some(dummy_package.name.to_string()), &CompileOptions { force_brillig, inliner_aggressiveness, ..Default::default() }, + |output, base| { + DefaultForeignCallBuilder::default().with_output(output).build_with_base(base) + }, ); (test_name, status) }) diff --git a/noir/noir-repo/tooling/nargo_fmt/build.rs b/noir/noir-repo/tooling/nargo_fmt/build.rs index bd2db5f5b18..a95cbe16525 100644 --- a/noir/noir-repo/tooling/nargo_fmt/build.rs +++ b/noir/noir-repo/tooling/nargo_fmt/build.rs @@ -1,10 +1,9 @@ use std::fs::File; use std::io::Write; use std::path::{Path, PathBuf}; -use std::{env, fs}; fn main() { - let out_dir = env::var("OUT_DIR").unwrap(); + let out_dir = std::env::var("OUT_DIR").unwrap(); let destination = Path::new(&out_dir).join("execute.rs"); let mut test_file = File::create(destination).unwrap(); @@ -24,7 +23,7 @@ fn generate_formatter_tests(test_file: &mut File, test_data_dir: &Path) { let outputs_dir = test_data_dir.join("expected"); let test_case_files = - fs::read_dir(inputs_dir).unwrap().flatten().filter(|c| c.path().is_file()); + std::fs::read_dir(inputs_dir).unwrap().flatten().filter(|c| c.path().is_file()); for file in test_case_files { let file_path = file.path(); diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter.rs index 9a9386e1911..2a8adb3fb28 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter.rs @@ -14,6 +14,7 @@ mod attribute; mod buffer; mod comments_and_whitespace; mod doc_comments; +mod enums; mod expression; mod function; mod generics; @@ -228,13 +229,11 @@ impl<'a> Formatter<'a> { /// Writes the current indentation to the buffer, but only if the buffer /// is empty or it ends with a newline (otherwise we'd be indenting when not needed). pub(crate) fn write_indentation(&mut self) { - if !(self.buffer.is_empty() || self.buffer.ends_with_newline()) { - return; - } - - for _ in 0..self.indentation { - for _ in 0..self.config.tab_spaces { - self.write(" "); + if self.buffer.is_empty() || self.buffer.ends_with_newline() { + for _ in 0..self.indentation { + for _ in 0..self.config.tab_spaces { + self.write(" "); + } } } } diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/enums.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/enums.rs new file mode 100644 index 00000000000..b596ec95c94 --- /dev/null +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/enums.rs @@ -0,0 +1,202 @@ +use noirc_frontend::{ + ast::NoirEnumeration, + token::{Keyword, Token}, +}; + +use super::Formatter; +use crate::chunks::ChunkGroup; + +impl<'a> Formatter<'a> { + pub(super) fn format_enum(&mut self, noir_enum: NoirEnumeration) { + self.format_secondary_attributes(noir_enum.attributes); + self.write_indentation(); + self.format_item_visibility(noir_enum.visibility); + self.write_keyword(Keyword::Enum); + self.write_space(); + self.write_identifier(noir_enum.name); + self.format_generics(noir_enum.generics); + self.skip_comments_and_whitespace(); + + // A case like `enum Foo;` + if self.is_at(Token::Semicolon) { + self.write_semicolon(); + return; + } + + // A case like `enum Foo { ... }` + self.write_space(); + self.write_left_brace(); + + if noir_enum.variants.is_empty() { + self.format_empty_block_contents(); + } else { + self.increase_indentation(); + self.write_line(); + + for (index, documented_variant) in noir_enum.variants.into_iter().enumerate() { + if index > 0 { + self.write_comma(); + self.write_line(); + } + + let doc_comments = documented_variant.doc_comments; + if !doc_comments.is_empty() { + self.format_outer_doc_comments(); + } + + let variant = documented_variant.item; + self.write_indentation(); + self.write_identifier(variant.name); + + if !variant.parameters.is_empty() { + self.write_token(Token::LeftParen); + for (i, parameter) in variant.parameters.into_iter().enumerate() { + if i != 0 { + self.write_comma(); + self.write_space(); + } + self.format_type(parameter); + } + self.write_token(Token::RightParen); + } else { + // Remove `()` from an empty `Variant()` + self.skip_comments_and_whitespace(); + if self.is_at(Token::LeftParen) { + self.bump(); + } + self.skip_comments_and_whitespace(); + if self.is_at(Token::RightParen) { + self.bump(); + } + } + } + + // Take the comment chunk so we can put it after a trailing comma we add, in case there's no comma + let mut group = ChunkGroup::new(); + let mut comments_and_whitespace_chunk = + self.chunk_formatter().skip_comments_and_whitespace_chunk(); + comments_and_whitespace_chunk.string = + comments_and_whitespace_chunk.string.trim_end().to_string(); + group.text(comments_and_whitespace_chunk); + + if self.is_at(Token::Comma) { + self.bump(); + } + self.write(","); + + self.format_chunk_group(group); + self.skip_comments_and_whitespace(); + + self.decrease_indentation(); + self.write_line(); + self.write_indentation(); + } + + self.write_right_brace(); + } +} + +#[cfg(test)] +mod tests { + use crate::assert_format; + + #[test] + fn format_empty_enum_with_generics() { + let src = " mod moo { enum Foo < A, B, let N : u32 > {} }"; + let expected = "mod moo { + enum Foo {} +} +"; + assert_format(src, expected); + } + + #[test] + fn format_enum_with_variants() { + let src = " mod moo { enum Foo { +// hello +/// comment + Variant ( Field , i32 ) , + // comment + Another ( ), + } }"; + let expected = "mod moo { + enum Foo { + // hello + /// comment + Variant(Field, i32), + // comment + Another, + } +} +"; + assert_format(src, expected); + } + + #[test] + fn format_enum_with_multiple_newlines() { + let src = " mod moo { + + + enum Foo { + + +X( Field) , + + +Y ( Field ) + + +} + + +}"; + let expected = "mod moo { + + enum Foo { + + X(Field), + + Y(Field), + } + +} +"; + assert_format(src, expected); + } + + #[test] + fn format_two_enums() { + let src = " enum Foo { } enum Bar {} + "; + let expected = "enum Foo {} +enum Bar {} +"; + assert_format(src, expected); + } + + #[test] + fn format_enum_field_without_trailing_comma_but_comment() { + let src = "enum Foo { + field(Field) // comment + }"; + let expected = "enum Foo { + field(Field), // comment +} +"; + assert_format(src, expected); + } + + #[test] + fn format_comment_after_last_enum_field() { + let src = "enum Foo { + field(Field) + /* comment */ + }"; + let expected = "enum Foo { + field(Field), + /* comment */ +} +"; + assert_format(src, expected); + } +} diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/expression.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/expression.rs index ecc9fab18ce..ef04276a605 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter/expression.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/expression.rs @@ -370,6 +370,7 @@ impl<'a, 'b> ChunkFormatter<'a, 'b> { ) -> ChunkGroup { let mut group = ChunkGroup::new(); group.text(self.chunk(|formatter| { + formatter.format_outer_doc_comments(); formatter.write_keyword(Keyword::Unsafe); formatter.write_space(); })); @@ -1910,14 +1911,16 @@ global y = 1; #[test] fn format_unsafe_one_expression() { - let src = "global x = unsafe { 1 } ;"; + let src = "global x = unsafe { + 1 } ;"; let expected = "global x = unsafe { 1 };\n"; assert_format(src, expected); } #[test] fn format_unsafe_two_expressions() { - let src = "global x = unsafe { 1; 2 } ;"; + let src = "global x = unsafe { + 1; 2 } ;"; let expected = "global x = unsafe { 1; 2 @@ -1926,6 +1929,21 @@ global y = 1; assert_format(src, expected); } + #[test] + fn format_unsafe_with_doc_comment() { + let src = "fn foo() { + /// Comment + unsafe { 1 } }"; + let expected = "fn foo() { + /// Comment + unsafe { + 1 + } +} +"; + assert_format(src, expected); + } + #[test] fn format_comptime_one_expression() { let src = "global x = comptime { 1 } ;"; diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/function.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/function.rs index 8207db5e486..ca905f3dcf8 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter/function.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/function.rs @@ -19,10 +19,11 @@ pub(super) struct FunctionToFormat { pub(super) return_visibility: Visibility, pub(super) where_clause: Vec, pub(super) body: Option, + pub(super) skip_visibility: bool, } impl<'a> Formatter<'a> { - pub(super) fn format_function(&mut self, func: NoirFunction) { + pub(super) fn format_function(&mut self, func: NoirFunction, skip_visibility: bool) { self.format_function_impl(FunctionToFormat { attributes: func.def.attributes, visibility: func.def.visibility, @@ -33,6 +34,7 @@ impl<'a> Formatter<'a> { return_visibility: func.def.return_visibility, where_clause: func.def.where_clause, body: Some(func.def.body), + skip_visibility, }); } @@ -41,7 +43,7 @@ impl<'a> Formatter<'a> { self.format_attributes(func.attributes); self.write_indentation(); - self.format_function_modifiers(func.visibility); + self.format_function_modifiers(func.visibility, func.skip_visibility); self.write_keyword(Keyword::Fn); self.write_space(); self.write_identifier(func.name); @@ -94,7 +96,11 @@ impl<'a> Formatter<'a> { } } - pub(super) fn format_function_modifiers(&mut self, visibility: ItemVisibility) { + pub(super) fn format_function_modifiers( + &mut self, + visibility: ItemVisibility, + skip_visibility: bool, + ) { // For backwards compatibility, unconstrained might come before visibility. // We'll remember this but put it after the visibility. let unconstrained = if self.is_at_keyword(Keyword::Unconstrained) { @@ -105,7 +111,14 @@ impl<'a> Formatter<'a> { false }; - self.format_item_visibility(visibility); + if skip_visibility { + // The intention here is to format the visibility into a temporary buffer that is discarded + self.chunk_formatter().chunk(|formatter| { + formatter.format_item_visibility(visibility); + }); + } else { + self.format_item_visibility(visibility); + } if unconstrained { self.write("unconstrained "); diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/impls.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/impls.rs index 1c2c25c9200..71548dd5efa 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter/impls.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/impls.rs @@ -38,7 +38,9 @@ impl<'a> Formatter<'a> { if !doc_comments.is_empty() { self.format_outer_doc_comments(); } - self.format_function(method); + self.format_function( + method, false, // skip visibility + ); } self.skip_comments_and_whitespace(); diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/item.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/item.rs index 521e476fe71..499acb8415c 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter/item.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/item.rs @@ -58,8 +58,12 @@ impl<'a> Formatter<'a> { ItemKind::Import(use_tree, item_visibility) => { self.format_import(use_tree, item_visibility); } - ItemKind::Function(noir_function) => self.format_function(noir_function), + ItemKind::Function(noir_function) => self.format_function( + noir_function, + false, // skip visibility + ), ItemKind::Struct(noir_struct) => self.format_struct(noir_struct), + ItemKind::Enum(noir_enum) => self.format_enum(noir_enum), ItemKind::Trait(noir_trait) => self.format_trait(noir_trait), ItemKind::TraitImpl(noir_trait_impl) => self.format_trait_impl(noir_trait_impl), ItemKind::Impl(type_impl) => self.format_impl(type_impl), diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/statement.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/statement.rs index 50c286ff161..983fb00db16 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter/statement.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/statement.rs @@ -4,7 +4,7 @@ use noirc_frontend::{ ForLoopStatement, ForRange, LetStatement, Pattern, Statement, StatementKind, UnresolvedType, UnresolvedTypeData, }, - token::{Keyword, SecondaryAttribute, Token}, + token::{Keyword, SecondaryAttribute, Token, TokenKind}, }; use crate::chunks::{ChunkFormatter, ChunkGroup, GroupKind}; @@ -23,7 +23,17 @@ impl<'a, 'b> ChunkFormatter<'a, 'b> { // Now write any leading comment respecting multiple newlines after them group.leading_comment(self.chunk(|formatter| { + // Doc comments for a let statement could come before a potential non-doc comment + if formatter.token.kind() == TokenKind::OuterDocComment { + formatter.format_outer_doc_comments(); + } + formatter.skip_comments_and_whitespace_writing_multiple_lines_if_found(); + + // Or doc comments could come after a potential non-doc comment + if formatter.token.kind() == TokenKind::OuterDocComment { + formatter.format_outer_doc_comments(); + } })); ignore_next |= self.ignore_next; @@ -65,6 +75,9 @@ impl<'a, 'b> ChunkFormatter<'a, 'b> { StatementKind::For(for_loop_statement) => { group.group(self.format_for_loop(for_loop_statement)); } + StatementKind::Loop(block) => { + group.group(self.format_loop(block)); + } StatementKind::Break => { group.text(self.chunk(|formatter| { formatter.write_keyword(Keyword::Break); @@ -267,6 +280,34 @@ impl<'a, 'b> ChunkFormatter<'a, 'b> { group } + fn format_loop(&mut self, block: Expression) -> ChunkGroup { + let mut group = ChunkGroup::new(); + + group.text(self.chunk(|formatter| { + formatter.write_keyword(Keyword::Loop); + })); + + group.space(self); + + let ExpressionKind::Block(block) = block.kind else { + panic!("Expected a block expression for loop body"); + }; + + group.group(self.format_block_expression( + block, true, // force multiple lines + )); + + // If there's a trailing semicolon, remove it + group.text(self.chunk(|formatter| { + formatter.skip_whitespace_if_it_is_not_a_newline(); + if formatter.is_at(Token::Semicolon) { + formatter.bump(); + } + })); + + group + } + fn format_comptime_statement(&mut self, statement: Statement) -> ChunkGroup { let mut group = ChunkGroup::new(); @@ -374,6 +415,34 @@ mod tests { assert_format(src, expected); } + #[test] + fn format_let_statement_with_unsafe() { + let src = " fn foo() { + /// Safety: some doc + let x = unsafe { 1 } ; } "; + let expected = "fn foo() { + /// Safety: some doc + let x = unsafe { 1 }; +} +"; + assert_format(src, expected); + } + + #[test] + fn format_let_statement_with_unsafe_and_comment_before_it() { + let src = " fn foo() { + // Some comment + /// Safety: some doc + let x = unsafe { 1 } ; } "; + let expected = "fn foo() { + // Some comment + /// Safety: some doc + let x = unsafe { 1 }; +} +"; + assert_format(src, expected); + } + #[test] fn format_assign() { let src = " fn foo() { x = 2 ; } "; @@ -489,7 +558,8 @@ mod tests { #[test] fn format_unsafe_statement() { - let src = " fn foo() { unsafe { 1 } } "; + let src = " fn foo() { unsafe { + 1 } } "; let expected = "fn foo() { unsafe { 1 @@ -710,4 +780,27 @@ mod tests { let expected = src; assert_format_with_max_width(src, expected, " let x = 123456;".len()); } + + #[test] + fn format_empty_loop() { + let src = " fn foo() { loop { } } "; + let expected = "fn foo() { + loop {} +} +"; + assert_format(src, expected); + } + + #[test] + fn format_non_empty_loop() { + let src = " fn foo() { loop { 1 ; 2 } } "; + let expected = "fn foo() { + loop { + 1; + 2 + } +} +"; + assert_format(src, expected); + } } diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/trait_impl.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/trait_impl.rs index b31da8a4101..5bb9a0d0025 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter/trait_impl.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/trait_impl.rs @@ -1,8 +1,5 @@ use noirc_frontend::{ - ast::{ - FunctionDefinition, ItemVisibility, NoirFunction, NoirTraitImpl, Pattern, TraitImplItem, - TraitImplItemKind, - }, + ast::{NoirTraitImpl, Pattern, TraitImplItem, TraitImplItemKind}, token::{Keyword, Token}, }; @@ -69,12 +66,10 @@ impl<'a> Formatter<'a> { fn format_trait_impl_item(&mut self, item: TraitImplItem) { match item.kind { TraitImplItemKind::Function(noir_function) => { - // Trait impl functions are public, but there's no `pub` keyword in the source code, - // so to format it we pass a private one. - let def = - FunctionDefinition { visibility: ItemVisibility::Private, ..noir_function.def }; - let noir_function = NoirFunction { def, ..noir_function }; - self.format_function(noir_function); + self.format_function( + noir_function, + true, // skip visibility + ); } TraitImplItemKind::Constant(name, typ, value) => { let pattern = Pattern::Identifier(name); @@ -179,6 +174,22 @@ fn foo ( ) { } assert_format(src, expected); } + #[test] + fn format_trait_impl_function_with_visibility() { + let src = " mod moo { impl Foo for Bar { + /// Some doc comment +pub fn foo ( ) { } + } }"; + let expected = "mod moo { + impl Foo for Bar { + /// Some doc comment + fn foo() {} + } +} +"; + assert_format(src, expected); + } + #[test] fn format_trait_impl_constant_without_type() { let src = " mod moo { impl Foo for Bar { diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/traits.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/traits.rs index 1f192be471e..175dcad6170 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter/traits.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/traits.rs @@ -113,6 +113,7 @@ impl<'a> Formatter<'a> { return_visibility: Visibility::Private, where_clause, body, + skip_visibility: true, }; self.format_function_impl(func); } @@ -236,12 +237,12 @@ mod tests { fn format_trait_with_function_without_body() { let src = " mod moo { trait Foo { /// hello - pub fn foo ( ); + fn foo ( ); } }"; let expected = "mod moo { trait Foo { /// hello - pub fn foo(); + fn foo(); } } "; @@ -252,12 +253,12 @@ mod tests { fn format_trait_with_function_with_body() { let src = " mod moo { trait Foo { /// hello - pub fn foo ( ) { 1 } + fn foo ( ) { 1 } } }"; let expected = "mod moo { trait Foo { /// hello - pub fn foo() { + fn foo() { 1 } } @@ -270,12 +271,12 @@ mod tests { fn format_trait_with_function_with_params() { let src = " mod moo { trait Foo { /// hello - pub fn foo ( x : i32 , y : Field ); + fn foo ( x : i32 , y : Field ); } }"; let expected = "mod moo { trait Foo { /// hello - pub fn foo(x: i32, y: Field); + fn foo(x: i32, y: Field); } } "; @@ -298,6 +299,24 @@ mod tests { assert_format(src, expected); } + #[test] + fn format_trait_with_function_with_visibility() { + let src = " mod moo { trait Foo { + /// hello + pub fn foo ( ) { 1 } + } }"; + let expected = "mod moo { + trait Foo { + /// hello + fn foo() { + 1 + } + } +} +"; + assert_format(src, expected); + } + #[test] fn format_multiple_traits() { let src = " trait Foo {} diff --git a/noir/noir-repo/tooling/nargo_fmt/src/formatter/use_tree_merge.rs b/noir/noir-repo/tooling/nargo_fmt/src/formatter/use_tree_merge.rs index 834280ddba3..a679e026435 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/formatter/use_tree_merge.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/formatter/use_tree_merge.rs @@ -1,8 +1,4 @@ -use std::{ - cmp::Ordering, - collections::BTreeMap, - fmt::{self, Display}, -}; +use std::{cmp::Ordering, collections::BTreeMap, fmt::Display}; use noirc_frontend::ast::{ItemVisibility, PathKind, UseTree, UseTreeKind}; @@ -157,7 +153,7 @@ impl Segment { } impl Display for Segment { - fn fmt(&self, f: &mut std::fmt::Formatter) -> fmt::Result { + fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result { match self { Segment::Crate => write!(f, "crate"), Segment::Super => write!(f, "super"), @@ -183,7 +179,12 @@ impl Ord for Segment { if let (Segment::Plain(self_string), Segment::Plain(other_string)) = (self, other) { // Case-insensitive comparison for plain segments - self_string.to_lowercase().cmp(&other_string.to_lowercase()) + let ordering = self_string.to_lowercase().cmp(&other_string.to_lowercase()); + if ordering == Ordering::Equal { + self_string.cmp(other_string) + } else { + ordering + } } else { order_number_ordering } @@ -624,4 +625,10 @@ use std::merkle::compute_merkle_root; let expected = "use std::{as_witness, merkle::compute_merkle_root};\n"; assert_format(src, expected); } + + #[test] + fn does_not_merge_same_identifiers_if_equal_case_insensitive() { + let src = "use bigint::{BigNum, bignum::BigNumTrait};\n"; + assert_format(src, src); + } } diff --git a/noir/noir-repo/tooling/nargo_fmt/src/lib.rs b/noir/noir-repo/tooling/nargo_fmt/src/lib.rs index eda77e78c7c..05bfb0e7d57 100644 --- a/noir/noir-repo/tooling/nargo_fmt/src/lib.rs +++ b/noir/noir-repo/tooling/nargo_fmt/src/lib.rs @@ -66,6 +66,7 @@ pub(crate) fn assert_format_with_config(src: &str, expected: &str, config: Confi use noirc_frontend::parser; let (parsed_module, errors) = parser::parse_program(src); + let errors: Vec<_> = errors.into_iter().filter(|error| !error.is_warning()).collect(); if !errors.is_empty() { panic!("Expected no errors, got: {:?}", errors); } @@ -78,6 +79,7 @@ pub(crate) fn assert_format_with_config(src: &str, expected: &str, config: Confi let src = &result; let (parsed_module, errors) = parser::parse_program(src); + let errors: Vec<_> = errors.into_iter().filter(|error| !error.is_warning()).collect(); if !errors.is_empty() { panic!("Expected no errors in idempotent check, got: {:?}", errors); } diff --git a/noir/noir-repo/tooling/nargo_fmt/tests/expected/unsafe.nr b/noir/noir-repo/tooling/nargo_fmt/tests/expected/unsafe.nr index 7d733c203de..0cf587ea549 100644 --- a/noir/noir-repo/tooling/nargo_fmt/tests/expected/unsafe.nr +++ b/noir/noir-repo/tooling/nargo_fmt/tests/expected/unsafe.nr @@ -1,6 +1,8 @@ fn main(x: pub u8, y: u8) { + /// Safety: testing unsafe {} + /// Safety: testing unsafe { assert_eq(x, y); } diff --git a/noir/noir-repo/tooling/nargo_fmt/tests/input/unsafe.nr b/noir/noir-repo/tooling/nargo_fmt/tests/input/unsafe.nr index 6e12ef975ee..0aeffd0b983 100644 --- a/noir/noir-repo/tooling/nargo_fmt/tests/input/unsafe.nr +++ b/noir/noir-repo/tooling/nargo_fmt/tests/input/unsafe.nr @@ -1,6 +1,9 @@ fn main(x: pub u8, y: u8) { - unsafe { } + /// Safety: testing + unsafe { + } + /// Safety: testing unsafe { assert_eq(x, y); } diff --git a/noir/noir-repo/tooling/nargo_toml/src/lib.rs b/noir/noir-repo/tooling/nargo_toml/src/lib.rs index b5c45977618..fa556d445a4 100644 --- a/noir/noir-repo/tooling/nargo_toml/src/lib.rs +++ b/noir/noir-repo/tooling/nargo_toml/src/lib.rs @@ -93,7 +93,7 @@ pub fn find_package_root(current_path: &Path) -> Result /// * `C:\foo\bar` -> `C:\foo` /// * `//shared/foo/bar` -> `//shared/foo` /// * `/foo` -> `/foo` -/// otherwise empty path. +/// otherwise empty path. fn path_root(path: &Path) -> PathBuf { let mut components = path.components(); diff --git a/noir/noir-repo/tooling/noir_codegen/package.json b/noir/noir-repo/tooling/noir_codegen/package.json index c96ecd22230..351a29ae595 100644 --- a/noir/noir-repo/tooling/noir_codegen/package.json +++ b/noir/noir-repo/tooling/noir_codegen/package.json @@ -3,7 +3,7 @@ "contributors": [ "The Noir Team " ], - "version": "1.0.0-beta.0", + "version": "1.0.0-beta.1", "packageManager": "yarn@3.5.1", "license": "(MIT OR Apache-2.0)", "type": "module", diff --git a/noir/noir-repo/tooling/noir_js/package.json b/noir/noir-repo/tooling/noir_js/package.json index 0f0e111c30b..2a6dc614cb0 100644 --- a/noir/noir-repo/tooling/noir_js/package.json +++ b/noir/noir-repo/tooling/noir_js/package.json @@ -3,7 +3,7 @@ "contributors": [ "The Noir Team " ], - "version": "1.0.0-beta.0", + "version": "1.0.0-beta.1", "packageManager": "yarn@3.5.1", "license": "(MIT OR Apache-2.0)", "type": "module", diff --git a/noir/noir-repo/tooling/noir_js_types/package.json b/noir/noir-repo/tooling/noir_js_types/package.json index 17e9efc7678..057ff29ef25 100644 --- a/noir/noir-repo/tooling/noir_js_types/package.json +++ b/noir/noir-repo/tooling/noir_js_types/package.json @@ -4,7 +4,7 @@ "The Noir Team " ], "packageManager": "yarn@3.5.1", - "version": "1.0.0-beta.0", + "version": "1.0.0-beta.1", "license": "(MIT OR Apache-2.0)", "homepage": "https://noir-lang.org/", "repository": { diff --git a/noir/noir-repo/tooling/noirc_abi/src/lib.rs b/noir/noir-repo/tooling/noirc_abi/src/lib.rs index bd5674d64f1..5f5f3748bc4 100644 --- a/noir/noir-repo/tooling/noirc_abi/src/lib.rs +++ b/noir/noir-repo/tooling/noirc_abi/src/lib.rs @@ -13,10 +13,7 @@ use acvm::{ use errors::AbiError; use input_parser::InputValue; use iter_extended::{try_btree_map, try_vecmap, vecmap}; -use noirc_printable_type::{ - decode_value as printable_type_decode_value, PrintableType, PrintableValue, - PrintableValueDisplay, -}; +use noirc_printable_type::{PrintableType, PrintableValue, PrintableValueDisplay}; use serde::{Deserialize, Serialize}; use std::borrow::Borrow; use std::{collections::BTreeMap, str}; @@ -30,8 +27,11 @@ mod arbitrary; pub mod errors; pub mod input_parser; +mod printable_type; mod serialization; +pub use printable_type::decode_value as decode_printable_value; + /// A map from the fields in an TOML/JSON file which correspond to some ABI to their values pub type InputMap = BTreeMap; @@ -417,7 +417,7 @@ pub fn decode_value( Ok(value) } -fn decode_string_value(field_elements: &[FieldElement]) -> String { +pub fn decode_string_value(field_elements: &[F]) -> String { let string_as_slice = vecmap(field_elements, |e| { let mut field_as_bytes = e.to_be_bytes(); let char_byte = field_as_bytes.pop().unwrap(); // A character in a string is represented by a u8, thus we just want the last byte of the element @@ -476,21 +476,21 @@ pub fn display_abi_error( AbiErrorType::FmtString { length, item_types } => { let mut fields_iter = fields.iter().copied(); let PrintableValue::String(string) = - printable_type_decode_value(&mut fields_iter, &PrintableType::String { length }) + decode_printable_value(&mut fields_iter, &PrintableType::String { length }) else { unreachable!("Got non-string from string decoding"); }; let _length_of_items = fields_iter.next(); let items = item_types.into_iter().map(|abi_type| { let printable_typ = (&abi_type).into(); - let decoded = printable_type_decode_value(&mut fields_iter, &printable_typ); + let decoded = decode_printable_value(&mut fields_iter, &printable_typ); (decoded, printable_typ) }); PrintableValueDisplay::FmtString(string, items.collect()) } AbiErrorType::Custom(abi_typ) => { let printable_type = (&abi_typ).into(); - let decoded = printable_type_decode_value(&mut fields.iter().copied(), &printable_type); + let decoded = decode_printable_value(&mut fields.iter().copied(), &printable_type); PrintableValueDisplay::Plain(decoded, printable_type) } AbiErrorType::String { string } => { diff --git a/noir/noir-repo/tooling/noirc_abi/src/printable_type.rs b/noir/noir-repo/tooling/noirc_abi/src/printable_type.rs new file mode 100644 index 00000000000..a81eb0ce8f6 --- /dev/null +++ b/noir/noir-repo/tooling/noirc_abi/src/printable_type.rs @@ -0,0 +1,78 @@ +use std::collections::BTreeMap; + +use acvm::acir::AcirField; +use iter_extended::vecmap; + +use noirc_printable_type::{PrintableType, PrintableValue}; + +use crate::decode_string_value; + +/// Assumes that `field_iterator` contains enough field elements in order to decode the [PrintableType] +pub fn decode_value( + field_iterator: &mut impl Iterator, + typ: &PrintableType, +) -> PrintableValue { + match typ { + PrintableType::Field + | PrintableType::SignedInteger { .. } + | PrintableType::UnsignedInteger { .. } + | PrintableType::Boolean => { + let field_element = field_iterator.next().unwrap(); + + PrintableValue::Field(field_element) + } + PrintableType::Array { length, typ } => { + let length = *length as usize; + let mut array_elements = Vec::with_capacity(length); + for _ in 0..length { + array_elements.push(decode_value(field_iterator, typ)); + } + + PrintableValue::Vec { array_elements, is_slice: false } + } + PrintableType::Slice { typ } => { + let length = field_iterator + .next() + .expect("not enough data to decode variable array length") + .to_u128() as usize; + let mut array_elements = Vec::with_capacity(length); + for _ in 0..length { + array_elements.push(decode_value(field_iterator, typ)); + } + + PrintableValue::Vec { array_elements, is_slice: true } + } + PrintableType::Tuple { types } => PrintableValue::Vec { + array_elements: vecmap(types, |typ| decode_value(field_iterator, typ)), + is_slice: false, + }, + PrintableType::String { length } => { + let field_elements: Vec = field_iterator.take(*length as usize).collect(); + + PrintableValue::String(decode_string_value(&field_elements)) + } + PrintableType::Struct { fields, .. } => { + let mut struct_map = BTreeMap::new(); + + for (field_key, param_type) in fields { + let field_value = decode_value(field_iterator, param_type); + + struct_map.insert(field_key.to_owned(), field_value); + } + + PrintableValue::Struct(struct_map) + } + PrintableType::Function { env, .. } => { + let field_element = field_iterator.next().unwrap(); + let func_ref = PrintableValue::Field(field_element); + // we want to consume the fields from the environment, but for now they are not actually printed + decode_value(field_iterator, env); + func_ref + } + PrintableType::MutableReference { typ } => { + // we decode the reference, but it's not really used for printing + decode_value(field_iterator, typ) + } + PrintableType::Unit => PrintableValue::Field(F::zero()), + } +} diff --git a/noir/noir-repo/tooling/noirc_abi_wasm/build.sh b/noir/noir-repo/tooling/noirc_abi_wasm/build.sh index c07d2d8a4c1..16fb26e55db 100755 --- a/noir/noir-repo/tooling/noirc_abi_wasm/build.sh +++ b/noir/noir-repo/tooling/noirc_abi_wasm/build.sh @@ -25,7 +25,7 @@ function run_if_available { require_command jq require_command cargo require_command wasm-bindgen -#require_command wasm-opt +require_command wasm-opt self_path=$(dirname "$(readlink -f "$0")") pname=$(cargo read-manifest | jq -r '.name') diff --git a/noir/noir-repo/tooling/noirc_abi_wasm/package.json b/noir/noir-repo/tooling/noirc_abi_wasm/package.json index 9194714454d..0148a9343db 100644 --- a/noir/noir-repo/tooling/noirc_abi_wasm/package.json +++ b/noir/noir-repo/tooling/noirc_abi_wasm/package.json @@ -3,7 +3,7 @@ "contributors": [ "The Noir Team " ], - "version": "1.0.0-beta.0", + "version": "1.0.0-beta.1", "license": "(MIT OR Apache-2.0)", "homepage": "https://noir-lang.org/", "repository": { diff --git a/noir/noir-repo/tooling/noirc_artifacts/src/debug_vars.rs b/noir/noir-repo/tooling/noirc_artifacts/src/debug_vars.rs index aa9328432b8..8efeeebb3aa 100644 --- a/noir/noir-repo/tooling/noirc_artifacts/src/debug_vars.rs +++ b/noir/noir-repo/tooling/noirc_artifacts/src/debug_vars.rs @@ -1,8 +1,9 @@ use acvm::AcirField; +use noirc_abi::decode_printable_value; use noirc_errors::debug_info::{ DebugFnId, DebugFunction, DebugInfo, DebugTypeId, DebugVarId, DebugVariable, }; -use noirc_printable_type::{decode_value, PrintableType, PrintableValue}; +use noirc_printable_type::{PrintableType, PrintableValue}; use std::collections::HashMap; #[derive(Debug, Default, Clone)] @@ -45,7 +46,7 @@ impl DebugVars { &'a self, fn_id: &DebugFnId, frame: &'a HashMap>, - ) -> StackFrame { + ) -> StackFrame<'a, F> { let debug_fn = &self.functions.get(fn_id).expect("failed to find function metadata"); let params: Vec<&str> = @@ -72,7 +73,7 @@ impl DebugVars { .last_mut() .expect("unexpected empty stack frames") .1 - .insert(var_id, decode_value(&mut values.iter().copied(), ptype)); + .insert(var_id, decode_printable_value(&mut values.iter().copied(), ptype)); } pub fn assign_field(&mut self, var_id: DebugVarId, indexes: Vec, values: &[F]) { @@ -143,7 +144,7 @@ impl DebugVars { } }; } - *cursor = decode_value(&mut values.iter().copied(), cursor_type); + *cursor = decode_printable_value(&mut values.iter().copied(), cursor_type); } pub fn assign_deref(&mut self, _var_id: DebugVarId, _values: &[F]) { diff --git a/noir/noir-repo/tooling/profiler/src/cli/execution_flamegraph_cmd.rs b/noir/noir-repo/tooling/profiler/src/cli/execution_flamegraph_cmd.rs index 76b23ebf739..dbb2a2c38bc 100644 --- a/noir/noir-repo/tooling/profiler/src/cli/execution_flamegraph_cmd.rs +++ b/noir/noir-repo/tooling/profiler/src/cli/execution_flamegraph_cmd.rs @@ -3,13 +3,13 @@ use std::path::{Path, PathBuf}; use acir::circuit::OpcodeLocation; use clap::Args; use color_eyre::eyre::{self, Context}; +use nargo::foreign_calls::DefaultForeignCallBuilder; use nargo::PrintOutput; use crate::flamegraph::{BrilligExecutionSample, FlamegraphGenerator, InfernoFlamegraphGenerator}; use crate::fs::{read_inputs_from_file, read_program_from_file}; use crate::opcode_formatter::format_brillig_opcode; use bn254_blackbox_solver::Bn254BlackBoxSolver; -use nargo::foreign_calls::DefaultForeignCallExecutor; use noirc_abi::input_parser::Format; use noirc_artifacts::debug::DebugArtifact; @@ -26,6 +26,12 @@ pub(crate) struct ExecutionFlamegraphCommand { /// The output folder for the flamegraph svg files #[clap(long, short)] output: PathBuf, + + /// Use pedantic ACVM solving, i.e. double-check some black-box function + /// assumptions when solving. + /// This is disabled by default. + #[clap(long, default_value = "false")] + pedantic_solving: bool, } pub(crate) fn run(args: ExecutionFlamegraphCommand) -> eyre::Result<()> { @@ -34,6 +40,7 @@ pub(crate) fn run(args: ExecutionFlamegraphCommand) -> eyre::Result<()> { &args.prover_toml_path, &InfernoFlamegraphGenerator { count_name: "samples".to_string() }, &args.output, + args.pedantic_solving, ) } @@ -42,6 +49,7 @@ fn run_with_generator( prover_toml_path: &Path, flamegraph_generator: &impl FlamegraphGenerator, output_path: &Path, + pedantic_solving: bool, ) -> eyre::Result<()> { let program = read_program_from_file(artifact_path).context("Error reading program from file")?; @@ -54,8 +62,8 @@ fn run_with_generator( let (_, mut profiling_samples) = nargo::ops::execute_program_with_profiling( &program.bytecode, initial_witness, - &Bn254BlackBoxSolver, - &mut DefaultForeignCallExecutor::new(PrintOutput::Stdout, None, None, None), + &Bn254BlackBoxSolver(pedantic_solving), + &mut DefaultForeignCallBuilder::default().with_output(PrintOutput::Stdout).build(), )?; println!("Executed"); diff --git a/noir/noir-repo/yarn.lock b/noir/noir-repo/yarn.lock index 3c8df2b1772..44bff5a28c7 100644 --- a/noir/noir-repo/yarn.lock +++ b/noir/noir-repo/yarn.lock @@ -221,9 +221,9 @@ __metadata: languageName: node linkType: hard -"@aztec/bb.js@portal:../../../../barretenberg/ts::locator=integration-tests%40workspace%3Acompiler%2Fintegration-tests": - version: 0.0.0-use.local - resolution: "@aztec/bb.js@portal:../../../../barretenberg/ts::locator=integration-tests%40workspace%3Acompiler%2Fintegration-tests" +"@aztec/bb.js@npm:0.66.0": + version: 0.66.0 + resolution: "@aztec/bb.js@npm:0.66.0" dependencies: comlink: ^4.4.1 commander: ^10.0.1 @@ -232,9 +232,10 @@ __metadata: pako: ^2.1.0 tslib: ^2.4.0 bin: - bb.js: ./dest/node/main.js + bb.js: dest/node/main.js + checksum: 7295bf6543afe1c2db795a95c7ed40806de63c77e44bb4dacb2ec6a9171d1d34749150844ab47bc2499d06676e623acb408857b6aa9da702d3c576efadb8c906 languageName: node - linkType: soft + linkType: hard "@babel/code-frame@npm:^7.0.0, @babel/code-frame@npm:^7.10.4, @babel/code-frame@npm:^7.12.11, @babel/code-frame@npm:^7.16.0, @babel/code-frame@npm:^7.22.13, @babel/code-frame@npm:^7.23.5, @babel/code-frame@npm:^7.8.3": version: 7.23.5 @@ -352,6 +353,19 @@ __metadata: languageName: node linkType: hard +"@babel/generator@npm:^7.14.5": + version: 7.26.5 + resolution: "@babel/generator@npm:7.26.5" + dependencies: + "@babel/parser": ^7.26.5 + "@babel/types": ^7.26.5 + "@jridgewell/gen-mapping": ^0.3.5 + "@jridgewell/trace-mapping": ^0.3.25 + jsesc: ^3.0.2 + checksum: baa42a98cd01efa3ae3634a6caa81d0738e5e0bdba4efbf1ac735216c8d7cf6bdffeab69c468e6ab2063b07db402346113def4962719746756518432f83c53ba + languageName: node + linkType: hard + "@babel/generator@npm:^7.25.0, @babel/generator@npm:^7.25.6": version: 7.25.6 resolution: "@babel/generator@npm:7.25.6" @@ -764,6 +778,13 @@ __metadata: languageName: node linkType: hard +"@babel/helper-string-parser@npm:^7.25.9": + version: 7.25.9 + resolution: "@babel/helper-string-parser@npm:7.25.9" + checksum: 6435ee0849e101681c1849868278b5aee82686ba2c1e27280e5e8aca6233af6810d39f8e4e693d2f2a44a3728a6ccfd66f72d71826a94105b86b731697cdfa99 + languageName: node + linkType: hard + "@babel/helper-validator-identifier@npm:^7.22.20": version: 7.22.20 resolution: "@babel/helper-validator-identifier@npm:7.22.20" @@ -778,6 +799,13 @@ __metadata: languageName: node linkType: hard +"@babel/helper-validator-identifier@npm:^7.25.9": + version: 7.25.9 + resolution: "@babel/helper-validator-identifier@npm:7.25.9" + checksum: 5b85918cb1a92a7f3f508ea02699e8d2422fe17ea8e82acd445006c0ef7520fbf48e3dbcdaf7b0a1d571fc3a2715a29719e5226636cb6042e15fe6ed2a590944 + languageName: node + linkType: hard + "@babel/helper-validator-option@npm:^7.22.15, @babel/helper-validator-option@npm:^7.23.5": version: 7.23.5 resolution: "@babel/helper-validator-option@npm:7.23.5" @@ -878,6 +906,17 @@ __metadata: languageName: node linkType: hard +"@babel/parser@npm:^7.26.5": + version: 7.26.5 + resolution: "@babel/parser@npm:7.26.5" + dependencies: + "@babel/types": ^7.26.5 + bin: + parser: ./bin/babel-parser.js + checksum: 663aebf27c1dc04813e6c1d6e8e8fcb2954163ec297a95bdb3f1d0c2a0f04b504bddc09588fe4b176b43fad28c8a4b2914838a1edffdd426537a42f3ac644f1e + languageName: node + linkType: hard + "@babel/plugin-bugfix-firefox-class-in-computed-class-key@npm:^7.25.3": version: 7.25.3 resolution: "@babel/plugin-bugfix-firefox-class-in-computed-class-key@npm:7.25.3" @@ -2824,6 +2863,15 @@ __metadata: languageName: node linkType: hard +"@babel/runtime@npm:^7.13.10, @babel/runtime@npm:^7.20.13, @babel/runtime@npm:^7.5.5, @babel/runtime@npm:^7.8.7": + version: 7.26.0 + resolution: "@babel/runtime@npm:7.26.0" + dependencies: + regenerator-runtime: ^0.14.0 + checksum: c8e2c0504ab271b3467a261a8f119bf2603eb857a0d71e37791f4e3fae00f681365073cc79f141ddaa90c6077c60ba56448004ad5429d07ac73532be9f7cf28a + languageName: node + linkType: hard + "@babel/template@npm:^7.12.7, @babel/template@npm:^7.22.15": version: 7.22.15 resolution: "@babel/template@npm:7.22.15" @@ -2890,6 +2938,16 @@ __metadata: languageName: node linkType: hard +"@babel/types@npm:^7.14.5, @babel/types@npm:^7.20.7, @babel/types@npm:^7.26.5": + version: 7.26.5 + resolution: "@babel/types@npm:7.26.5" + dependencies: + "@babel/helper-string-parser": ^7.25.9 + "@babel/helper-validator-identifier": ^7.25.9 + checksum: 65dc14aa32ace22655c5edadeb99df80776c09cd93c105feaf49cc0583f3116aff0581b7eab630888c39ba61151f251c1399ec982b93585b0d1d1bf4a45b54f9 + languageName: node + linkType: hard + "@babel/types@npm:^7.21.3, @babel/types@npm:^7.24.7, @babel/types@npm:^7.24.8, @babel/types@npm:^7.25.0, @babel/types@npm:^7.25.2, @babel/types@npm:^7.25.6": version: 7.25.6 resolution: "@babel/types@npm:7.25.6" @@ -2908,6 +2966,87 @@ __metadata: languageName: node linkType: hard +"@cookbookdev/docsbot@npm:^4.24.9": + version: 4.24.9 + resolution: "@cookbookdev/docsbot@npm:4.24.9" + dependencies: + "@cookbookdev/sonner": 1.5.1 + "@headlessui/react": ^1.7.18 + "@headlessui/tailwindcss": ^0.2.0 + "@lingui/detect-locale": ^4.11.4 + "@lingui/macro": ^4.11.4 + "@lingui/react": ^4.11.4 + "@lingui/remote-loader": ^3.11.0 + "@monaco-editor/react": ^4.6.0 + "@radix-ui/react-avatar": ^1.0.4 + "@radix-ui/react-checkbox": ^1.0.4 + "@radix-ui/react-collapsible": ^1.0.3 + "@radix-ui/react-dialog": ^1.0.5 + "@radix-ui/react-dropdown-menu": ^2.0.6 + "@radix-ui/react-hover-card": ^1.0.7 + "@radix-ui/react-icons": ^1.3.0 + "@radix-ui/react-label": ^2.0.2 + "@radix-ui/react-popover": ^1.0.7 + "@radix-ui/react-scroll-area": ^1.0.5 + "@radix-ui/react-select": ^2.0.0 + "@radix-ui/react-separator": ^1.0.3 + "@radix-ui/react-slider": ^1.1.2 + "@radix-ui/react-slot": ^1.0.2 + "@radix-ui/react-tabs": ^1.0.4 + "@radix-ui/react-toast": ^1.1.5 + "@radix-ui/react-tooltip": ^1.0.7 + "@shikijs/rehype": ^1.12.1 + "@statsig/js-client": ^3.1.0 + "@statsig/react-bindings": ^3.1.0 + "@tailwindcss/line-clamp": ^0.4.4 + "@tanstack/react-table": ^8.11.7 + "@vercel/edge": ^1.1.1 + bcp-47: ^2.1.0 + can-dom-mutate: ^2.0.9 + clsx: ^2.1.0 + cmdk-react17: ^1.0.0 + color2k: ^2.0.3 + fflate: ^0.8.2 + file-saver: ^2.0.5 + framer-motion: ^6.5.1 + idb: ^8.0.0 + iso-639-1: ^3.1.3 + jszip: ^3.10.1 + nanoid: 3.3.7 + posthog-js: ^1.136.8 + prop-types: ^15.8.1 + react-complex-tree: ^2.3.7 + react-remark: ^2.1.0 + react-resizable-panels: 2.0.19 + recharts: ^2.12.4 + rehype-react: ^8.0.0 + remark-gfm: ^4.0.0 + remark-parse: ^11.0.0 + remark-rehype: ^11.1.0 + solc: ^0.8.25 + styled-components: ^6.1.8 + swr: ^2.2.5 + tailwind-merge: ^2.2.1 + tailwindcss-animate: ^1.0.7 + use-sync-external-store: ^1.2.0 + viem: 2.9.16 + peerDependencies: + react: ^16.8 || ^17.0 || ^18.0 + react-dom: ^16.8 || ^17.0 || ^18.0 + checksum: d1264012469a410d7232c3fa82f2855a2d603806142746ea16c05c21fda54632475214cbe1a2c390ee001d37823277c9d4e2d821d6a826e3cd5acef5b1da1b66 + languageName: node + linkType: hard + +"@cookbookdev/sonner@npm:1.5.1": + version: 1.5.1 + resolution: "@cookbookdev/sonner@npm:1.5.1" + peerDependencies: + react: ^18.0.0 + react-dom: ^18.0.0 + checksum: 658774d5055f47e1ccb64f97e9affc149bba6042b15318cd861ce790550041c9a3c96605b773a7853e32af0980d37eabd6db1077d0b2a103495ea9d025d4ff4e + languageName: node + linkType: hard + "@cspell/cspell-bundled-dicts@npm:8.3.2": version: 8.3.2 resolution: "@cspell/cspell-bundled-dicts@npm:8.3.2" @@ -4256,6 +4395,45 @@ __metadata: languageName: node linkType: hard +"@emotion/is-prop-valid@npm:1.2.2": + version: 1.2.2 + resolution: "@emotion/is-prop-valid@npm:1.2.2" + dependencies: + "@emotion/memoize": ^0.8.1 + checksum: 61f6b128ea62b9f76b47955057d5d86fcbe2a6989d2cd1e583daac592901a950475a37d049b9f7a7c6aa8758a33b408735db759fdedfd1f629df0f85ab60ea25 + languageName: node + linkType: hard + +"@emotion/is-prop-valid@npm:^0.8.2": + version: 0.8.8 + resolution: "@emotion/is-prop-valid@npm:0.8.8" + dependencies: + "@emotion/memoize": 0.7.4 + checksum: bb7ec6d48c572c540e24e47cc94fc2f8dec2d6a342ae97bc9c8b6388d9b8d283862672172a1bb62d335c02662afe6291e10c71e9b8642664a8b43416cdceffac + languageName: node + linkType: hard + +"@emotion/memoize@npm:0.7.4": + version: 0.7.4 + resolution: "@emotion/memoize@npm:0.7.4" + checksum: 4e3920d4ec95995657a37beb43d3f4b7d89fed6caa2b173a4c04d10482d089d5c3ea50bbc96618d918b020f26ed6e9c4026bbd45433566576c1f7b056c3271dc + languageName: node + linkType: hard + +"@emotion/memoize@npm:^0.8.1": + version: 0.8.1 + resolution: "@emotion/memoize@npm:0.8.1" + checksum: a19cc01a29fcc97514948eaab4dc34d8272e934466ed87c07f157887406bc318000c69ae6f813a9001c6a225364df04249842a50e692ef7a9873335fbcc141b0 + languageName: node + linkType: hard + +"@emotion/unitless@npm:0.8.1": + version: 0.8.1 + resolution: "@emotion/unitless@npm:0.8.1" + checksum: 385e21d184d27853bb350999471f00e1429fa4e83182f46cd2c164985999d9b46d558dc8b9cc89975cb337831ce50c31ac2f33b15502e85c299892e67e7b4a88 + languageName: node + linkType: hard + "@esbuild/android-arm64@npm:0.17.19": version: 0.17.19 resolution: "@esbuild/android-arm64@npm:0.17.19" @@ -4835,6 +5013,44 @@ __metadata: languageName: node linkType: hard +"@floating-ui/core@npm:^1.6.0": + version: 1.6.9 + resolution: "@floating-ui/core@npm:1.6.9" + dependencies: + "@floating-ui/utils": ^0.2.9 + checksum: 21cbcac72a40172399570dedf0eb96e4f24b0d829980160e8d14edf08c2955ac6feffb7b94e1530c78fb7944635e52669c9257ad08570e0295efead3b5a9af91 + languageName: node + linkType: hard + +"@floating-ui/dom@npm:^1.0.0": + version: 1.6.13 + resolution: "@floating-ui/dom@npm:1.6.13" + dependencies: + "@floating-ui/core": ^1.6.0 + "@floating-ui/utils": ^0.2.9 + checksum: eabab9d860d3b5beab1c2d6936287efc4d9ab352de99062380589ef62870d59e8730397489c34a96657e128498001b5672330c4a9da0159fe8b2401ac59fe314 + languageName: node + linkType: hard + +"@floating-ui/react-dom@npm:^2.0.0": + version: 2.1.2 + resolution: "@floating-ui/react-dom@npm:2.1.2" + dependencies: + "@floating-ui/dom": ^1.0.0 + peerDependencies: + react: ">=16.8.0" + react-dom: ">=16.8.0" + checksum: 25bb031686e23062ed4222a8946e76b3f9021d40a48437bd747233c4964a766204b8a55f34fa8b259839af96e60db7c6e3714d81f1de06914294f90e86ffbc48 + languageName: node + linkType: hard + +"@floating-ui/utils@npm:^0.2.9": + version: 0.2.9 + resolution: "@floating-ui/utils@npm:0.2.9" + checksum: d518b80cec5a323e54a069a1dd99a20f8221a4853ed98ac16c75275a0cc22f75de4f8ac5b121b4f8990bd45da7ad1fb015b9a1e4bac27bb1cd62444af84e9784 + languageName: node + linkType: hard + "@hapi/hoek@npm:^9.0.0": version: 9.3.0 resolution: "@hapi/hoek@npm:9.3.0" @@ -4851,6 +5067,28 @@ __metadata: languageName: node linkType: hard +"@headlessui/react@npm:^1.7.18": + version: 1.7.19 + resolution: "@headlessui/react@npm:1.7.19" + dependencies: + "@tanstack/react-virtual": ^3.0.0-beta.60 + client-only: ^0.0.1 + peerDependencies: + react: ^16 || ^17 || ^18 + react-dom: ^16 || ^17 || ^18 + checksum: 2a343a5fcf1f45e870cc94613231b89a8da78114001ffafa4751a0eceae7569ff9237aff1f2aedfa6f6e53ee3bb9ba5e5d19ebf1878fee3ff4f3c733fddc1087 + languageName: node + linkType: hard + +"@headlessui/tailwindcss@npm:^0.2.0": + version: 0.2.1 + resolution: "@headlessui/tailwindcss@npm:0.2.1" + peerDependencies: + tailwindcss: ^3.0 + checksum: 5cc9b30af4348dd133ce7fe6b7bdae9bec681796480cbd91cc8d20cd8a0841169ac741ec04ded4235c520199d79775ab439489d869b2785cc58a94002193a6f0 + languageName: node + linkType: hard + "@humanwhocodes/config-array@npm:^0.11.14": version: 0.11.14 resolution: "@humanwhocodes/config-array@npm:0.11.14" @@ -5027,6 +5265,89 @@ __metadata: languageName: node linkType: hard +"@lingui/conf@npm:4.14.1": + version: 4.14.1 + resolution: "@lingui/conf@npm:4.14.1" + dependencies: + "@babel/runtime": ^7.20.13 + chalk: ^4.1.0 + cosmiconfig: ^8.0.0 + jest-validate: ^29.4.3 + jiti: ^1.17.1 + lodash.get: ^4.4.2 + checksum: 6108724647a85c2114fb5bd3212de16f56f1495bf94fffa3fb944a1187e7e30df63947628534bc9880483e3b290761287e317ec48d3c12fe90d654332d199b78 + languageName: node + linkType: hard + +"@lingui/core@npm:4.14.1": + version: 4.14.1 + resolution: "@lingui/core@npm:4.14.1" + dependencies: + "@babel/runtime": ^7.20.13 + "@lingui/message-utils": 4.14.1 + unraw: ^3.0.0 + checksum: 6bf82be81a84a7e5a34a5a7a3d9d503b4eca6d310655a306549e575a50aae3c6e6964022ab98229c7cbeef45ab368d5c6bc6a713abefc367f1f16b1d80e9d226 + languageName: node + linkType: hard + +"@lingui/detect-locale@npm:^4.11.4": + version: 4.14.1 + resolution: "@lingui/detect-locale@npm:4.14.1" + checksum: 5d2e06bb11a98634eb62dfaa587baa2aaaf6e5f4e577447eac67bacae37c7aa85fac835272080cc07bb07b27b96351ddc6bd630bec7fe852e346b2b5d281fb8d + languageName: node + linkType: hard + +"@lingui/macro@npm:^4.11.4": + version: 4.14.1 + resolution: "@lingui/macro@npm:4.14.1" + dependencies: + "@babel/runtime": ^7.20.13 + "@babel/types": ^7.20.7 + "@lingui/conf": 4.14.1 + "@lingui/core": 4.14.1 + "@lingui/message-utils": 4.14.1 + peerDependencies: + "@lingui/react": ^4.0.0 + babel-plugin-macros: 2 || 3 + checksum: f479ca66cea835044fb40170734aa99ff1b0e7c20b9efd4ec04a33d324df1b80b2c708ce12c24a6e839fdc737104170d674d21a54f679e6a4a81a23f3355c3aa + languageName: node + linkType: hard + +"@lingui/message-utils@npm:4.14.1": + version: 4.14.1 + resolution: "@lingui/message-utils@npm:4.14.1" + dependencies: + "@messageformat/parser": ^5.0.0 + js-sha256: ^0.10.1 + checksum: 1b699891b3e62e54cdf69fcdedb5c9c7bbe15e02798f44b617d999a3e15a85b38f49d43d47b335e983c85335b50fc2d220d562b1fd0c241608875241d13006ec + languageName: node + linkType: hard + +"@lingui/react@npm:^4.11.4": + version: 4.14.1 + resolution: "@lingui/react@npm:4.14.1" + dependencies: + "@babel/runtime": ^7.20.13 + "@lingui/core": 4.14.1 + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + checksum: b448eb9fc83415428b5cff501497993c2d0f40768f5ae10d9cf6e5d4ae6f711a08c573380165fda277c0a9805447d7b9aff901a33705abf1eaadde97d330adf4 + languageName: node + linkType: hard + +"@lingui/remote-loader@npm:^3.11.0": + version: 3.11.0 + resolution: "@lingui/remote-loader@npm:3.11.0" + dependencies: + "@babel/generator": ^7.14.5 + "@babel/types": ^7.14.5 + json5: ^2.2.0 + messageformat-parser: ^4.1.3 + ramda: ^0.27.1 + checksum: 714ac3c93f20a92d7c1e3f184adb28c8a6e77c614cbb159af0297fc2c962bbc1e644a68be23e5343f5e9c56487139f69e9c6694042647530e600d0fe9f85777f + languageName: node + linkType: hard + "@ltd/j-toml@npm:^1.38.0": version: 1.38.0 resolution: "@ltd/j-toml@npm:1.38.0" @@ -5034,6 +5355,15 @@ __metadata: languageName: node linkType: hard +"@mapbox/hast-util-table-cell-style@npm:^0.2.0": + version: 0.2.1 + resolution: "@mapbox/hast-util-table-cell-style@npm:0.2.1" + dependencies: + unist-util-visit: ^1.4.1 + checksum: d7151365d10cb8a02f08973064fd2619b4ce1dab4139c8c5e9ee21f9c364ecf5e2fae066136e815cf8035496070c6280bb5215b5bfe9c9cdc804b50025e94fa2 + languageName: node + linkType: hard + "@mdn/browser-compat-data@npm:^4.0.0": version: 4.2.1 resolution: "@mdn/browser-compat-data@npm:4.2.1" @@ -5118,6 +5448,15 @@ __metadata: languageName: node linkType: hard +"@messageformat/parser@npm:^5.0.0": + version: 5.1.1 + resolution: "@messageformat/parser@npm:5.1.1" + dependencies: + moo: ^0.5.1 + checksum: 530a53b1445709f3a0361c378892fe84a976a3977869dfb7e31580375f3ddb6e9fac933dfb4f1f214c22e716e8e52f6ada2c47c765c5f3d87ca4747f5076eeaa + languageName: node + linkType: hard + "@metamask/eth-sig-util@npm:^4.0.0": version: 4.0.1 resolution: "@metamask/eth-sig-util@npm:4.0.1" @@ -5131,7 +5470,96 @@ __metadata: languageName: node linkType: hard -"@noble/curves@npm:1.2.0": +"@monaco-editor/loader@npm:^1.4.0": + version: 1.4.0 + resolution: "@monaco-editor/loader@npm:1.4.0" + dependencies: + state-local: ^1.0.6 + peerDependencies: + monaco-editor: ">= 0.21.0 < 1" + checksum: 374ec0ea872ee15b33310e105a43217148161480d3955c5cece87d0f801754cd2c45a3f6c539a75da18a066c1615756fb87eaf1003f1df6a64a0cbce5d2c3749 + languageName: node + linkType: hard + +"@monaco-editor/react@npm:^4.6.0": + version: 4.6.0 + resolution: "@monaco-editor/react@npm:4.6.0" + dependencies: + "@monaco-editor/loader": ^1.4.0 + peerDependencies: + monaco-editor: ">= 0.25.0 < 1" + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 + checksum: 9d44e76c5baad6db5f84c90a5540fbd3c9af691b97d76cf2a99b3c8273004d0efe44c2572d80e9d975c9af10022c21e4a66923924950a5201e82017c8b20428c + languageName: node + linkType: hard + +"@motionone/animation@npm:^10.12.0": + version: 10.18.0 + resolution: "@motionone/animation@npm:10.18.0" + dependencies: + "@motionone/easing": ^10.18.0 + "@motionone/types": ^10.17.1 + "@motionone/utils": ^10.18.0 + tslib: ^2.3.1 + checksum: 841cb9f4843a89e5e4560b9f960f52cbe78afc86f87c769f71e9edb3aadd53fb87982b7e11914428f228b29fd580756be531369c2ffac06432550afa4e87d1c3 + languageName: node + linkType: hard + +"@motionone/dom@npm:10.12.0": + version: 10.12.0 + resolution: "@motionone/dom@npm:10.12.0" + dependencies: + "@motionone/animation": ^10.12.0 + "@motionone/generators": ^10.12.0 + "@motionone/types": ^10.12.0 + "@motionone/utils": ^10.12.0 + hey-listen: ^1.0.8 + tslib: ^2.3.1 + checksum: 123356f28e44362c4f081aae3df22e576f46bfcb07e01257b2ac64a115668448f29b8de67e4b6e692c5407cffb78ffe7cf9fa1bc064007482bab5dd23a69d380 + languageName: node + linkType: hard + +"@motionone/easing@npm:^10.18.0": + version: 10.18.0 + resolution: "@motionone/easing@npm:10.18.0" + dependencies: + "@motionone/utils": ^10.18.0 + tslib: ^2.3.1 + checksum: 6bd37f7a9d5a88f868cc0ad6e47d2ba8d9fefd7da84fccfea7ed77ec08c2e6d1e42df88dda462665102a5cf03f748231a1a077de7054b5a8ccb0fbf36f61b1e7 + languageName: node + linkType: hard + +"@motionone/generators@npm:^10.12.0": + version: 10.18.0 + resolution: "@motionone/generators@npm:10.18.0" + dependencies: + "@motionone/types": ^10.17.1 + "@motionone/utils": ^10.18.0 + tslib: ^2.3.1 + checksum: 51a0e075681697b11d0771998cac8c76a745f00141502f81adb953896992b7f49478965e4afe696bc83361afaae8d2f1057d71c25b21035fe67258ff73764f1c + languageName: node + linkType: hard + +"@motionone/types@npm:^10.12.0, @motionone/types@npm:^10.17.1": + version: 10.17.1 + resolution: "@motionone/types@npm:10.17.1" + checksum: 3fa74db64e371e61a7f7669d7d541d11c9a8dd871032d59c69041e3b2e07a67ad2ed8767cb9273bac90eed4e1f76efc1f14c8673c2e9a288f6070ee0fef64a25 + languageName: node + linkType: hard + +"@motionone/utils@npm:^10.12.0, @motionone/utils@npm:^10.18.0": + version: 10.18.0 + resolution: "@motionone/utils@npm:10.18.0" + dependencies: + "@motionone/types": ^10.17.1 + hey-listen: ^1.0.8 + tslib: ^2.3.1 + checksum: a27f9afde693a0cbbbcb33962b12bbe40dd2cfa514b0732f3c7953c5ef4beed738e1e8172a2de89e3b9f74a253ef0a70d7f3efb730be97b77d7176a3ffacb67a + languageName: node + linkType: hard + +"@noble/curves@npm:1.2.0, @noble/curves@npm:~1.2.0": version: 1.2.0 resolution: "@noble/curves@npm:1.2.0" dependencies: @@ -5154,6 +5582,13 @@ __metadata: languageName: node linkType: hard +"@noble/hashes@npm:~1.3.0, @noble/hashes@npm:~1.3.2": + version: 1.3.3 + resolution: "@noble/hashes@npm:1.3.3" + checksum: 8a6496d1c0c64797339bc694ad06cdfaa0f9e56cd0c3f68ae3666cfb153a791a55deb0af9c653c7ed2db64d537aa3e3054629740d2f2338bb1dcb7ab60cd205b + languageName: node + linkType: hard + "@noble/secp256k1@npm:1.7.1, @noble/secp256k1@npm:~1.7.0": version: 1.7.1 resolution: "@noble/secp256k1@npm:1.7.1" @@ -5825,68 +6260,1231 @@ __metadata: languageName: node linkType: hard -"@rollup/plugin-node-resolve@npm:^15.0.1": - version: 15.2.3 - resolution: "@rollup/plugin-node-resolve@npm:15.2.3" +"@radix-ui/number@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/number@npm:1.1.0" + checksum: e4fc7483c19141c25dbaf3d140b75e2b7fed0bfa3ad969f4441f0266ed34b35413f57a35df7b025e2a977152bbe6131849d3444fc6f15a73345dfc2bfdc105fa + languageName: node + linkType: hard + +"@radix-ui/primitive@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/primitive@npm:1.0.1" dependencies: - "@rollup/pluginutils": ^5.0.1 - "@types/resolve": 1.20.2 - deepmerge: ^4.2.2 - is-builtin-module: ^3.2.1 - is-module: ^1.0.0 - resolve: ^1.22.1 + "@babel/runtime": ^7.13.10 + checksum: 2b93e161d3fdabe9a64919def7fa3ceaecf2848341e9211520c401181c9eaebb8451c630b066fad2256e5c639c95edc41de0ba59c40eff37e799918d019822d1 + languageName: node + linkType: hard + +"@radix-ui/primitive@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/primitive@npm:1.1.1" + checksum: d7e819177590108b74139809d52ec043c0962ae3513e947998be575fb13639c5c1c091896ddcf1d6a22a777d44ade59d22c2019ce9099607fc62a5de09c59707 + languageName: node + linkType: hard + +"@radix-ui/react-arrow@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/react-arrow@npm:1.1.1" + dependencies: + "@radix-ui/react-primitive": 2.0.1 peerDependencies: - rollup: ^2.78.0||^3.0.0||^4.0.0 + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc peerDependenciesMeta: - rollup: + "@types/react": optional: true - checksum: 730f32c2f8fdddff07cf0fca86a5dac7c475605fb96930197a868c066e62eb6388c557545e4f7d99b7a283411754c9fbf98944ab086b6074e04fc1292e234aa8 + "@types/react-dom": + optional: true + checksum: c75505c2858cffff7c742e888b635879f9a6d95e08bf5ae939be33f97e1171379bc6b5354ec0cd3d12624bdbe5a830ee6aa0fb1f46b1af160b488bc54e64d486 languageName: node linkType: hard -"@rollup/pluginutils@npm:^5.0.1": - version: 5.1.0 - resolution: "@rollup/pluginutils@npm:5.1.0" +"@radix-ui/react-avatar@npm:^1.0.4": + version: 1.1.2 + resolution: "@radix-ui/react-avatar@npm:1.1.2" dependencies: - "@types/estree": ^1.0.0 - estree-walker: ^2.0.2 - picomatch: ^2.3.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + "@radix-ui/react-use-layout-effect": 1.1.0 peerDependencies: - rollup: ^1.20.0||^2.0.0||^3.0.0||^4.0.0 + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc peerDependenciesMeta: - rollup: + "@types/react": optional: true - checksum: 3cc5a6d91452a6eabbfd1ae79b4dd1f1e809d2eecda6e175deb784e75b0911f47e9ecce73f8dd315d6a8b3f362582c91d3c0f66908b6ced69345b3cbe28f8ce8 + "@types/react-dom": + optional: true + checksum: fa2c95c3f9393cee30b5416147f9701edd335c27f1132782627db4dbd2545dbc5fdcea0a6e66945e3072ea91c80ec45183a07159a882c974ea5fb6bfc140fbb4 languageName: node linkType: hard -"@rollup/rollup-android-arm-eabi@npm:4.9.4": - version: 4.9.4 - resolution: "@rollup/rollup-android-arm-eabi@npm:4.9.4" - conditions: os=android & cpu=arm +"@radix-ui/react-checkbox@npm:^1.0.4": + version: 1.1.3 + resolution: "@radix-ui/react-checkbox@npm:1.1.3" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + "@radix-ui/react-use-previous": 1.1.0 + "@radix-ui/react-use-size": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: ff2e62ba5f4bfdd6ef0cbbdf38a9f2249793d3a95b8fa55f95506198276e01c849b09bf963ec435c3f3ed43b18e27ddcb3087be4f6558e8d4f4dab8a27c49399 languageName: node linkType: hard -"@rollup/rollup-android-arm64@npm:4.9.4": - version: 4.9.4 - resolution: "@rollup/rollup-android-arm64@npm:4.9.4" - conditions: os=android & cpu=arm64 +"@radix-ui/react-collapsible@npm:^1.0.3": + version: 1.1.2 + resolution: "@radix-ui/react-collapsible@npm:1.1.2" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + "@radix-ui/react-use-layout-effect": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: c605d2eeb21417e243b3e7bf975af21959c56d3d77677ae320eac7f38227806ca1f620833000c077a649e3b5e93652eaeb129756265c1423cff9f791af0b715d languageName: node linkType: hard -"@rollup/rollup-darwin-arm64@npm:4.9.4": - version: 4.9.4 - resolution: "@rollup/rollup-darwin-arm64@npm:4.9.4" - conditions: os=darwin & cpu=arm64 +"@radix-ui/react-collection@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/react-collection@npm:1.1.1" + dependencies: + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-slot": 1.1.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 413fbcb542ae5efefab05053a9cc3380586e7bc3d36427845a62749d90284986333e609066b952cc6ccd0a8ef49e8de773ea1cd294aecff3cdb1eb4047e97c2f languageName: node linkType: hard -"@rollup/rollup-darwin-x64@npm:4.9.4": - version: 4.9.4 - resolution: "@rollup/rollup-darwin-x64@npm:4.9.4" - conditions: os=darwin & cpu=x64 - languageName: node - linkType: hard +"@radix-ui/react-compose-refs@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/react-compose-refs@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.13.10 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 2b9a613b6db5bff8865588b6bf4065f73021b3d16c0a90b2d4c23deceeb63612f1f15de188227ebdc5f88222cab031be617a9dd025874c0487b303be3e5cc2a8 + languageName: node + linkType: hard + +"@radix-ui/react-compose-refs@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/react-compose-refs@npm:1.1.1" + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 1be82f9f7fab96cc10f167a2e4f976e0135a63d473334f664c06f02af13bc5ea1994cb0505f89ed190d756cb65d57506721c030908af07e49b9e3cfd36044f33 + languageName: node + linkType: hard + +"@radix-ui/react-context@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/react-context@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.13.10 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 60e9b81d364f40c91a6213ec953f7c64fcd9d75721205a494a5815b3e5ae0719193429b62ee6c7002cd6aaf70f8c0e2f08bdbaba9ffcc233044d32b56d2127d1 + languageName: node + linkType: hard + +"@radix-ui/react-context@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/react-context@npm:1.1.1" + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 9a04db236685dacc2f5ab2bdcfc4c82b974998e712ab97d79b11d5b4ef073d24aa9392398c876ef6cb3c59f40299285ceee3646187ad818cdad4fe1c74469d3f + languageName: node + linkType: hard + +"@radix-ui/react-dialog@npm:1.0.5": + version: 1.0.5 + resolution: "@radix-ui/react-dialog@npm:1.0.5" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/primitive": 1.0.1 + "@radix-ui/react-compose-refs": 1.0.1 + "@radix-ui/react-context": 1.0.1 + "@radix-ui/react-dismissable-layer": 1.0.5 + "@radix-ui/react-focus-guards": 1.0.1 + "@radix-ui/react-focus-scope": 1.0.4 + "@radix-ui/react-id": 1.0.1 + "@radix-ui/react-portal": 1.0.4 + "@radix-ui/react-presence": 1.0.1 + "@radix-ui/react-primitive": 1.0.3 + "@radix-ui/react-slot": 1.0.2 + "@radix-ui/react-use-controllable-state": 1.0.1 + aria-hidden: ^1.1.1 + react-remove-scroll: 2.5.5 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 + react-dom: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 3d11ca31afb794a6dd286005ab7894cb0ce7bc2de5481de98900470b11d495256401306763de030f5e35aa545ff90d34632ffd54a1b29bf55afba813be4bb84a + languageName: node + linkType: hard + +"@radix-ui/react-dialog@npm:^1.0.5": + version: 1.1.4 + resolution: "@radix-ui/react-dialog@npm:1.1.4" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-dismissable-layer": 1.1.3 + "@radix-ui/react-focus-guards": 1.1.1 + "@radix-ui/react-focus-scope": 1.1.1 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-portal": 1.1.3 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-slot": 1.1.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + aria-hidden: ^1.1.1 + react-remove-scroll: ^2.6.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 695b35c7283adfe2be4ad88d30f0ad08be099a55dfd54e49ede61074846255b426eb57734509b5fc6f349439a051b48e44e58b542c2605383aa721a1b2bd7861 + languageName: node + linkType: hard + +"@radix-ui/react-direction@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/react-direction@npm:1.1.0" + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 25ad0d1d65ad08c93cebfbefdff9ef2602e53f4573a66b37d2c366ede9485e75ec6fc8e7dd7d2939b34ea5504ca0fe6ac4a3acc2f6ee9b62d131d65486eafd49 + languageName: node + linkType: hard + +"@radix-ui/react-dismissable-layer@npm:1.0.5": + version: 1.0.5 + resolution: "@radix-ui/react-dismissable-layer@npm:1.0.5" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/primitive": 1.0.1 + "@radix-ui/react-compose-refs": 1.0.1 + "@radix-ui/react-primitive": 1.0.3 + "@radix-ui/react-use-callback-ref": 1.0.1 + "@radix-ui/react-use-escape-keydown": 1.0.3 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 + react-dom: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: e73cf4bd3763f4d55b1bea7486a9700384d7d94dc00b1d5a75e222b2f1e4f32bc667a206ca4ed3baaaf7424dce7a239afd0ba59a6f0d89c3462c4e6e8d029a04 + languageName: node + linkType: hard + +"@radix-ui/react-dismissable-layer@npm:1.1.3": + version: 1.1.3 + resolution: "@radix-ui/react-dismissable-layer@npm:1.1.3" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + "@radix-ui/react-use-escape-keydown": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 26d15726bdb274aeb8d801fd163051c270707fb19e9bac4e0e90b368e79063a5347a0b15dc3aadc0bbafa157674e9e796d785d720bd5132c059ac5294ac73a81 + languageName: node + linkType: hard + +"@radix-ui/react-dropdown-menu@npm:^2.0.6": + version: 2.1.4 + resolution: "@radix-ui/react-dropdown-menu@npm:2.1.4" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-menu": 2.1.4 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 51014b38daab21d32164813d228b182e5dbf90c77115e5c32b8bbd37696a303485fe30fa2e0a097e1ca3f474cad0dd15efcecc3f567b5edf0a79b5092fe3b4e0 + languageName: node + linkType: hard + +"@radix-ui/react-focus-guards@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/react-focus-guards@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.13.10 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 1f8ca8f83b884b3612788d0742f3f054e327856d90a39841a47897dbed95e114ee512362ae314177de226d05310047cabbf66b686ae86ad1b65b6b295be24ef7 + languageName: node + linkType: hard + +"@radix-ui/react-focus-guards@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/react-focus-guards@npm:1.1.1" + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: ac8dd31f48fa0500bafd9368f2f06c5a06918dccefa89fa5dc77ca218dc931a094a81ca57f6b181138029822f7acdd5280dceccf5ba4d9263c754fb8f7961879 + languageName: node + linkType: hard + +"@radix-ui/react-focus-scope@npm:1.0.4": + version: 1.0.4 + resolution: "@radix-ui/react-focus-scope@npm:1.0.4" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/react-compose-refs": 1.0.1 + "@radix-ui/react-primitive": 1.0.3 + "@radix-ui/react-use-callback-ref": 1.0.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 + react-dom: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 3481db1a641513a572734f0bcb0e47fefeba7bccd6ec8dde19f520719c783ef0b05a55ef0d5292078ed051cc5eda46b698d5d768da02e26e836022f46b376fd1 + languageName: node + linkType: hard + +"@radix-ui/react-focus-scope@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/react-focus-scope@npm:1.1.1" + dependencies: + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 8716fe9b029a66f81b37e4e22457dd0fc7b4dba573d712454e18ead850f256d84cd994eeebcc31dd7780cf1028b6410d9ebe152fff4478d3b4ce2700690a38f4 + languageName: node + linkType: hard + +"@radix-ui/react-hover-card@npm:^1.0.7": + version: 1.1.4 + resolution: "@radix-ui/react-hover-card@npm:1.1.4" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-dismissable-layer": 1.1.3 + "@radix-ui/react-popper": 1.2.1 + "@radix-ui/react-portal": 1.1.3 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 1c3e0d8edc01f714c1ca389fb28f4ed843726e0cb56b21dc7d79e9680f2c19138ce494eb3c333ef7ad524a94209494321b4ca70a44f56fdb01c6a90d30c02058 + languageName: node + linkType: hard + +"@radix-ui/react-icons@npm:^1.3.0": + version: 1.3.2 + resolution: "@radix-ui/react-icons@npm:1.3.2" + peerDependencies: + react: ^16.x || ^17.x || ^18.x || ^19.0.0 || ^19.0.0-rc + checksum: 4dbd60d9ca3481d3f01d99862ecfc2a06ac731603a54d45fe3027bbdb986c60a3c080c06b689b1fcc1628a935b07f765e8088a21902df55c2366ec61acbe9b30 + languageName: node + linkType: hard + +"@radix-ui/react-id@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/react-id@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/react-use-layout-effect": 1.0.1 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 446a453d799cc790dd2a1583ff8328da88271bff64530b5a17c102fa7fb35eece3cf8985359d416f65e330cd81aa7b8fe984ea125fc4f4eaf4b3801d698e49fe + languageName: node + linkType: hard + +"@radix-ui/react-id@npm:1.1.0, @radix-ui/react-id@npm:^1.0.1": + version: 1.1.0 + resolution: "@radix-ui/react-id@npm:1.1.0" + dependencies: + "@radix-ui/react-use-layout-effect": 1.1.0 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 6fbc9d1739b3b082412da10359e63967b4f3a60383ebda4c9e56b07a722d29bee53b203b3b1418f88854a29315a7715867133bb149e6e22a027a048cdd20d970 + languageName: node + linkType: hard + +"@radix-ui/react-label@npm:^2.0.2": + version: 2.1.1 + resolution: "@radix-ui/react-label@npm:2.1.1" + dependencies: + "@radix-ui/react-primitive": 2.0.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 40525096bf6fb5a1cfbf5c87aa46fd13d1329155fa62a42e35b4b55db5559d42081b97e77bf5f7ac45064fdae947ee6f78abb1f55b18821760a5712492243e6d + languageName: node + linkType: hard + +"@radix-ui/react-menu@npm:2.1.4": + version: 2.1.4 + resolution: "@radix-ui/react-menu@npm:2.1.4" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-collection": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-direction": 1.1.0 + "@radix-ui/react-dismissable-layer": 1.1.3 + "@radix-ui/react-focus-guards": 1.1.1 + "@radix-ui/react-focus-scope": 1.1.1 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-popper": 1.2.1 + "@radix-ui/react-portal": 1.1.3 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-roving-focus": 1.1.1 + "@radix-ui/react-slot": 1.1.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + aria-hidden: ^1.1.1 + react-remove-scroll: ^2.6.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 2a20db7c017075d3ceb07b076dfdbdc3c4d26825642e2f52d4bda9947078725db129c2b41a93a6f1393526eca3bf586f54dc47a9a66e1eef621c94a55a216aa4 + languageName: node + linkType: hard + +"@radix-ui/react-popover@npm:^1.0.7": + version: 1.1.4 + resolution: "@radix-ui/react-popover@npm:1.1.4" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-dismissable-layer": 1.1.3 + "@radix-ui/react-focus-guards": 1.1.1 + "@radix-ui/react-focus-scope": 1.1.1 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-popper": 1.2.1 + "@radix-ui/react-portal": 1.1.3 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-slot": 1.1.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + aria-hidden: ^1.1.1 + react-remove-scroll: ^2.6.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: f4525ac9a2f5957ad709749daddb78e58d8b1471dfd8683ca713d1ade9aac202b30c7179b798471e90ab13a01f01a70a3bc4002a872c279ee383bd3ad8ad49e6 + languageName: node + linkType: hard + +"@radix-ui/react-popper@npm:1.2.1": + version: 1.2.1 + resolution: "@radix-ui/react-popper@npm:1.2.1" + dependencies: + "@floating-ui/react-dom": ^2.0.0 + "@radix-ui/react-arrow": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + "@radix-ui/react-use-layout-effect": 1.1.0 + "@radix-ui/react-use-rect": 1.1.0 + "@radix-ui/react-use-size": 1.1.0 + "@radix-ui/rect": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 1416acda53d06d497d40a587e02ef821dcb955f2eee86bad3a9acacfd7fda8601e5d36a9cbe5e47d200052169ccd2d840b685c51e7192afdf3fc7fa072274ee0 + languageName: node + linkType: hard + +"@radix-ui/react-portal@npm:1.0.4": + version: 1.0.4 + resolution: "@radix-ui/react-portal@npm:1.0.4" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/react-primitive": 1.0.3 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 + react-dom: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: c4cf35e2f26a89703189d0eef3ceeeb706ae0832e98e558730a5e929ca7c72c7cb510413a24eca94c7732f8d659a1e81942bec7b90540cb73ce9e4885d040b64 + languageName: node + linkType: hard + +"@radix-ui/react-portal@npm:1.1.3": + version: 1.1.3 + resolution: "@radix-ui/react-portal@npm:1.1.3" + dependencies: + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-layout-effect": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 470fb50c940772d05cc268e219b3d15848909dcd0a2dc1952965d0af905992f0ccab99e99c490dea6564c441397eba720b8425ba9f4582c94bef40ebe27ac0d0 + languageName: node + linkType: hard + +"@radix-ui/react-presence@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/react-presence@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/react-compose-refs": 1.0.1 + "@radix-ui/react-use-layout-effect": 1.0.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 + react-dom: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: ed2ff9faf9e4257a4065034d3771459e5a91c2d840b2fcec94661761704dbcb65bcdd927d28177a2a129b3dab5664eb90a9b88309afe0257a9f8ba99338c0d95 + languageName: node + linkType: hard + +"@radix-ui/react-presence@npm:1.1.2": + version: 1.1.2 + resolution: "@radix-ui/react-presence@npm:1.1.2" + dependencies: + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-use-layout-effect": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 0345bc8d3e1ddcbf4b864025833c71f3d76e4801ce16ad126a98aed816be6e819c4fe01097c6c1320771b947f5a14929cc610d18e7a1438cfb5573289fa4d4a6 + languageName: node + linkType: hard + +"@radix-ui/react-primitive@npm:1.0.3": + version: 1.0.3 + resolution: "@radix-ui/react-primitive@npm:1.0.3" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/react-slot": 1.0.2 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 + react-dom: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 9402bc22923c8e5c479051974a721c301535c36521c0237b83e5fa213d013174e77f3ad7905e6d60ef07e14f88ec7f4ea69891dc7a2b39047f8d3640e8f8d713 + languageName: node + linkType: hard + +"@radix-ui/react-primitive@npm:2.0.1": + version: 2.0.1 + resolution: "@radix-ui/react-primitive@npm:2.0.1" + dependencies: + "@radix-ui/react-slot": 1.1.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: d75882209101155f20babcff9475b887929db6473cd8e5b56d0c24d24d0042202e0fa785e6d6c6b322a96d9777cd0ef7610def9e11ea69839c6b204f1c99cf16 + languageName: node + linkType: hard + +"@radix-ui/react-roving-focus@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/react-roving-focus@npm:1.1.1" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-collection": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-direction": 1.1.0 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + "@radix-ui/react-use-controllable-state": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: f3a78bd15cca322b384758c938106414b4cbe99aa58ca3eb12ec9d66552c7fa137128e846f413a6b7027f4bed04b2233c9aad8adb1ba07e14ed7a697f972d4f6 + languageName: node + linkType: hard + +"@radix-ui/react-scroll-area@npm:^1.0.5": + version: 1.2.2 + resolution: "@radix-ui/react-scroll-area@npm:1.2.2" + dependencies: + "@radix-ui/number": 1.1.0 + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-direction": 1.1.0 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + "@radix-ui/react-use-layout-effect": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 2cd63d159442d55951c8e9c447a513f9e3c435a7c9555165b26dceda6d4047432a25acded3f6c18f51c9abbae2ccf6b38d81cc4767d78133fb89d3de73ddaca8 + languageName: node + linkType: hard + +"@radix-ui/react-select@npm:^2.0.0": + version: 2.1.4 + resolution: "@radix-ui/react-select@npm:2.1.4" + dependencies: + "@radix-ui/number": 1.1.0 + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-collection": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-direction": 1.1.0 + "@radix-ui/react-dismissable-layer": 1.1.3 + "@radix-ui/react-focus-guards": 1.1.1 + "@radix-ui/react-focus-scope": 1.1.1 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-popper": 1.2.1 + "@radix-ui/react-portal": 1.1.3 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-slot": 1.1.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + "@radix-ui/react-use-controllable-state": 1.1.0 + "@radix-ui/react-use-layout-effect": 1.1.0 + "@radix-ui/react-use-previous": 1.1.0 + "@radix-ui/react-visually-hidden": 1.1.1 + aria-hidden: ^1.1.1 + react-remove-scroll: ^2.6.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 68571da6bebc83f0b685b2b49d804227e787adc72e759523d2d5ce9ee63b51d61de2a9ea92455f33d386eced626d4b2bf73639f4c08d2dbb595f5ec3af47533c + languageName: node + linkType: hard + +"@radix-ui/react-separator@npm:^1.0.3": + version: 1.1.1 + resolution: "@radix-ui/react-separator@npm:1.1.1" + dependencies: + "@radix-ui/react-primitive": 2.0.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 85de387e6b94548d777b5ebafe3f170eb0bab31ec1850ee8bc58f7fc48b725d6cf33c4fb2c92be3196ca843a14819fa32108dd78d5675994e31adb5500484e01 + languageName: node + linkType: hard + +"@radix-ui/react-slider@npm:^1.1.2": + version: 1.2.2 + resolution: "@radix-ui/react-slider@npm:1.2.2" + dependencies: + "@radix-ui/number": 1.1.0 + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-collection": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-direction": 1.1.0 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + "@radix-ui/react-use-layout-effect": 1.1.0 + "@radix-ui/react-use-previous": 1.1.0 + "@radix-ui/react-use-size": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 43acd22d5b0840092463ecd60a1c00db01d38330255b7e27d4ccd754379955ce7e6a305572a2691d34ea77df8904aa89030c5cb2596b0ddd54ceac5a62f916e5 + languageName: node + linkType: hard + +"@radix-ui/react-slot@npm:1.0.2": + version: 1.0.2 + resolution: "@radix-ui/react-slot@npm:1.0.2" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/react-compose-refs": 1.0.1 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: edf5edf435ff594bea7e198bf16d46caf81b6fb559493acad4fa8c308218896136acb16f9b7238c788fd13e94a904f2fd0b6d834e530e4cae94522cdb8f77ce9 + languageName: node + linkType: hard + +"@radix-ui/react-slot@npm:1.1.1, @radix-ui/react-slot@npm:^1.0.2": + version: 1.1.1 + resolution: "@radix-ui/react-slot@npm:1.1.1" + dependencies: + "@radix-ui/react-compose-refs": 1.1.1 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: ac391b921dcde1a71db8307247b36cd6908e0886d7a7b0babeb25158292bc29b61ccfb3f83279bfad11fe1f0f90e3e2f3de93b1174f36d107d77b073fe1a652a + languageName: node + linkType: hard + +"@radix-ui/react-tabs@npm:^1.0.4": + version: 1.1.2 + resolution: "@radix-ui/react-tabs@npm:1.1.2" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-direction": 1.1.0 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-roving-focus": 1.1.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 21c9ed2e65538beebbde6c2d5d47c8e7f8279cacc87e34c7581109f9524a11860375958b65fe72e70cfb0ade9be71bbc27dd4d94b90465966c7875ca6d704116 + languageName: node + linkType: hard + +"@radix-ui/react-toast@npm:^1.1.5": + version: 1.2.4 + resolution: "@radix-ui/react-toast@npm:1.2.4" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-collection": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-dismissable-layer": 1.1.3 + "@radix-ui/react-portal": 1.1.3 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-use-callback-ref": 1.1.0 + "@radix-ui/react-use-controllable-state": 1.1.0 + "@radix-ui/react-use-layout-effect": 1.1.0 + "@radix-ui/react-visually-hidden": 1.1.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: 6479af12ec9ae4f3cdbb8ca66d2926c8323e60fdca57fc3de6af1f41f64bf4cb9e0ea70bdac4e9aac568f1abc2222ebfb67e048d77f36ec10bb4971ae4870b73 + languageName: node + linkType: hard + +"@radix-ui/react-tooltip@npm:^1.0.7": + version: 1.1.6 + resolution: "@radix-ui/react-tooltip@npm:1.1.6" + dependencies: + "@radix-ui/primitive": 1.1.1 + "@radix-ui/react-compose-refs": 1.1.1 + "@radix-ui/react-context": 1.1.1 + "@radix-ui/react-dismissable-layer": 1.1.3 + "@radix-ui/react-id": 1.1.0 + "@radix-ui/react-popper": 1.2.1 + "@radix-ui/react-portal": 1.1.3 + "@radix-ui/react-presence": 1.1.2 + "@radix-ui/react-primitive": 2.0.1 + "@radix-ui/react-slot": 1.1.1 + "@radix-ui/react-use-controllable-state": 1.1.0 + "@radix-ui/react-visually-hidden": 1.1.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: aabbb2c3a7592419fcf41d306582c57307e9518ee29d80e0a8f811bb29ade72144ee35a4f4a120e5143dee46813017fe4e087f351503929553a94e3af986305f + languageName: node + linkType: hard + +"@radix-ui/react-use-callback-ref@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/react-use-callback-ref@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.13.10 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: b9fd39911c3644bbda14a84e4fca080682bef84212b8d8931fcaa2d2814465de242c4cfd8d7afb3020646bead9c5e539d478cea0a7031bee8a8a3bb164f3bc4c + languageName: node + linkType: hard + +"@radix-ui/react-use-callback-ref@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/react-use-callback-ref@npm:1.1.0" + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 2ec7903c67e3034b646005556f44fd975dc5204db6885fc58403e3584f27d95f0b573bc161de3d14fab9fda25150bf3b91f718d299fdfc701c736bd0bd2281fa + languageName: node + linkType: hard + +"@radix-ui/react-use-controllable-state@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/react-use-controllable-state@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/react-use-callback-ref": 1.0.1 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: dee2be1937d293c3a492cb6d279fc11495a8f19dc595cdbfe24b434e917302f9ac91db24e8cc5af9a065f3f209c3423115b5442e65a5be9fd1e9091338972be9 + languageName: node + linkType: hard + +"@radix-ui/react-use-controllable-state@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/react-use-controllable-state@npm:1.1.0" + dependencies: + "@radix-ui/react-use-callback-ref": 1.1.0 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: a6c167cf8eb0744effbeab1f92ea6c0ad71838b222670c0488599f28eecd941d87ac1eed4b5d3b10df6dc7b7b2edb88a54e99d92c2942ce3b21f81d5c188f32d + languageName: node + linkType: hard + +"@radix-ui/react-use-escape-keydown@npm:1.0.3": + version: 1.0.3 + resolution: "@radix-ui/react-use-escape-keydown@npm:1.0.3" + dependencies: + "@babel/runtime": ^7.13.10 + "@radix-ui/react-use-callback-ref": 1.0.1 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: c6ed0d9ce780f67f924980eb305af1f6cce2a8acbaf043a58abe0aa3cc551d9aa76ccee14531df89bbee302ead7ecc7fce330886f82d4672c5eda52f357ef9b8 + languageName: node + linkType: hard + +"@radix-ui/react-use-escape-keydown@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/react-use-escape-keydown@npm:1.1.0" + dependencies: + "@radix-ui/react-use-callback-ref": 1.1.0 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 9bf88ea272b32ea0f292afd336780a59c5646f795036b7e6105df2d224d73c54399ee5265f61d571eb545d28382491a8b02dc436e3088de8dae415d58b959b71 + languageName: node + linkType: hard + +"@radix-ui/react-use-layout-effect@npm:1.0.1": + version: 1.0.1 + resolution: "@radix-ui/react-use-layout-effect@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.13.10 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: bed9c7e8de243a5ec3b93bb6a5860950b0dba359b6680c84d57c7a655e123dec9b5891c5dfe81ab970652e7779fe2ad102a23177c7896dde95f7340817d47ae5 + languageName: node + linkType: hard + +"@radix-ui/react-use-layout-effect@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/react-use-layout-effect@npm:1.1.0" + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 271ea0bf1cd74718895a68414a6e95537737f36e02ad08eeb61a82b229d6abda9cff3135a479e134e1f0ce2c3ff97bb85babbdce751985fb755a39b231d7ccf2 + languageName: node + linkType: hard + +"@radix-ui/react-use-previous@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/react-use-previous@npm:1.1.0" + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 8a2407e3db6248ab52bf425f5f4161355d09f1a228038094959250ae53552e73543532b3bb80e452f6ad624621e2e1c6aebb8c702f2dfaa5e89f07ec629d9304 + languageName: node + linkType: hard + +"@radix-ui/react-use-rect@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/react-use-rect@npm:1.1.0" + dependencies: + "@radix-ui/rect": 1.1.0 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: facc9528af43df3b01952dbb915ff751b5924db2c31d41f053ddea19a7cc5cac5b096c4d7a2059e8f564a3f0d4a95bcd909df8faed52fa01709af27337628e2c + languageName: node + linkType: hard + +"@radix-ui/react-use-size@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/react-use-size@npm:1.1.0" + dependencies: + "@radix-ui/react-use-layout-effect": 1.1.0 + peerDependencies: + "@types/react": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 01a11d4c07fc620b8a081e53d7ec8495b19a11e02688f3d9f47cf41a5fe0428d1e52ed60b2bf88dfd447dc2502797b9dad2841097389126dd108530913c4d90d + languageName: node + linkType: hard + +"@radix-ui/react-visually-hidden@npm:1.1.1": + version: 1.1.1 + resolution: "@radix-ui/react-visually-hidden@npm:1.1.1" + dependencies: + "@radix-ui/react-primitive": 2.0.1 + peerDependencies: + "@types/react": "*" + "@types/react-dom": "*" + react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + "@types/react-dom": + optional: true + checksum: ccbdf29811283fb257f0b0f8604923e6fe349a264986463f6d6a20946fc51e243527985e69f0af27659f78fd7a4199dacbba5bfc7af3667aa409cd23a0ae3283 + languageName: node + linkType: hard + +"@radix-ui/rect@npm:1.1.0": + version: 1.1.0 + resolution: "@radix-ui/rect@npm:1.1.0" + checksum: 1ad93efbc9fc3b878bae5e8bb26ffa1005235d8b5b9fca8339eb5dbcf7bf53abc9ccd2a8ce128557820168c8600521e48e0ea4dda96aa5f116381f66f46aeda3 + languageName: node + linkType: hard + +"@rollup/plugin-node-resolve@npm:^15.0.1": + version: 15.2.3 + resolution: "@rollup/plugin-node-resolve@npm:15.2.3" + dependencies: + "@rollup/pluginutils": ^5.0.1 + "@types/resolve": 1.20.2 + deepmerge: ^4.2.2 + is-builtin-module: ^3.2.1 + is-module: ^1.0.0 + resolve: ^1.22.1 + peerDependencies: + rollup: ^2.78.0||^3.0.0||^4.0.0 + peerDependenciesMeta: + rollup: + optional: true + checksum: 730f32c2f8fdddff07cf0fca86a5dac7c475605fb96930197a868c066e62eb6388c557545e4f7d99b7a283411754c9fbf98944ab086b6074e04fc1292e234aa8 + languageName: node + linkType: hard + +"@rollup/pluginutils@npm:^5.0.1": + version: 5.1.0 + resolution: "@rollup/pluginutils@npm:5.1.0" + dependencies: + "@types/estree": ^1.0.0 + estree-walker: ^2.0.2 + picomatch: ^2.3.1 + peerDependencies: + rollup: ^1.20.0||^2.0.0||^3.0.0||^4.0.0 + peerDependenciesMeta: + rollup: + optional: true + checksum: 3cc5a6d91452a6eabbfd1ae79b4dd1f1e809d2eecda6e175deb784e75b0911f47e9ecce73f8dd315d6a8b3f362582c91d3c0f66908b6ced69345b3cbe28f8ce8 + languageName: node + linkType: hard + +"@rollup/rollup-android-arm-eabi@npm:4.9.4": + version: 4.9.4 + resolution: "@rollup/rollup-android-arm-eabi@npm:4.9.4" + conditions: os=android & cpu=arm + languageName: node + linkType: hard + +"@rollup/rollup-android-arm64@npm:4.9.4": + version: 4.9.4 + resolution: "@rollup/rollup-android-arm64@npm:4.9.4" + conditions: os=android & cpu=arm64 + languageName: node + linkType: hard + +"@rollup/rollup-darwin-arm64@npm:4.9.4": + version: 4.9.4 + resolution: "@rollup/rollup-darwin-arm64@npm:4.9.4" + conditions: os=darwin & cpu=arm64 + languageName: node + linkType: hard + +"@rollup/rollup-darwin-x64@npm:4.9.4": + version: 4.9.4 + resolution: "@rollup/rollup-darwin-x64@npm:4.9.4" + conditions: os=darwin & cpu=x64 + languageName: node + linkType: hard "@rollup/rollup-linux-arm-gnueabihf@npm:4.9.4": version: 4.9.4 @@ -5958,6 +7556,13 @@ __metadata: languageName: node linkType: hard +"@scure/base@npm:~1.1.2": + version: 1.1.9 + resolution: "@scure/base@npm:1.1.9" + checksum: 120820a37dfe9dfe4cab2b7b7460552d08e67dee8057ed5354eb68d8e3440890ae983ce3bee957d2b45684950b454a2b6d71d5ee77c1fd3fddc022e2a510337f + languageName: node + linkType: hard + "@scure/bip32@npm:1.1.5": version: 1.1.5 resolution: "@scure/bip32@npm:1.1.5" @@ -5969,13 +7574,34 @@ __metadata: languageName: node linkType: hard +"@scure/bip32@npm:1.3.2": + version: 1.3.2 + resolution: "@scure/bip32@npm:1.3.2" + dependencies: + "@noble/curves": ~1.2.0 + "@noble/hashes": ~1.3.2 + "@scure/base": ~1.1.2 + checksum: c5ae84fae43490853693b481531132b89e056d45c945fc8b92b9d032577f753dfd79c5a7bbcbf0a7f035951006ff0311b6cf7a389e26c9ec6335e42b20c53157 + languageName: node + linkType: hard + "@scure/bip39@npm:1.1.1": version: 1.1.1 resolution: "@scure/bip39@npm:1.1.1" dependencies: - "@noble/hashes": ~1.2.0 + "@noble/hashes": ~1.2.0 + "@scure/base": ~1.1.0 + checksum: fbb594c50696fa9c14e891d872f382e50a3f919b6c96c55ef2fb10c7102c546dafb8f099a62bd114c12a00525b595dcf7381846f383f0ddcedeaa6e210747d2f + languageName: node + linkType: hard + +"@scure/bip39@npm:1.2.1": + version: 1.2.1 + resolution: "@scure/bip39@npm:1.2.1" + dependencies: + "@noble/hashes": ~1.3.0 "@scure/base": ~1.1.0 - checksum: fbb594c50696fa9c14e891d872f382e50a3f919b6c96c55ef2fb10c7102c546dafb8f099a62bd114c12a00525b595dcf7381846f383f0ddcedeaa6e210747d2f + checksum: c5bd6f1328fdbeae2dcdd891825b1610225310e5e62a4942714db51066866e4f7bef242c7b06a1b9dcc8043a4a13412cf5c5df76d3b10aa9e36b82e9b6e3eeaa languageName: node linkType: hard @@ -6061,6 +7687,90 @@ __metadata: languageName: node linkType: hard +"@shikijs/core@npm:1.26.2": + version: 1.26.2 + resolution: "@shikijs/core@npm:1.26.2" + dependencies: + "@shikijs/engine-javascript": 1.26.2 + "@shikijs/engine-oniguruma": 1.26.2 + "@shikijs/types": 1.26.2 + "@shikijs/vscode-textmate": ^10.0.1 + "@types/hast": ^3.0.4 + hast-util-to-html: ^9.0.4 + checksum: b7bad4c281102bdd74f0974aa780efca06117208419c205005f172b247221d42685608d96dba97bd215eb6af99463d914517bad77fdc507e3254527f08f95975 + languageName: node + linkType: hard + +"@shikijs/engine-javascript@npm:1.26.2": + version: 1.26.2 + resolution: "@shikijs/engine-javascript@npm:1.26.2" + dependencies: + "@shikijs/types": 1.26.2 + "@shikijs/vscode-textmate": ^10.0.1 + oniguruma-to-es: ^1.0.0 + checksum: 8df3d284033b17f50625d07e0cdcd26d24ad8e821bb8e58cc91aebffd28befe7cf80169aadbc5bcc038433d68a2e18148b5098ef4e243f18c9cb634ba20ea034 + languageName: node + linkType: hard + +"@shikijs/engine-oniguruma@npm:1.26.2": + version: 1.26.2 + resolution: "@shikijs/engine-oniguruma@npm:1.26.2" + dependencies: + "@shikijs/types": 1.26.2 + "@shikijs/vscode-textmate": ^10.0.1 + checksum: d2d4978be0b4e8b3b26fd01ea0480568ac2264135704442e54ee14fb3b05f564492f95cae521378d6114641f249e892a76dc01fb0e4ae64522e56b3b353ce08d + languageName: node + linkType: hard + +"@shikijs/langs@npm:1.26.2": + version: 1.26.2 + resolution: "@shikijs/langs@npm:1.26.2" + dependencies: + "@shikijs/types": 1.26.2 + checksum: c3f1882401f19bc50cbf5fb3d62b287c2fc07bdbc98b281d1d63b51dac8602d00b8446918aae73d8e76e455b205542cc9a73526b64fe3c1b020952af8ebe26c5 + languageName: node + linkType: hard + +"@shikijs/rehype@npm:^1.12.1": + version: 1.26.2 + resolution: "@shikijs/rehype@npm:1.26.2" + dependencies: + "@shikijs/types": 1.26.2 + "@types/hast": ^3.0.4 + hast-util-to-string: ^3.0.1 + shiki: 1.26.2 + unified: ^11.0.5 + unist-util-visit: ^5.0.0 + checksum: b1ed82152275ebf8deadd97308b1ead12b91ce451409fce70307b6d6953c9a9c60d18dd1c3d6317039b30df14f33e9c01c42c7f0e149662f94221b110b3a569c + languageName: node + linkType: hard + +"@shikijs/themes@npm:1.26.2": + version: 1.26.2 + resolution: "@shikijs/themes@npm:1.26.2" + dependencies: + "@shikijs/types": 1.26.2 + checksum: 1e11093ae6e4fbf3573ca626dea19e78f07e1eb33a720ef56503e830743e7789e2203203738162f9b6f1bb273fe23b51b545ea3acdd5818b39fb7052475993fb + languageName: node + linkType: hard + +"@shikijs/types@npm:1.26.2": + version: 1.26.2 + resolution: "@shikijs/types@npm:1.26.2" + dependencies: + "@shikijs/vscode-textmate": ^10.0.1 + "@types/hast": ^3.0.4 + checksum: 77e0823a60dce4f37b85b2648ae75f00750fc897b6efaf7f8d765b3e57849456c5bfbe6ad9f1faa269c2ccddf7e4d7ab6dc5b43ab69c70ee09b4bde37f360cc4 + languageName: node + linkType: hard + +"@shikijs/vscode-textmate@npm:^10.0.1": + version: 10.0.1 + resolution: "@shikijs/vscode-textmate@npm:10.0.1" + checksum: c5a8490417b9439b055844c6c09c3435fc435b1fc3923eb28f05ee346fd68e69df2d93cdaab319a51193970558ff1bf49c5ab047c9ed4fd86c3f9d062457a565 + languageName: node + linkType: hard + "@sideway/address@npm:^4.1.3": version: 4.1.4 resolution: "@sideway/address@npm:4.1.4" @@ -6195,6 +7905,34 @@ __metadata: languageName: node linkType: hard +"@statsig/client-core@npm:3.9.0": + version: 3.9.0 + resolution: "@statsig/client-core@npm:3.9.0" + checksum: 2c8b8a99ac4ce1d24fe4e9263f55af37d4861455450dc348787bfa3cf2cf523a12e75a3aaa4fb676a853493174481f7e29d99196bcfd1ce30ffcb48137ff6453 + languageName: node + linkType: hard + +"@statsig/js-client@npm:3.9.0, @statsig/js-client@npm:^3.1.0": + version: 3.9.0 + resolution: "@statsig/js-client@npm:3.9.0" + dependencies: + "@statsig/client-core": 3.9.0 + checksum: 0f715c5043fb529baeadd256f8487f63aa05502cbe51d1f2053bcedf306638dc831eec05b0ad31522c05d28edabd854f9da4a4844e48b9fc27109e089679fb47 + languageName: node + linkType: hard + +"@statsig/react-bindings@npm:^3.1.0": + version: 3.9.0 + resolution: "@statsig/react-bindings@npm:3.9.0" + dependencies: + "@statsig/client-core": 3.9.0 + "@statsig/js-client": 3.9.0 + peerDependencies: + react: ^16.6.3 || ^17.0.0 || ^18.0.0 || ^19.0.0 + checksum: e6b2cb68e3ca720c1714c5b8bc1f1b67fbb2fff7eab07b7395b4c32e64957877233158e29f38129a404ba3ede670490db665b22b8d4c1ce3fda71a95866008e7 + languageName: node + linkType: hard + "@svgr/babel-plugin-add-jsx-attribute@npm:8.0.0": version: 8.0.0 resolution: "@svgr/babel-plugin-add-jsx-attribute@npm:8.0.0" @@ -6507,6 +8245,53 @@ __metadata: languageName: node linkType: hard +"@tailwindcss/line-clamp@npm:^0.4.4": + version: 0.4.4 + resolution: "@tailwindcss/line-clamp@npm:0.4.4" + peerDependencies: + tailwindcss: ">=2.0.0 || >=3.0.0 || >=3.0.0-alpha.1" + checksum: 3d2ad992aa9263fe9b5cdb23bcfca521a6ab00f65e0f7167be35d2cb46b1635af72889ff9f6d5b2febf5aa5a36e3128eaad8ed43e43af4512c74c74f1058c4c0 + languageName: node + linkType: hard + +"@tanstack/react-table@npm:^8.11.7": + version: 8.20.6 + resolution: "@tanstack/react-table@npm:8.20.6" + dependencies: + "@tanstack/table-core": 8.20.5 + peerDependencies: + react: ">=16.8" + react-dom: ">=16.8" + checksum: 709f7216cf31af65abe10121e92233c0d9f04b9167769a0997052200cdbc35af525cbf1879d456f27c6df70399a14d006d117b06c5db665d9c486038efc2f4b1 + languageName: node + linkType: hard + +"@tanstack/react-virtual@npm:^3.0.0-beta.60": + version: 3.11.2 + resolution: "@tanstack/react-virtual@npm:3.11.2" + dependencies: + "@tanstack/virtual-core": 3.11.2 + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + checksum: a1136da0ec4c2ecbd4f996d8b84f228f0b8d851b15806e01049a160ad1d9b2eef0e0a491035fe017c6f84a0e125334f69ea23b32c180df23614ea4a8eeb7490c + languageName: node + linkType: hard + +"@tanstack/table-core@npm:8.20.5": + version: 8.20.5 + resolution: "@tanstack/table-core@npm:8.20.5" + checksum: f8b175f11eb9ee1e029bb5e91c1038527714382de4bd14750377f43c25e69b687b21bfb181ee07131637d3432618964a4b7a898608fc8411ca50da1e7e8ed4c5 + languageName: node + linkType: hard + +"@tanstack/virtual-core@npm:3.11.2": + version: 3.11.2 + resolution: "@tanstack/virtual-core@npm:3.11.2" + checksum: b5c91662461e3edd1cba0efbaa89e1d061c8bb605bb78d1e87e2a687335c740a731c96a81798b05491df4882ff2fbd27b312f5e7440e4f9d553a81fb2283156a + languageName: node + linkType: hard + "@tootallnate/quickjs-emscripten@npm:^0.23.0": version: 0.23.0 resolution: "@tootallnate/quickjs-emscripten@npm:0.23.0" @@ -6698,6 +8483,75 @@ __metadata: languageName: node linkType: hard +"@types/d3-array@npm:^3.0.3": + version: 3.2.1 + resolution: "@types/d3-array@npm:3.2.1" + checksum: 8a41cee0969e53bab3f56cc15c4e6c9d76868d6daecb2b7d8c9ce71e0ececccc5a8239697cc52dadf5c665f287426de5c8ef31a49e7ad0f36e8846889a383df4 + languageName: node + linkType: hard + +"@types/d3-color@npm:*": + version: 3.1.3 + resolution: "@types/d3-color@npm:3.1.3" + checksum: 8a0e79a709929502ec4effcee2c786465b9aec51b653ba0b5d05dbfec3e84f418270dd603002d94021885061ff592f614979193bd7a02ad76317f5608560e357 + languageName: node + linkType: hard + +"@types/d3-ease@npm:^3.0.0": + version: 3.0.2 + resolution: "@types/d3-ease@npm:3.0.2" + checksum: 0885219966294bfc99548f37297e1c75e75da812a5f3ec941977ebb57dcab0a25acec5b2bbd82d09a49d387daafca08521ca269b7e4c27ddca7768189e987b54 + languageName: node + linkType: hard + +"@types/d3-interpolate@npm:^3.0.1": + version: 3.0.4 + resolution: "@types/d3-interpolate@npm:3.0.4" + dependencies: + "@types/d3-color": "*" + checksum: efd2770e174e84fc7316fdafe03cf3688451f767dde1fa6211610137f495be7f3923db7e1723a6961a0e0e9ae0ed969f4f47c038189fa0beb1d556b447922622 + languageName: node + linkType: hard + +"@types/d3-path@npm:*": + version: 3.1.0 + resolution: "@types/d3-path@npm:3.1.0" + checksum: 1e81b56ed33ba1ac954a8c42c78c3fcf2716927fe5d01b2003591193ad3b639572a3dfcedd9bf78b6b73215a5cfb01cede8f25c936e95ac18fbe3858f9b62f5c + languageName: node + linkType: hard + +"@types/d3-scale@npm:^4.0.2": + version: 4.0.8 + resolution: "@types/d3-scale@npm:4.0.8" + dependencies: + "@types/d3-time": "*" + checksum: 3b1906da895564f73bb3d0415033d9a8aefe7c4f516f970176d5b2ff7a417bd27ae98486e9a9aa0472001dc9885a9204279a1973a985553bdb3ee9bbc1b94018 + languageName: node + linkType: hard + +"@types/d3-shape@npm:^3.1.0": + version: 3.1.7 + resolution: "@types/d3-shape@npm:3.1.7" + dependencies: + "@types/d3-path": "*" + checksum: 776b982e2c4fc04763782af5100993c02bca338632ff2c76d2423ace398300ba7c48cd745f95b5f51edefabbfd026c45829a146c411f8facde09ef92580b20ce + languageName: node + linkType: hard + +"@types/d3-time@npm:*, @types/d3-time@npm:^3.0.0": + version: 3.0.4 + resolution: "@types/d3-time@npm:3.0.4" + checksum: 0c296884571ce70c4bbd4ea9cd1c93c0c8aee602c6c806b056187dd4ee49daf70c2f41da94b25ba0d796edf8ca83cbb87fe6d1cdda7ca669ab800170ece1c12b + languageName: node + linkType: hard + +"@types/d3-timer@npm:^3.0.0": + version: 3.0.2 + resolution: "@types/d3-timer@npm:3.0.2" + checksum: 1643eebfa5f4ae3eb00b556bbc509444d88078208ec2589ddd8e4a24f230dd4cf2301e9365947e70b1bee33f63aaefab84cd907822aae812b9bc4871b98ab0e1 + languageName: node + linkType: hard + "@types/debounce@npm:^1.2.0": version: 1.2.4 resolution: "@types/debounce@npm:1.2.4" @@ -6799,6 +8653,15 @@ __metadata: languageName: node linkType: hard +"@types/hast@npm:^3.0.4": + version: 3.0.4 + resolution: "@types/hast@npm:3.0.4" + dependencies: + "@types/unist": "*" + checksum: 7a973e8d16fcdf3936090fa2280f408fb2b6a4f13b42edeb5fbd614efe042b82eac68e298e556d50f6b4ad585a3a93c353e9c826feccdc77af59de8dd400d044 + languageName: node + linkType: hard + "@types/history@npm:^4.7.11": version: 4.7.11 resolution: "@types/history@npm:4.7.11" @@ -7284,6 +9147,13 @@ __metadata: languageName: node linkType: hard +"@types/stylis@npm:4.2.5": + version: 4.2.5 + resolution: "@types/stylis@npm:4.2.5" + checksum: 24f91719db5569979e9e2f197e050ef82e1fd72474e8dc45bca38d48ee56481eae0f0d4a7ac172540d7774b45a2a78d901a4c6d07bba77a33dbccff464ea3edf + languageName: node + linkType: hard + "@types/unist@npm:*, @types/unist@npm:^3.0.0": version: 3.0.2 resolution: "@types/unist@npm:3.0.2" @@ -7470,6 +9340,13 @@ __metadata: languageName: node linkType: hard +"@vercel/edge@npm:^1.1.1": + version: 1.2.1 + resolution: "@vercel/edge@npm:1.2.1" + checksum: 3cddadf0bd9d733400cab5bdd73bf5c0fcba02edcfa5e8cc483491e9d5b31791667601ffa9b566b375d463f05d1e0408c39fd6a03044f289736569fde065c99b + languageName: node + linkType: hard + "@wasm-tool/wasm-pack-plugin@npm:^1.7.0": version: 1.7.0 resolution: "@wasm-tool/wasm-pack-plugin@npm:1.7.0" @@ -7965,6 +9842,21 @@ __metadata: languageName: node linkType: hard +"abitype@npm:1.0.0": + version: 1.0.0 + resolution: "abitype@npm:1.0.0" + peerDependencies: + typescript: ">=5.0.4" + zod: ^3 >=3.22.0 + peerDependenciesMeta: + typescript: + optional: true + zod: + optional: true + checksum: ea2c0548c3ba58c37a6de7483d63389074da498e63d803b742bbe94eb4eaa1f51a35d000c424058b2583aef56698cf07c696eb3bc4dd0303bc20c6f0826a241a + languageName: node + linkType: hard + "abort-controller@npm:^3.0.0": version: 3.0.0 resolution: "abort-controller@npm:3.0.0" @@ -8266,6 +10158,13 @@ __metadata: languageName: node linkType: hard +"ansi-styles@npm:^5.0.0": + version: 5.2.0 + resolution: "ansi-styles@npm:5.2.0" + checksum: d7f4e97ce0623aea6bc0d90dcd28881ee04cba06c570b97fd3391bd7a268eedfd9d5e2dd4fdcbdd82b8105df5faf6f24aaedc08eaf3da898e702db5948f63469 + languageName: node + linkType: hard + "ansi-styles@npm:^6.1.0": version: 6.2.1 resolution: "ansi-styles@npm:6.2.1" @@ -8320,6 +10219,15 @@ __metadata: languageName: node linkType: hard +"aria-hidden@npm:^1.1.1": + version: 1.2.4 + resolution: "aria-hidden@npm:1.2.4" + dependencies: + tslib: ^2.0.0 + checksum: 2ac90b70d29c6349d86d90e022cf01f4885f9be193932d943a14127cf28560dd0baf068a6625f084163437a4be0578f513cf7892f4cc63bfe91aa41dce27c6b2 + languageName: node + linkType: hard + "array-back@npm:^3.0.1, array-back@npm:^3.1.0": version: 3.1.0 resolution: "array-back@npm:3.1.0" @@ -8732,6 +10640,17 @@ __metadata: languageName: node linkType: hard +"bcp-47@npm:^2.1.0": + version: 2.1.0 + resolution: "bcp-47@npm:2.1.0" + dependencies: + is-alphabetical: ^2.0.0 + is-alphanumerical: ^2.0.0 + is-decimal: ^2.0.0 + checksum: 2ae12b551f0ef4da3684617d12941430091efa1114a89028f6ee05eba3df06e314cca2988cde43e7d66dc9d5799eac1201556b0c3a5df99efe514928144bab1e + languageName: node + linkType: hard + "big.js@npm:^5.2.2": version: 5.2.2 resolution: "big.js@npm:5.2.2" @@ -9191,6 +11110,62 @@ __metadata: languageName: node linkType: hard +"camelize@npm:^1.0.0": + version: 1.0.1 + resolution: "camelize@npm:1.0.1" + checksum: 91d8611d09af725e422a23993890d22b2b72b4cabf7239651856950c76b4bf53fe0d0da7c5e4db05180e898e4e647220e78c9fbc976113bd96d603d1fcbfcb99 + languageName: node + linkType: hard + +"can-dom-mutate@npm:^2.0.9": + version: 2.0.9 + resolution: "can-dom-mutate@npm:2.0.9" + dependencies: + can-globals: ^1.0.0 + can-namespace: 1.0.0 + can-reflect: ^1.17.6 + can-symbol: ^1.6.4 + checksum: 60165c5671335bcea1a1f432865ff5ee0e4e1b4980f1ff06eacea405a597d6605e106da7920d2d9618b18da0d0238a75163b81c6f9f92427c99474bfd16c689c + languageName: node + linkType: hard + +"can-globals@npm:^1.0.0": + version: 1.2.2 + resolution: "can-globals@npm:1.2.2" + dependencies: + can-namespace: ^1.0.0 + can-reflect: ^1.2.6 + can-symbol: ^1.0.0 + checksum: f7cdb19227e4118923ea09ab24d20e3be23a72973ca2f181c4d9da2949b23cfea14566b91f865cda2c50520bd774060eeca0b1ea360df2131b3f55d9ca346865 + languageName: node + linkType: hard + +"can-namespace@npm:1.0.0, can-namespace@npm:^1.0.0": + version: 1.0.0 + resolution: "can-namespace@npm:1.0.0" + checksum: 93a27be2c89bde52941bc18b606399c0cbbed596ac525f611a6fc66b7f684e678f4675ac9a53d0e25457159762e656505d23b55fdb1d116c30a7d78d18723577 + languageName: node + linkType: hard + +"can-reflect@npm:^1.17.6, can-reflect@npm:^1.2.6": + version: 1.19.2 + resolution: "can-reflect@npm:1.19.2" + dependencies: + can-namespace: ^1.0.0 + can-symbol: ^1.7.0 + checksum: 7022a5dc8b182beda745a7e3c33b67d9d7fd81bd75277e8db19714c54cfc485bce0899c5e2f879f993d484e7dba03494ff8d5494dd69ef28fb70462846050a5e + languageName: node + linkType: hard + +"can-symbol@npm:^1.0.0, can-symbol@npm:^1.6.4, can-symbol@npm:^1.7.0": + version: 1.7.0 + resolution: "can-symbol@npm:1.7.0" + dependencies: + can-namespace: ^1.0.0 + checksum: b51dc2abfb8cfc75cf1057fcef32317a278904860018e2cdec7852d23ad310cacb5d5ebf216030bf0ed1a6eae65eb12bca83f9e78ea81f33eb195d5e38cd8cf1 + languageName: node + linkType: hard + "caniuse-api@npm:^3.0.0": version: 3.0.0 resolution: "caniuse-api@npm:3.0.0" @@ -9568,6 +11543,13 @@ __metadata: languageName: node linkType: hard +"client-only@npm:^0.0.1": + version: 0.0.1 + resolution: "client-only@npm:0.0.1" + checksum: 0c16bf660dadb90610553c1d8946a7fdfb81d624adea073b8440b7d795d5b5b08beb3c950c6a2cf16279365a3265158a236876d92bce16423c485c322d7dfaf8 + languageName: node + linkType: hard + "clipboardy@npm:3.0.0": version: 3.0.0 resolution: "clipboardy@npm:3.0.0" @@ -9649,6 +11631,28 @@ __metadata: languageName: node linkType: hard +"clsx@npm:^2.1.0": + version: 2.1.1 + resolution: "clsx@npm:2.1.1" + checksum: acd3e1ab9d8a433ecb3cc2f6a05ab95fe50b4a3cfc5ba47abb6cbf3754585fcb87b84e90c822a1f256c4198e3b41c7f6c391577ffc8678ad587fc0976b24fd57 + languageName: node + linkType: hard + +"cmdk-react17@npm:^1.0.0": + version: 1.0.0 + resolution: "cmdk-react17@npm:1.0.0" + dependencies: + "@radix-ui/react-dialog": 1.0.5 + "@radix-ui/react-id": ^1.0.1 + "@radix-ui/react-primitive": 1.0.3 + use-sync-external-store: ^1.2.0 + peerDependencies: + react: ^17.0.0 + react-dom: ^17.0.0 + checksum: d10812fef42f0d9bd34aa66dbe7b294148ae7b2404e934e89f1fbe9de0b0f28b8f30bf38a79f6928e8eed37340aa7660cf45a4527b68d040b8c44c268de5dc24 + languageName: node + linkType: hard + "co-body@npm:^6.1.0": version: 6.1.0 resolution: "co-body@npm:6.1.0" @@ -9714,6 +11718,13 @@ __metadata: languageName: node linkType: hard +"color2k@npm:^2.0.3": + version: 2.0.3 + resolution: "color2k@npm:2.0.3" + checksum: 0748e16e43c1740045af61f44de5d181f3f7a2a9cb0d5cccdccee23d04e3f107f02aaafebb7ca3335cca1d11849c6321aba702eb2f893e993a77f65761de7661 + languageName: node + linkType: hard + "colord@npm:^2.9.1, colord@npm:^2.9.3": version: 2.9.3 resolution: "colord@npm:2.9.3" @@ -10110,6 +12121,13 @@ __metadata: languageName: node linkType: hard +"core-js@npm:^3.38.1": + version: 3.40.0 + resolution: "core-js@npm:3.40.0" + checksum: fc962b93470fd4a129555c765b630c1741fc38706bca68779879f0feaef3b6eec11a33904e3111b2b0e8ba206e8cfbc2a70193271227cfa2f2d13a986f78e557 + languageName: node + linkType: hard + "core-util-is@npm:^1.0.3, core-util-is@npm:~1.0.0": version: 1.0.3 resolution: "core-util-is@npm:1.0.3" @@ -10143,7 +12161,7 @@ __metadata: languageName: node linkType: hard -"cosmiconfig@npm:^8.1.3, cosmiconfig@npm:^8.2.0": +"cosmiconfig@npm:^8.0.0, cosmiconfig@npm:^8.1.3, cosmiconfig@npm:^8.2.0": version: 8.3.6 resolution: "cosmiconfig@npm:8.3.6" dependencies: @@ -10365,6 +12383,13 @@ __metadata: languageName: node linkType: hard +"css-color-keywords@npm:^1.0.0": + version: 1.0.0 + resolution: "css-color-keywords@npm:1.0.0" + checksum: 8f125e3ad477bd03c77b533044bd9e8a6f7c0da52d49bbc0bbe38327b3829d6ba04d368ca49dd9ff3b667d2fc8f1698d891c198bbf8feade1a5501bf5a296408 + languageName: node + linkType: hard + "css-declaration-sorter@npm:^6.3.1": version: 6.4.1 resolution: "css-declaration-sorter@npm:6.4.1" @@ -10485,6 +12510,17 @@ __metadata: languageName: node linkType: hard +"css-to-react-native@npm:3.2.0": + version: 3.2.0 + resolution: "css-to-react-native@npm:3.2.0" + dependencies: + camelize: ^1.0.0 + css-color-keywords: ^1.0.0 + postcss-value-parser: ^4.0.2 + checksum: 263be65e805aef02c3f20c064665c998a8c35293e1505dbe6e3054fb186b01a9897ac6cf121f9840e5a9dfe3fb3994f6fcd0af84a865f1df78ba5bf89e77adce + languageName: node + linkType: hard + "css-tree@npm:^1.1.2, css-tree@npm:^1.1.3": version: 1.1.3 resolution: "css-tree@npm:1.1.3" @@ -10699,15 +12735,108 @@ __metadata: version: 5.0.5 resolution: "csso@npm:5.0.5" dependencies: - css-tree: ~2.2.0 - checksum: 0ad858d36bf5012ed243e9ec69962a867509061986d2ee07cc040a4b26e4d062c00d4c07e5ba8d430706ceb02dd87edd30a52b5937fd45b1b6f2119c4993d59a + css-tree: ~2.2.0 + checksum: 0ad858d36bf5012ed243e9ec69962a867509061986d2ee07cc040a4b26e4d062c00d4c07e5ba8d430706ceb02dd87edd30a52b5937fd45b1b6f2119c4993d59a + languageName: node + linkType: hard + +"csstype@npm:3.1.3, csstype@npm:^3.0.2": + version: 3.1.3 + resolution: "csstype@npm:3.1.3" + checksum: 8db785cc92d259102725b3c694ec0c823f5619a84741b5c7991b8ad135dfaa66093038a1cc63e03361a6cd28d122be48f2106ae72334e067dd619a51f49eddf7 + languageName: node + linkType: hard + +"d3-array@npm:2 - 3, d3-array@npm:2.10.0 - 3, d3-array@npm:^3.1.6": + version: 3.2.4 + resolution: "d3-array@npm:3.2.4" + dependencies: + internmap: 1 - 2 + checksum: a5976a6d6205f69208478bb44920dd7ce3e788c9dceb86b304dbe401a4bfb42ecc8b04c20facde486e9adcb488b5d1800d49393a3f81a23902b68158e12cddd0 + languageName: node + linkType: hard + +"d3-color@npm:1 - 3": + version: 3.1.0 + resolution: "d3-color@npm:3.1.0" + checksum: 4931fbfda5d7c4b5cfa283a13c91a954f86e3b69d75ce588d06cde6c3628cebfc3af2069ccf225e982e8987c612aa7948b3932163ce15eb3c11cd7c003f3ee3b + languageName: node + linkType: hard + +"d3-ease@npm:^3.0.1": + version: 3.0.1 + resolution: "d3-ease@npm:3.0.1" + checksum: 06e2ee5326d1e3545eab4e2c0f84046a123dcd3b612e68858219aa034da1160333d9ce3da20a1d3486d98cb5c2a06f7d233eee1bc19ce42d1533458bd85dedcd + languageName: node + linkType: hard + +"d3-format@npm:1 - 3": + version: 3.1.0 + resolution: "d3-format@npm:3.1.0" + checksum: f345ec3b8ad3cab19bff5dead395bd9f5590628eb97a389b1dd89f0b204c7c4fc1d9520f13231c2c7cf14b7c9a8cf10f8ef15bde2befbab41454a569bd706ca2 + languageName: node + linkType: hard + +"d3-interpolate@npm:1.2.0 - 3, d3-interpolate@npm:^3.0.1": + version: 3.0.1 + resolution: "d3-interpolate@npm:3.0.1" + dependencies: + d3-color: 1 - 3 + checksum: a42ba314e295e95e5365eff0f604834e67e4a3b3c7102458781c477bd67e9b24b6bb9d8e41ff5521050a3f2c7c0c4bbbb6e187fd586daa3980943095b267e78b + languageName: node + linkType: hard + +"d3-path@npm:^3.1.0": + version: 3.1.0 + resolution: "d3-path@npm:3.1.0" + checksum: 2306f1bd9191e1eac895ec13e3064f732a85f243d6e627d242a313f9777756838a2215ea11562f0c7630c7c3b16a19ec1fe0948b1c82f3317fac55882f6ee5d8 + languageName: node + linkType: hard + +"d3-scale@npm:^4.0.2": + version: 4.0.2 + resolution: "d3-scale@npm:4.0.2" + dependencies: + d3-array: 2.10.0 - 3 + d3-format: 1 - 3 + d3-interpolate: 1.2.0 - 3 + d3-time: 2.1.1 - 3 + d3-time-format: 2 - 4 + checksum: a9c770d283162c3bd11477c3d9d485d07f8db2071665f1a4ad23eec3e515e2cefbd369059ec677c9ac849877d1a765494e90e92051d4f21111aa56791c98729e + languageName: node + linkType: hard + +"d3-shape@npm:^3.1.0": + version: 3.2.0 + resolution: "d3-shape@npm:3.2.0" + dependencies: + d3-path: ^3.1.0 + checksum: de2af5fc9a93036a7b68581ca0bfc4aca2d5a328aa7ba7064c11aedd44d24f310c20c40157cb654359d4c15c3ef369f95ee53d71221017276e34172c7b719cfa + languageName: node + linkType: hard + +"d3-time-format@npm:2 - 4": + version: 4.1.0 + resolution: "d3-time-format@npm:4.1.0" + dependencies: + d3-time: 1 - 3 + checksum: 7342bce28355378152bbd4db4e275405439cabba082d9cd01946d40581140481c8328456d91740b0fe513c51ec4a467f4471ffa390c7e0e30ea30e9ec98fcdf4 + languageName: node + linkType: hard + +"d3-time@npm:1 - 3, d3-time@npm:2.1.1 - 3, d3-time@npm:^3.0.0": + version: 3.1.0 + resolution: "d3-time@npm:3.1.0" + dependencies: + d3-array: 2 - 3 + checksum: 613b435352a78d9f31b7f68540788186d8c331b63feca60ad21c88e9db1989fe888f97f242322ebd6365e45ec3fb206a4324cd4ca0dfffa1d9b5feb856ba00a7 languageName: node linkType: hard -"csstype@npm:^3.0.2": - version: 3.1.3 - resolution: "csstype@npm:3.1.3" - checksum: 8db785cc92d259102725b3c694ec0c823f5619a84741b5c7991b8ad135dfaa66093038a1cc63e03361a6cd28d122be48f2106ae72334e067dd619a51f49eddf7 +"d3-timer@npm:^3.0.1": + version: 3.0.1 + resolution: "d3-timer@npm:3.0.1" + checksum: 1cfddf86d7bca22f73f2c427f52dfa35c49f50d64e187eb788dcad6e927625c636aa18ae4edd44d084eb9d1f81d8ca4ec305dae7f733c15846a824575b789d73 languageName: node linkType: hard @@ -10762,6 +12891,13 @@ __metadata: languageName: node linkType: hard +"decimal.js-light@npm:^2.4.1": + version: 2.5.1 + resolution: "decimal.js-light@npm:2.5.1" + checksum: f5a2c7eac1c4541c8ab8a5c8abea64fc1761cefc7794bd5f8afd57a8a78d1b51785e0c4e4f85f4895a043eaa90ddca1edc3981d1263eb6ddce60f32bf5fe66c9 + languageName: node + linkType: hard + "decode-named-character-reference@npm:^1.0.0": version: 1.0.2 resolution: "decode-named-character-reference@npm:1.0.2" @@ -10973,7 +13109,7 @@ __metadata: languageName: node linkType: hard -"dequal@npm:^2.0.0": +"dequal@npm:^2.0.0, dequal@npm:^2.0.3": version: 2.0.3 resolution: "dequal@npm:2.0.3" checksum: 8679b850e1a3d0ebbc46ee780d5df7b478c23f335887464023a631d1b9af051ad4a6595a44220f9ff8ff95a8ddccf019b5ad778a976fd7bbf77383d36f412f90 @@ -10996,6 +13132,13 @@ __metadata: languageName: node linkType: hard +"detect-node-es@npm:^1.1.0": + version: 1.1.0 + resolution: "detect-node-es@npm:1.1.0" + checksum: e46307d7264644975b71c104b9f028ed1d3d34b83a15b8a22373640ce5ea630e5640b1078b8ea15f202b54641da71e4aa7597093bd4b91f113db520a26a37449 + languageName: node + linkType: hard + "detect-node@npm:^2.0.4": version: 2.1.0 resolution: "detect-node@npm:2.1.0" @@ -11095,6 +13238,7 @@ __metadata: version: 0.0.0-use.local resolution: "docs@workspace:docs" dependencies: + "@cookbookdev/docsbot": ^4.24.9 "@docusaurus/core": ^3.5.2 "@docusaurus/module-type-aliases": ^3.5.2 "@docusaurus/preset-classic": ^3.5.2 @@ -11157,6 +13301,16 @@ __metadata: languageName: node linkType: hard +"dom-helpers@npm:^5.0.1": + version: 5.2.1 + resolution: "dom-helpers@npm:5.2.1" + dependencies: + "@babel/runtime": ^7.8.7 + csstype: ^3.0.2 + checksum: 863ba9e086f7093df3376b43e74ce4422571d404fc9828bf2c56140963d5edf0e56160f9b2f3bb61b282c07f8fc8134f023c98fd684bddcb12daf7b0f14d951c + languageName: node + linkType: hard + "dom-serializer@npm:^1.0.1": version: 1.4.1 resolution: "dom-serializer@npm:1.4.1" @@ -11311,6 +13465,13 @@ __metadata: languageName: node linkType: hard +"emoji-regex-xs@npm:^1.0.0": + version: 1.0.0 + resolution: "emoji-regex-xs@npm:1.0.0" + checksum: c33be159da769836f83281f2802d90169093ebf3c2c1643d6801d891c53beac5ef785fd8279f9b02fa6dc6c47c367818e076949f1e13bd1b3f921b416de4cbea + languageName: node + linkType: hard + "emoji-regex@npm:^8.0.0": version: 8.0.0 resolution: "emoji-regex@npm:8.0.0" @@ -12043,7 +14204,7 @@ __metadata: languageName: node linkType: hard -"eventemitter3@npm:^4.0.0": +"eventemitter3@npm:^4.0.0, eventemitter3@npm:^4.0.1": version: 4.0.7 resolution: "eventemitter3@npm:4.0.7" checksum: 1875311c42fcfe9c707b2712c32664a245629b42bb0a5a84439762dd0fd637fc54d078155ea83c2af9e0323c9ac13687e03cfba79b03af9f40c89b4960099374 @@ -12280,6 +14441,13 @@ __metadata: languageName: node linkType: hard +"fflate@npm:^0.4.8": + version: 0.4.8 + resolution: "fflate@npm:0.4.8" + checksum: 29d8cbe44d5e7f53e7f5a160ac7f9cc025480c7b3bfd85c5f898cbe20dfa2dad4732daa534982664bf30b35896a90af44ea33ede5d94c5ffd1b8b0c0a0a56ca2 + languageName: node + linkType: hard + "fflate@npm:^0.8.0": version: 0.8.1 resolution: "fflate@npm:0.8.1" @@ -12287,6 +14455,13 @@ __metadata: languageName: node linkType: hard +"fflate@npm:^0.8.2": + version: 0.8.2 + resolution: "fflate@npm:0.8.2" + checksum: 29470337b85d3831826758e78f370e15cda3169c5cd4477c9b5eea2402261a74b2975bae816afabe1c15d21d98591e0d30a574f7103aa117bff60756fa3035d4 + languageName: node + linkType: hard + "file-entry-cache@npm:^6.0.1": version: 6.0.1 resolution: "file-entry-cache@npm:6.0.1" @@ -12317,6 +14492,13 @@ __metadata: languageName: node linkType: hard +"file-saver@npm:^2.0.5": + version: 2.0.5 + resolution: "file-saver@npm:2.0.5" + checksum: c62d96e5cebc58b4bdf3ae8a60d5cf9607ad82f75f798c33a4ee63435ac2203002584d5256a2a780eda7feb5e19dc3b6351c2212e58b3f529e63d265a7cc79f7 + languageName: node + linkType: hard + "filesize@npm:^8.0.6": version: 8.0.7 resolution: "filesize@npm:8.0.7" @@ -12591,6 +14773,36 @@ __metadata: languageName: node linkType: hard +"framer-motion@npm:^6.5.1": + version: 6.5.1 + resolution: "framer-motion@npm:6.5.1" + dependencies: + "@emotion/is-prop-valid": ^0.8.2 + "@motionone/dom": 10.12.0 + framesync: 6.0.1 + hey-listen: ^1.0.8 + popmotion: 11.0.3 + style-value-types: 5.0.0 + tslib: ^2.1.0 + peerDependencies: + react: ">=16.8 || ^17.0.0 || ^18.0.0" + react-dom: ">=16.8 || ^17.0.0 || ^18.0.0" + dependenciesMeta: + "@emotion/is-prop-valid": + optional: true + checksum: 737959063137b4ccafe01e0ac0c9e5a9531bf3f729f62c34ca7a5d7955e6664f70affd22b044f7db51df41acb21d120a4f71a860e17a80c4db766ad66f2153a1 + languageName: node + linkType: hard + +"framesync@npm:6.0.1": + version: 6.0.1 + resolution: "framesync@npm:6.0.1" + dependencies: + tslib: ^2.1.0 + checksum: a23ebe8f7e20a32c0b99c2f8175b6f07af3ec6316aad52a2316316a6d011d717af8d2175dcc2827031c59fabb30232ed3e19a720a373caba7f070e1eae436325 + languageName: node + linkType: hard + "fresh@npm:0.5.2, fresh@npm:~0.5.2": version: 0.5.2 resolution: "fresh@npm:0.5.2" @@ -12780,6 +14992,13 @@ __metadata: languageName: node linkType: hard +"get-nonce@npm:^1.0.0": + version: 1.0.1 + resolution: "get-nonce@npm:1.0.1" + checksum: e2614e43b4694c78277bb61b0f04583d45786881289285c73770b07ded246a98be7e1f78b940c80cbe6f2b07f55f0b724e6db6fd6f1bcbd1e8bdac16521074ed + languageName: node + linkType: hard + "get-own-enumerable-property-symbols@npm:^3.0.0": version: 3.0.2 resolution: "get-own-enumerable-property-symbols@npm:3.0.2" @@ -13448,6 +15667,25 @@ __metadata: languageName: node linkType: hard +"hast-util-to-html@npm:^9.0.4": + version: 9.0.4 + resolution: "hast-util-to-html@npm:9.0.4" + dependencies: + "@types/hast": ^3.0.0 + "@types/unist": ^3.0.0 + ccount: ^2.0.0 + comma-separated-tokens: ^2.0.0 + hast-util-whitespace: ^3.0.0 + html-void-elements: ^3.0.0 + mdast-util-to-hast: ^13.0.0 + property-information: ^6.0.0 + space-separated-tokens: ^2.0.0 + stringify-entities: ^4.0.0 + zwitch: ^2.0.4 + checksum: 6b97f641bca4c1de66bd74dd5a965bc5fd5c4b8e09328448c4952226ebd691c107cc990ce4e29ccb1e6bfff0278d8956fc8159533456c167f94ae067b4b42b11 + languageName: node + linkType: hard + "hast-util-to-jsx-runtime@npm:^2.0.0": version: 2.3.0 resolution: "hast-util-to-jsx-runtime@npm:2.3.0" @@ -13499,6 +15737,15 @@ __metadata: languageName: node linkType: hard +"hast-util-to-string@npm:^3.0.1": + version: 3.0.1 + resolution: "hast-util-to-string@npm:3.0.1" + dependencies: + "@types/hast": ^3.0.0 + checksum: 556f3cb118fc09e3a6cd149ee4b4056a49028a3858a7d37617e4c6d2c9c5e2336d5fb87eb5f41211b1977a964c705aa70e419464c12debc1959ed03fdad5bed6 + languageName: node + linkType: hard + "hast-util-to-text@npm:^4.0.0": version: 4.0.0 resolution: "hast-util-to-text@npm:4.0.0" @@ -13555,6 +15802,13 @@ __metadata: languageName: node linkType: hard +"hey-listen@npm:^1.0.8": + version: 1.0.8 + resolution: "hey-listen@npm:1.0.8" + checksum: 6bad60b367688f5348e25e7ca3276a74b59ac5a09b0455e6ff8ab7d4a9e38cd2116c708a7dcd8a954d27253ce1d8717ec891d175723ea739885b828cf44e4072 + languageName: node + linkType: hard + "history@npm:^4.9.0": version: 4.10.1 resolution: "history@npm:4.10.1" @@ -13936,6 +16190,13 @@ __metadata: languageName: node linkType: hard +"idb@npm:^8.0.0": + version: 8.0.1 + resolution: "idb@npm:8.0.1" + checksum: 1ba2a1896920bcbfa5b7f826e2d152b0af46415711b74f32649322885c32b94a93fe6b0b78187edda5e77ca77790b74d0e7324a7b14893e0bd4842efcc9bedc2 + languageName: node + linkType: hard + "ieee754@npm:^1.1.13, ieee754@npm:^1.2.1": version: 1.2.1 resolution: "ieee754@npm:1.2.1" @@ -13968,6 +16229,13 @@ __metadata: languageName: node linkType: hard +"immediate@npm:~3.0.5": + version: 3.0.6 + resolution: "immediate@npm:3.0.6" + checksum: f9b3486477555997657f70318cc8d3416159f208bec4cca3ff3442fd266bc23f50f0c9bd8547e1371a6b5e82b821ec9a7044a4f7b944798b25aa3cc6d5e63e62 + languageName: node + linkType: hard + "immer@npm:^9.0.7": version: 9.0.21 resolution: "immer@npm:9.0.21" @@ -14123,7 +16391,7 @@ __metadata: version: 0.0.0-use.local resolution: "integration-tests@workspace:compiler/integration-tests" dependencies: - "@aztec/bb.js": "portal:../../../../barretenberg/ts" + "@aztec/bb.js": 0.66.0 "@noir-lang/noir_js": "workspace:*" "@noir-lang/noir_wasm": "workspace:*" "@nomicfoundation/hardhat-chai-matchers": ^2.0.0 @@ -14143,6 +16411,13 @@ __metadata: languageName: unknown linkType: soft +"internmap@npm:1 - 2": + version: 2.0.3 + resolution: "internmap@npm:2.0.3" + checksum: 7ca41ec6aba8f0072fc32fa8a023450a9f44503e2d8e403583c55714b25efd6390c38a87161ec456bf42d7bc83aab62eb28f5aef34876b1ac4e60693d5e1d241 + languageName: node + linkType: hard + "interpret@npm:^1.0.0": version: 1.4.0 resolution: "interpret@npm:1.4.0" @@ -14709,6 +16984,13 @@ __metadata: languageName: node linkType: hard +"iso-639-1@npm:^3.1.3": + version: 3.1.3 + resolution: "iso-639-1@npm:3.1.3" + checksum: 9a4cf417a91f638af247328e2b92ca135ec82eedbab139246cfd0a53d2dee052a8abc1639ca997e84fbebb0cd536cb7fb88910433c922122bcb7250a6a16d8e9 + languageName: node + linkType: hard + "isobject@npm:^3.0.1": version: 3.0.1 resolution: "isobject@npm:3.0.1" @@ -14716,6 +16998,15 @@ __metadata: languageName: node linkType: hard +"isows@npm:1.0.3": + version: 1.0.3 + resolution: "isows@npm:1.0.3" + peerDependencies: + ws: "*" + checksum: 9cacd5cf59f67deb51e825580cd445ab1725ecb05a67c704050383fb772856f3cd5e7da8ad08f5a3bd2823680d77d099459d0c6a7037972a74d6429af61af440 + languageName: node + linkType: hard + "istanbul-lib-coverage@npm:^3.0.0": version: 3.2.2 resolution: "istanbul-lib-coverage@npm:3.2.2" @@ -14757,6 +17048,13 @@ __metadata: languageName: node linkType: hard +"jest-get-type@npm:^29.6.3": + version: 29.6.3 + resolution: "jest-get-type@npm:29.6.3" + checksum: 88ac9102d4679d768accae29f1e75f592b760b44277df288ad76ce5bf038c3f5ce3719dea8aa0f035dac30e9eb034b848ce716b9183ad7cc222d029f03e92205 + languageName: node + linkType: hard + "jest-util@npm:^29.7.0": version: 29.7.0 resolution: "jest-util@npm:29.7.0" @@ -14771,6 +17069,20 @@ __metadata: languageName: node linkType: hard +"jest-validate@npm:^29.4.3": + version: 29.7.0 + resolution: "jest-validate@npm:29.7.0" + dependencies: + "@jest/types": ^29.6.3 + camelcase: ^6.2.0 + chalk: ^4.0.0 + jest-get-type: ^29.6.3 + leven: ^3.1.0 + pretty-format: ^29.7.0 + checksum: 191fcdc980f8a0de4dbdd879fa276435d00eb157a48683af7b3b1b98b0f7d9de7ffe12689b617779097ff1ed77601b9f7126b0871bba4f776e222c40f62e9dae + languageName: node + linkType: hard + "jest-worker@npm:^27.4.5": version: 27.5.1 resolution: "jest-worker@npm:27.5.1" @@ -14794,6 +17106,15 @@ __metadata: languageName: node linkType: hard +"jiti@npm:^1.17.1": + version: 1.21.7 + resolution: "jiti@npm:1.21.7" + bin: + jiti: bin/jiti.js + checksum: 9cd20dabf82e3a4cceecb746a69381da7acda93d34eed0cdb9c9bdff3bce07e4f2f4a016ca89924392c935297d9aedc58ff9f7d3281bc5293319ad244926e0b7 + languageName: node + linkType: hard + "jiti@npm:^1.18.2, jiti@npm:^1.20.0": version: 1.21.0 resolution: "jiti@npm:1.21.0" @@ -14816,6 +17137,13 @@ __metadata: languageName: node linkType: hard +"js-sha256@npm:^0.10.1": + version: 0.10.1 + resolution: "js-sha256@npm:0.10.1" + checksum: 6eb5c9f95aa902cec1930f036deb3bc664024b75fede456c0ac74b855797776c18620f47efec36787077a56ba2f3b8d6aacc7733ff8a2b5bb9ce6b655a35c5e6 + languageName: node + linkType: hard + "js-sha3@npm:0.8.0": version: 0.8.0 resolution: "js-sha3@npm:0.8.0" @@ -14862,6 +17190,15 @@ __metadata: languageName: node linkType: hard +"jsesc@npm:^3.0.2": + version: 3.1.0 + resolution: "jsesc@npm:3.1.0" + bin: + jsesc: bin/jsesc + checksum: 19c94095ea026725540c0d29da33ab03144f6bcf2d4159e4833d534976e99e0c09c38cefa9a575279a51fc36b31166f8d6d05c9fe2645d5f15851d690b41f17f + languageName: node + linkType: hard + "jsesc@npm:~0.5.0": version: 0.5.0 resolution: "jsesc@npm:0.5.0" @@ -14936,7 +17273,7 @@ __metadata: languageName: node linkType: hard -"json5@npm:^2.1.2, json5@npm:^2.2.3": +"json5@npm:^2.1.2, json5@npm:^2.2.0, json5@npm:^2.2.3": version: 2.2.3 resolution: "json5@npm:2.2.3" bin: @@ -14977,6 +17314,18 @@ __metadata: languageName: node linkType: hard +"jszip@npm:^3.10.1": + version: 3.10.1 + resolution: "jszip@npm:3.10.1" + dependencies: + lie: ~3.3.0 + pako: ~1.0.2 + readable-stream: ~2.3.6 + setimmediate: ^1.0.5 + checksum: abc77bfbe33e691d4d1ac9c74c8851b5761fba6a6986630864f98d876f3fcc2d36817dfc183779f32c00157b5d53a016796677298272a714ae096dfe6b1c8b60 + languageName: node + linkType: hard + "just-extend@npm:^4.0.2": version: 4.2.1 resolution: "just-extend@npm:4.2.1" @@ -15283,6 +17632,15 @@ __metadata: languageName: node linkType: hard +"lie@npm:~3.3.0": + version: 3.3.0 + resolution: "lie@npm:3.3.0" + dependencies: + immediate: ~3.0.5 + checksum: 33102302cf19766f97919a6a98d481e01393288b17a6aa1f030a3542031df42736edde8dab29ffdbf90bebeffc48c761eb1d064dc77592ca3ba3556f9fe6d2a8 + languageName: node + linkType: hard + "lighthouse-logger@npm:^1.0.0": version: 1.4.2 resolution: "lighthouse-logger@npm:1.4.2" @@ -15740,6 +18098,19 @@ __metadata: languageName: node linkType: hard +"mdast-util-from-markdown@npm:^0.8.0": + version: 0.8.5 + resolution: "mdast-util-from-markdown@npm:0.8.5" + dependencies: + "@types/mdast": ^3.0.0 + mdast-util-to-string: ^2.0.0 + micromark: ~2.11.0 + parse-entities: ^2.0.0 + unist-util-stringify-position: ^2.0.0 + checksum: 5a9d0d753a42db763761e874c22365d0c7c9934a5a18b5ff76a0643610108a208a041ffdb2f3d3dd1863d3d915225a4020a0aade282af0facfd0df110601eee6 + languageName: node + linkType: hard + "mdast-util-from-markdown@npm:^2.0.0": version: 2.0.0 resolution: "mdast-util-from-markdown@npm:2.0.0" @@ -15954,6 +18325,22 @@ __metadata: languageName: node linkType: hard +"mdast-util-to-hast@npm:^10.2.0": + version: 10.2.0 + resolution: "mdast-util-to-hast@npm:10.2.0" + dependencies: + "@types/mdast": ^3.0.0 + "@types/unist": ^2.0.0 + mdast-util-definitions: ^4.0.0 + mdurl: ^1.0.0 + unist-builder: ^2.0.0 + unist-util-generated: ^1.0.0 + unist-util-position: ^3.0.0 + unist-util-visit: ^2.0.0 + checksum: 72df2dd9bfa2d07b4750a333444f82e0f3752dae75b6e300cf0a716407a185eb75095a54ecad90cbd6f6d133b20dea8844ff76c1ea78612550de170b43d4fa85 + languageName: node + linkType: hard + "mdast-util-to-hast@npm:^13.0.0": version: 13.0.2 resolution: "mdast-util-to-hast@npm:13.0.2" @@ -16086,6 +18473,13 @@ __metadata: languageName: node linkType: hard +"messageformat-parser@npm:^4.1.3": + version: 4.1.3 + resolution: "messageformat-parser@npm:4.1.3" + checksum: d5a72581116b1813460241672ffb6ff3e2f05c311d48c7d81daf368f5254447dda095756e53830664383c51c707dbf387c21b35cd341b7edd4f2945656ac476f + languageName: node + linkType: hard + "methods@npm:~1.1.2": version: 1.1.2 resolution: "methods@npm:1.1.2" @@ -16606,6 +19000,16 @@ __metadata: languageName: node linkType: hard +"micromark@npm:~2.11.0": + version: 2.11.4 + resolution: "micromark@npm:2.11.4" + dependencies: + debug: ^4.0.0 + parse-entities: ^2.0.0 + checksum: f8a5477d394908a5d770227aea71657a76423d420227c67ea0699e659a5f62eb39d504c1f7d69ec525a6af5aaeb6a7bffcdba95614968c03d41d3851edecb0d6 + languageName: node + linkType: hard + "micromatch@npm:^4.0.0, micromatch@npm:^4.0.2, micromatch@npm:^4.0.4, micromatch@npm:^4.0.5": version: 4.0.5 resolution: "micromatch@npm:4.0.5" @@ -16905,6 +19309,13 @@ __metadata: languageName: node linkType: hard +"moo@npm:^0.5.1": + version: 0.5.2 + resolution: "moo@npm:0.5.2" + checksum: 5a41ddf1059fd0feb674d917c4774e41c877f1ca980253be4d3aae1a37f4bc513f88815041243f36f5cf67a62fb39324f3f997cf7fb17b6cb00767c165e7c499 + languageName: node + linkType: hard + "mrmime@npm:^1.0.0": version: 1.0.1 resolution: "mrmime@npm:1.0.1" @@ -16961,7 +19372,7 @@ __metadata: languageName: node linkType: hard -"nanoid@npm:^3.1.25, nanoid@npm:^3.3.7": +"nanoid@npm:3.3.7, nanoid@npm:^3.1.25, nanoid@npm:^3.3.7": version: 3.3.7 resolution: "nanoid@npm:3.3.7" bin: @@ -17304,6 +19715,17 @@ __metadata: languageName: node linkType: hard +"oniguruma-to-es@npm:^1.0.0": + version: 1.0.0 + resolution: "oniguruma-to-es@npm:1.0.0" + dependencies: + emoji-regex-xs: ^1.0.0 + regex: ^5.1.1 + regex-recursion: ^5.1.1 + checksum: 2d88b3f0c670b1b7c87bf5c4caefea12771748c5970f691f04284604f3dce745107f7558573395c9103bea56154062b421d0a2a7005ada93968a7071316f5d1e + languageName: node + linkType: hard + "only@npm:~0.0.2": version: 0.0.2 resolution: "only@npm:0.0.2" @@ -17577,6 +19999,13 @@ __metadata: languageName: node linkType: hard +"pako@npm:~1.0.2": + version: 1.0.11 + resolution: "pako@npm:1.0.11" + checksum: 1be2bfa1f807608c7538afa15d6f25baa523c30ec870a3228a89579e474a4d992f4293859524e46d5d87fd30fa17c5edf34dbef0671251d9749820b488660b16 + languageName: node + linkType: hard + "param-case@npm:^3.0.4": version: 3.0.4 resolution: "param-case@npm:3.0.4" @@ -17910,6 +20339,18 @@ __metadata: languageName: node linkType: hard +"popmotion@npm:11.0.3": + version: 11.0.3 + resolution: "popmotion@npm:11.0.3" + dependencies: + framesync: 6.0.1 + hey-listen: ^1.0.8 + style-value-types: 5.0.0 + tslib: ^2.1.0 + checksum: 9fe7d03b4ec0e85bfb9dadc23b745147bfe42e16f466ba06e6327197d0e38b72015afc2f918a8051dedc3680310417f346ffdc463be6518e2e92e98f48e30268 + languageName: node + linkType: hard + "portfinder@npm:^1.0.32": version: 1.0.32 resolution: "portfinder@npm:1.0.32" @@ -18698,7 +21139,7 @@ __metadata: languageName: node linkType: hard -"postcss-value-parser@npm:^4.1.0, postcss-value-parser@npm:^4.2.0": +"postcss-value-parser@npm:^4.0.2, postcss-value-parser@npm:^4.1.0, postcss-value-parser@npm:^4.2.0": version: 4.2.0 resolution: "postcss-value-parser@npm:4.2.0" checksum: 819ffab0c9d51cf0acbabf8996dffbfafbafa57afc0e4c98db88b67f2094cb44488758f06e5da95d7036f19556a4a732525e84289a425f4f6fd8e412a9d7442f @@ -18723,6 +21164,17 @@ __metadata: languageName: node linkType: hard +"postcss@npm:8.4.38": + version: 8.4.38 + resolution: "postcss@npm:8.4.38" + dependencies: + nanoid: ^3.3.7 + picocolors: ^1.0.0 + source-map-js: ^1.2.0 + checksum: 649f9e60a763ca4b5a7bbec446a069edf07f057f6d780a5a0070576b841538d1ecf7dd888f2fbfd1f76200e26c969e405aeeae66332e6927dbdc8bdcb90b9451 + languageName: node + linkType: hard + "postcss@npm:^8.4.14, postcss@npm:^8.4.17, postcss@npm:^8.4.21, postcss@npm:^8.4.26": version: 8.4.32 resolution: "postcss@npm:8.4.32" @@ -18745,6 +21197,25 @@ __metadata: languageName: node linkType: hard +"posthog-js@npm:^1.136.8": + version: 1.205.1 + resolution: "posthog-js@npm:1.205.1" + dependencies: + core-js: ^3.38.1 + fflate: ^0.4.8 + preact: ^10.19.3 + web-vitals: ^4.2.0 + checksum: d91ce45d2d3d5784b6db8d2f4408619db6512bd3d5fae8022f6fc91864b3b2d2c14c0081b670aefa1a8b63d7a498b2a9e3e58e58583e419893b54276832f1635 + languageName: node + linkType: hard + +"preact@npm:^10.19.3": + version: 10.25.4 + resolution: "preact@npm:10.25.4" + checksum: 309f3128267c5bcac828c70a7a97fba0fdfed7b9ef2ece32a50bf94d257b5c825975df0648d9d5c79f90201d3a295cb2c9f511640aca37cb6879e509688b885a + languageName: node + linkType: hard + "prelude-ls@npm:^1.2.1": version: 1.2.1 resolution: "prelude-ls@npm:1.2.1" @@ -18796,6 +21267,17 @@ __metadata: languageName: node linkType: hard +"pretty-format@npm:^29.7.0": + version: 29.7.0 + resolution: "pretty-format@npm:29.7.0" + dependencies: + "@jest/schemas": ^29.6.3 + ansi-styles: ^5.0.0 + react-is: ^18.0.0 + checksum: 032c1602383e71e9c0c02a01bbd25d6759d60e9c7cf21937dde8357aa753da348fcec5def5d1002c9678a8524d5fe099ad98861286550ef44de8808cc61e43b6 + languageName: node + linkType: hard + "pretty-time@npm:^1.1.0": version: 1.1.0 resolution: "pretty-time@npm:1.1.0" @@ -18870,7 +21352,7 @@ __metadata: languageName: node linkType: hard -"prop-types@npm:^15.6.2, prop-types@npm:^15.7.2": +"prop-types@npm:^15.6.2, prop-types@npm:^15.7.2, prop-types@npm:^15.8.1": version: 15.8.1 resolution: "prop-types@npm:15.8.1" dependencies: @@ -19055,6 +21537,13 @@ __metadata: languageName: node linkType: hard +"ramda@npm:^0.27.1": + version: 0.27.2 + resolution: "ramda@npm:0.27.2" + checksum: 28d6735dd1eea1a796c56cf6111f3673c6105bbd736e521cdd7826c46a18eeff337c2dba4668f6eed990d539b9961fd6db19aa46ccc1530ba67a396c0a9f580d + languageName: node + linkType: hard + "randombytes@npm:^2.1.0": version: 2.1.0 resolution: "randombytes@npm:2.1.0" @@ -19116,6 +21605,15 @@ __metadata: languageName: node linkType: hard +"react-complex-tree@npm:^2.3.7": + version: 2.4.6 + resolution: "react-complex-tree@npm:2.4.6" + peerDependencies: + react: ">=16.0.0" + checksum: 1e0b9ff514fb265e5b95ebda752d20dd994c3ec044e4dc1dffc292afc763e2c02834fb4ebbead6f9a2bd607628590b11547325b017060a05ceb9ddaa71a7af8a + languageName: node + linkType: hard + "react-dev-utils@npm:^12.0.1": version: 12.0.1 resolution: "react-dev-utils@npm:12.0.1" @@ -19211,6 +21709,13 @@ __metadata: languageName: node linkType: hard +"react-is@npm:^18.0.0, react-is@npm:^18.3.1": + version: 18.3.1 + resolution: "react-is@npm:18.3.1" + checksum: e20fe84c86ff172fc8d898251b7cc2c43645d108bf96d0b8edf39b98f9a2cae97b40520ee7ed8ee0085ccc94736c4886294456033304151c3f94978cec03df21 + languageName: node + linkType: hard + "react-json-view-lite@npm:^1.2.0": version: 1.2.1 resolution: "react-json-view-lite@npm:1.2.1" @@ -19220,26 +21725,104 @@ __metadata: languageName: node linkType: hard -"react-loadable-ssr-addon-v5-slorber@npm:^1.0.1": - version: 1.0.1 - resolution: "react-loadable-ssr-addon-v5-slorber@npm:1.0.1" +"react-loadable-ssr-addon-v5-slorber@npm:^1.0.1": + version: 1.0.1 + resolution: "react-loadable-ssr-addon-v5-slorber@npm:1.0.1" + dependencies: + "@babel/runtime": ^7.10.3 + peerDependencies: + react-loadable: "*" + webpack: ">=4.41.1 || 5.x" + checksum: 1cf7ceb488d329a5be15f891dae16727fb7ade08ef57826addd21e2c3d485e2440259ef8be94f4d54e9afb4bcbd2fcc22c3c5bad92160c9c06ae6ba7b5562497 + languageName: node + linkType: hard + +"react-loadable@npm:@docusaurus/react-loadable@6.0.0": + version: 6.0.0 + resolution: "@docusaurus/react-loadable@npm:6.0.0" + dependencies: + "@types/react": "*" + peerDependencies: + react: "*" + checksum: 4c32061b2fc10689d5d8ba11ead71b69e4c8a55fcfeafb551a6747b1a7b496c4f2d8dbb5d023f5cafc2a9aea9d14582bdb324d11e6f9b8c3049d45b74439203f + languageName: node + linkType: hard + +"react-remark@npm:^2.1.0": + version: 2.1.0 + resolution: "react-remark@npm:2.1.0" + dependencies: + rehype-react: ^6.0.0 + remark-parse: ^9.0.0 + remark-rehype: ^8.0.0 + unified: ^9.0.0 + peerDependencies: + react: ">=16.8" + checksum: 96d7e221f15e26b9ea25e496a4490b92488331aa68678dcecc9e0621e07b1dfefc9a22a7d99c99bb81791397a3910aa532c53aff8d0a52892f0e79d84cf874e8 + languageName: node + linkType: hard + +"react-remove-scroll-bar@npm:^2.3.3, react-remove-scroll-bar@npm:^2.3.7": + version: 2.3.8 + resolution: "react-remove-scroll-bar@npm:2.3.8" + dependencies: + react-style-singleton: ^2.2.2 + tslib: ^2.0.0 + peerDependencies: + "@types/react": "*" + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: c4663247f689dbe51c370836edf735487f6d8796acb7f15b09e8a1c14e84c7997360e8e3d54de2bc9c0e782fed2b2c4127d15b4053e4d2cf26839e809e57605f + languageName: node + linkType: hard + +"react-remove-scroll@npm:2.5.5": + version: 2.5.5 + resolution: "react-remove-scroll@npm:2.5.5" dependencies: - "@babel/runtime": ^7.10.3 + react-remove-scroll-bar: ^2.3.3 + react-style-singleton: ^2.2.1 + tslib: ^2.1.0 + use-callback-ref: ^1.3.0 + use-sidecar: ^1.1.2 peerDependencies: - react-loadable: "*" - webpack: ">=4.41.1 || 5.x" - checksum: 1cf7ceb488d329a5be15f891dae16727fb7ade08ef57826addd21e2c3d485e2440259ef8be94f4d54e9afb4bcbd2fcc22c3c5bad92160c9c06ae6ba7b5562497 + "@types/react": ^16.8.0 || ^17.0.0 || ^18.0.0 + react: ^16.8.0 || ^17.0.0 || ^18.0.0 + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 2c7fe9cbd766f5e54beb4bec2e2efb2de3583037b23fef8fa511ab426ed7f1ae992382db5acd8ab5bfb030a4b93a06a2ebca41377d6eeaf0e6791bb0a59616a4 languageName: node linkType: hard -"react-loadable@npm:@docusaurus/react-loadable@6.0.0": - version: 6.0.0 - resolution: "@docusaurus/react-loadable@npm:6.0.0" +"react-remove-scroll@npm:^2.6.1": + version: 2.6.2 + resolution: "react-remove-scroll@npm:2.6.2" dependencies: + react-remove-scroll-bar: ^2.3.7 + react-style-singleton: ^2.2.1 + tslib: ^2.1.0 + use-callback-ref: ^1.3.3 + use-sidecar: ^1.1.2 + peerDependencies: "@types/react": "*" + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 310e6e6d2f28226a1751dc5084a2dce49167f0b69e3d78d6510f329f423ee313d4f6477f5e1adccb68baef40a7af75541e980a8c398cb82ea0d3573e514e8124 + languageName: node + linkType: hard + +"react-resizable-panels@npm:2.0.19": + version: 2.0.19 + resolution: "react-resizable-panels@npm:2.0.19" peerDependencies: - react: "*" - checksum: 4c32061b2fc10689d5d8ba11ead71b69e4c8a55fcfeafb551a6747b1a7b496c4f2d8dbb5d023f5cafc2a9aea9d14582bdb324d11e6f9b8c3049d45b74439203f + react: ^16.14.0 || ^17.0.0 || ^18.0.0 + react-dom: ^16.14.0 || ^17.0.0 || ^18.0.0 + checksum: d09e07bf2514fb8bb60b6150708943bb784ca1827c4c45dfe8354fb33e4edde795292789b56383886e974d381503333112194679d606808fbc9bab7ec232430e languageName: node linkType: hard @@ -19291,6 +21874,20 @@ __metadata: languageName: node linkType: hard +"react-smooth@npm:^4.0.0": + version: 4.0.4 + resolution: "react-smooth@npm:4.0.4" + dependencies: + fast-equals: ^5.0.1 + prop-types: ^15.8.1 + react-transition-group: ^4.4.5 + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + react-dom: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + checksum: 909305d40bae79a011ff21a10c4bc7ddadc87ac5ff093b4a5f827f730f093ec4e044c4330688d29b3ad2db83aab8997c3bb1bae550a9c66de74521b8ed52cc53 + languageName: node + linkType: hard + "react-spinners@npm:^0.13.8": version: 0.13.8 resolution: "react-spinners@npm:0.13.8" @@ -19301,6 +21898,37 @@ __metadata: languageName: node linkType: hard +"react-style-singleton@npm:^2.2.1, react-style-singleton@npm:^2.2.2": + version: 2.2.3 + resolution: "react-style-singleton@npm:2.2.3" + dependencies: + get-nonce: ^1.0.0 + tslib: ^2.0.0 + peerDependencies: + "@types/react": "*" + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: a7b0bf493c9231065ebafa84c4237aed997c746c561196121b7de82fe155a5355b372db5070a3ac9fe980cf7f60dc0f1e8cf6402a2aa5b2957392932ccf76e76 + languageName: node + linkType: hard + +"react-transition-group@npm:^4.4.5": + version: 4.4.5 + resolution: "react-transition-group@npm:4.4.5" + dependencies: + "@babel/runtime": ^7.5.5 + dom-helpers: ^5.0.1 + loose-envify: ^1.4.0 + prop-types: ^15.6.2 + peerDependencies: + react: ">=16.6.0" + react-dom: ">=16.6.0" + checksum: 75602840106aa9c6545149d6d7ae1502fb7b7abadcce70a6954c4b64a438ff1cd16fc77a0a1e5197cdd72da398f39eb929ea06f9005c45b132ed34e056ebdeb1 + languageName: node + linkType: hard + "react@npm:^18.2.0": version: 18.2.0 resolution: "react@npm:18.2.0" @@ -19322,7 +21950,7 @@ __metadata: languageName: node linkType: hard -"readable-stream@npm:^2.0.1, readable-stream@npm:^2.2.2": +"readable-stream@npm:^2.0.1, readable-stream@npm:^2.2.2, readable-stream@npm:~2.3.6": version: 2.3.8 resolution: "readable-stream@npm:2.3.8" dependencies: @@ -19389,6 +22017,34 @@ __metadata: languageName: node linkType: hard +"recharts-scale@npm:^0.4.4": + version: 0.4.5 + resolution: "recharts-scale@npm:0.4.5" + dependencies: + decimal.js-light: ^2.4.1 + checksum: e970377190a610e684a32c7461c7684ac3603c2e0ac0020bbba1eea9d099b38138143a8e80bf769bb49c0b7cecf22a2f5c6854885efed2d56f4540d4aa7052bd + languageName: node + linkType: hard + +"recharts@npm:^2.12.4": + version: 2.15.0 + resolution: "recharts@npm:2.15.0" + dependencies: + clsx: ^2.0.0 + eventemitter3: ^4.0.1 + lodash: ^4.17.21 + react-is: ^18.3.1 + react-smooth: ^4.0.0 + recharts-scale: ^0.4.4 + tiny-invariant: ^1.3.1 + victory-vendor: ^36.6.8 + peerDependencies: + react: ^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + react-dom: ^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + checksum: b49337c1df995f60c302101e44e638660ca4ea5370467ffbc61c0ee829770ffae0337564c0d1dc8b46786c91d9201f6bcad9e0155173c4c6fca44a5f3a6decc3 + languageName: node + linkType: hard + "rechoir@npm:^0.6.2": version: 0.6.2 resolution: "rechoir@npm:0.6.2" @@ -19455,6 +22111,32 @@ __metadata: languageName: node linkType: hard +"regex-recursion@npm:^5.1.1": + version: 5.1.1 + resolution: "regex-recursion@npm:5.1.1" + dependencies: + regex: ^5.1.1 + regex-utilities: ^2.3.0 + checksum: 4f203ae8f4a2ebf9004f4e4119df5106ba07b39bd3778d7040a83b17f3a82fe22c202661adc3f5586e4eb782fece77e8a01eba8b7033f92147ad7a1e7e1531d7 + languageName: node + linkType: hard + +"regex-utilities@npm:^2.3.0": + version: 2.3.0 + resolution: "regex-utilities@npm:2.3.0" + checksum: 41408777df45cefe1b276281030213235aa1143809c4c10eb5573d2cc27ff2c4aa746c6f4d4c235e3d2f4830eff76b28906ce82fbe72895beca8e15204c2da51 + languageName: node + linkType: hard + +"regex@npm:^5.1.1": + version: 5.1.1 + resolution: "regex@npm:5.1.1" + dependencies: + regex-utilities: ^2.3.0 + checksum: bff664d0c001bf2929c2a5c92399419f719ef5ac9e7198bce653695d37628a3bd21595cea571f93ee13b55c5bbeff7fbab307a9ef569e36b149caf09601d4a31 + languageName: node + linkType: hard + "regexpu-core@npm:^5.3.1": version: 5.3.2 resolution: "regexpu-core@npm:5.3.2" @@ -19561,6 +22243,27 @@ __metadata: languageName: node linkType: hard +"rehype-react@npm:^6.0.0": + version: 6.2.1 + resolution: "rehype-react@npm:6.2.1" + dependencies: + "@mapbox/hast-util-table-cell-style": ^0.2.0 + hast-to-hyperscript: ^9.0.0 + checksum: 7f16a2cfc38d1fe0d40944f8f27309fc16440d69d3bb53f8e0ca6d6825572f7d1cac4212193a4f4de456b2342fea66ea0919648ad09c0663cfb1e4ca12bed647 + languageName: node + linkType: hard + +"rehype-react@npm:^8.0.0": + version: 8.0.0 + resolution: "rehype-react@npm:8.0.0" + dependencies: + "@types/hast": ^3.0.0 + hast-util-to-jsx-runtime: ^2.0.0 + unified: ^11.0.0 + checksum: dc5dfbaff4ba029a466e01729c63431cab4efa8937b811c9a2d25889d6e91839cff08592a11edff622ace2c08bfb0a564210ae5e27e21e22049cbbed77c02822 + languageName: node + linkType: hard + "relateurl@npm:^0.2.7": version: 0.2.7 resolution: "relateurl@npm:0.2.7" @@ -19711,6 +22414,15 @@ __metadata: languageName: node linkType: hard +"remark-parse@npm:^9.0.0": + version: 9.0.0 + resolution: "remark-parse@npm:9.0.0" + dependencies: + mdast-util-from-markdown: ^0.8.0 + checksum: 50104880549639b7dd7ae6f1e23c214915fe9c054f02f3328abdaee3f6de6d7282bf4357c3c5b106958fe75e644a3c248c2197755df34f9955e8e028fc74868f + languageName: node + linkType: hard + "remark-rehype@npm:^11.0.0": version: 11.0.0 resolution: "remark-rehype@npm:11.0.0" @@ -19724,6 +22436,28 @@ __metadata: languageName: node linkType: hard +"remark-rehype@npm:^11.1.0": + version: 11.1.1 + resolution: "remark-rehype@npm:11.1.1" + dependencies: + "@types/hast": ^3.0.0 + "@types/mdast": ^4.0.0 + mdast-util-to-hast: ^13.0.0 + unified: ^11.0.0 + vfile: ^6.0.0 + checksum: e199dff098ae6a560e13dd1778dec9c61440f979cc931c4ca4dcde0d58e51c0723243a901c1842379b189083cf4d74f224f57833738095ffa9535179d7cf3f01 + languageName: node + linkType: hard + +"remark-rehype@npm:^8.0.0": + version: 8.1.0 + resolution: "remark-rehype@npm:8.1.0" + dependencies: + mdast-util-to-hast: ^10.2.0 + checksum: e1152464cfa83c14b570b1cb85eb9b3667795b5bed2f6b16d1c6e96c369816b07945a3c04eb0e1fd57a19cc1837969527d0056d5b6d179f1290688db2a7e2c5f + languageName: node + linkType: hard + "remark-squeeze-paragraphs@npm:4.0.0": version: 4.0.0 resolution: "remark-squeeze-paragraphs@npm:4.0.0" @@ -20443,7 +23177,7 @@ __metadata: languageName: node linkType: hard -"shallowequal@npm:^1.1.0": +"shallowequal@npm:1.1.0, shallowequal@npm:^1.1.0": version: 1.1.0 resolution: "shallowequal@npm:1.1.0" checksum: f4c1de0837f106d2dbbfd5d0720a5d059d1c66b42b580965c8f06bb1db684be8783538b684092648c981294bf817869f743a066538771dbecb293df78f765e00 @@ -20486,6 +23220,22 @@ __metadata: languageName: node linkType: hard +"shiki@npm:1.26.2": + version: 1.26.2 + resolution: "shiki@npm:1.26.2" + dependencies: + "@shikijs/core": 1.26.2 + "@shikijs/engine-javascript": 1.26.2 + "@shikijs/engine-oniguruma": 1.26.2 + "@shikijs/langs": 1.26.2 + "@shikijs/themes": 1.26.2 + "@shikijs/types": 1.26.2 + "@shikijs/vscode-textmate": ^10.0.1 + "@types/hast": ^3.0.4 + checksum: 0cc7af769eb57de4bc1423ab60a62e8c68071914456b0bd94c6188051d770f65b8e76d94341f552274bb76221c734c0ccc802106fa2c969b701df5dffaf26a14 + languageName: node + linkType: hard + "shiki@npm:^0.14.1": version: 0.14.6 resolution: "shiki@npm:0.14.6" @@ -20683,6 +23433,23 @@ __metadata: languageName: node linkType: hard +"solc@npm:^0.8.25": + version: 0.8.28 + resolution: "solc@npm:0.8.28" + dependencies: + command-exists: ^1.2.8 + commander: ^8.1.0 + follow-redirects: ^1.12.1 + js-sha3: 0.8.0 + memorystream: ^0.3.1 + semver: ^5.5.0 + tmp: 0.0.33 + bin: + solcjs: solc.js + checksum: c22ec95c23505409e83fd7e2c1af47a2b2092d1886c1cf2256b9bc28958963d5bfc9944dc34d0d2e0cbc00975878068b8d87ab7407cec0f9ca5ac57951ee789a + languageName: node + linkType: hard + "sort-css-media-queries@npm:2.1.0": version: 2.1.0 resolution: "sort-css-media-queries@npm:2.1.0" @@ -20697,7 +23464,7 @@ __metadata: languageName: node linkType: hard -"source-map-js@npm:^1.0.1, source-map-js@npm:^1.2.1": +"source-map-js@npm:^1.0.1, source-map-js@npm:^1.2.0, source-map-js@npm:^1.2.1": version: 1.2.1 resolution: "source-map-js@npm:1.2.1" checksum: 4eb0cd997cdf228bc253bcaff9340afeb706176e64868ecd20efbe6efea931465f43955612346d6b7318789e5265bdc419bc7669c1cebe3db0eb255f57efa76b @@ -20829,6 +23596,13 @@ __metadata: languageName: node linkType: hard +"state-local@npm:^1.0.6": + version: 1.0.7 + resolution: "state-local@npm:1.0.7" + checksum: d1afcf1429e7e6eb08685b3a94be8797db847369316d4776fd51f3962b15b984dacc7f8e401ad20968e5798c9565b4b377afedf4e4c4d60fe7495e1cbe14a251 + languageName: node + linkType: hard + "state-toggle@npm:^1.0.0": version: 1.0.3 resolution: "state-toggle@npm:1.0.3" @@ -21061,6 +23835,36 @@ __metadata: languageName: node linkType: hard +"style-value-types@npm:5.0.0": + version: 5.0.0 + resolution: "style-value-types@npm:5.0.0" + dependencies: + hey-listen: ^1.0.8 + tslib: ^2.1.0 + checksum: 16d198302cd102edf9dba94e7752a2364c93b1eaa5cc7c32b42b28eef4af4ccb5149a3f16bc2a256adc02616a2404f4612bd15f3081c1e8ca06132cae78be6c0 + languageName: node + linkType: hard + +"styled-components@npm:^6.1.8": + version: 6.1.14 + resolution: "styled-components@npm:6.1.14" + dependencies: + "@emotion/is-prop-valid": 1.2.2 + "@emotion/unitless": 0.8.1 + "@types/stylis": 4.2.5 + css-to-react-native: 3.2.0 + csstype: 3.1.3 + postcss: 8.4.38 + shallowequal: 1.1.0 + stylis: 4.3.2 + tslib: 2.6.2 + peerDependencies: + react: ">= 16.8.0" + react-dom: ">= 16.8.0" + checksum: 88c0ce99f9c5bf593eedd0e287cc4f66dc6e64e49f36ad89a632ea36ee514e9b155d8927d06ec53af68ccce4c525208bd4f65d2019d373af4c60be6ebb53df38 + languageName: node + linkType: hard + "stylehacks@npm:^5.1.1": version: 5.1.1 resolution: "stylehacks@npm:5.1.1" @@ -21085,6 +23889,13 @@ __metadata: languageName: node linkType: hard +"stylis@npm:4.3.2": + version: 4.3.2 + resolution: "stylis@npm:4.3.2" + checksum: 0faa8a97ff38369f47354376cd9f0def9bf12846da54c28c5987f64aaf67dcb6f00dce88a8632013bfb823b2c4d1d62a44f4ac20363a3505a7ab4e21b70179fc + languageName: node + linkType: hard + "superstruct@npm:^1.0.3": version: 1.0.3 resolution: "superstruct@npm:1.0.3" @@ -21167,6 +23978,18 @@ __metadata: languageName: node linkType: hard +"swr@npm:^2.2.5": + version: 2.3.0 + resolution: "swr@npm:2.3.0" + dependencies: + dequal: ^2.0.3 + use-sync-external-store: ^1.4.0 + peerDependencies: + react: ^16.11.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + checksum: ea5503b9a34763e8af0681ee1570c5f8d301da6f46343512614799a07cadadf454cb2f832f8d49d2607da9147c9a8b8130994c4f4546c9da5d680bd9e90a8bcc + languageName: node + linkType: hard + "synckit@npm:^0.8.6": version: 0.8.8 resolution: "synckit@npm:0.8.8" @@ -21206,6 +24029,22 @@ __metadata: languageName: node linkType: hard +"tailwind-merge@npm:^2.2.1": + version: 2.6.0 + resolution: "tailwind-merge@npm:2.6.0" + checksum: 18976c4096920bc6125f1dc837479805de996d86bcc636f98436f65c297003bde89ffe51dfd325b7c97fc71b1dbba8505459dd96010e7b181badd29aea996440 + languageName: node + linkType: hard + +"tailwindcss-animate@npm:^1.0.7": + version: 1.0.7 + resolution: "tailwindcss-animate@npm:1.0.7" + peerDependencies: + tailwindcss: "*" + checksum: c1760983eb3fec0c8421e95082bf308e6845df43e2f90862386366e82545c801b26b4d189c4cd23d6915252b76d18005c8e5f591f8b119944c7fb8650d0f8bce + languageName: node + linkType: hard + "tapable@npm:^1.0.0": version: 1.1.3 resolution: "tapable@npm:1.1.3" @@ -21371,6 +24210,13 @@ __metadata: languageName: node linkType: hard +"tiny-invariant@npm:^1.3.1": + version: 1.3.3 + resolution: "tiny-invariant@npm:1.3.3" + checksum: 5e185c8cc2266967984ce3b352a4e57cb89dad5a8abb0dea21468a6ecaa67cd5bb47a3b7a85d08041008644af4f667fb8b6575ba38ba5fb00b3b5068306e59fe + languageName: node + linkType: hard + "tiny-warning@npm:^1.0.0": version: 1.0.3 resolution: "tiny-warning@npm:1.0.3" @@ -21588,6 +24434,13 @@ __metadata: languageName: node linkType: hard +"tslib@npm:2.6.2, tslib@npm:^2.0.1, tslib@npm:^2.0.3, tslib@npm:^2.1.0, tslib@npm:^2.4.0, tslib@npm:^2.5.0, tslib@npm:^2.6.0, tslib@npm:^2.6.2": + version: 2.6.2 + resolution: "tslib@npm:2.6.2" + checksum: 329ea56123005922f39642318e3d1f0f8265d1e7fcb92c633e0809521da75eeaca28d2cf96d7248229deb40e5c19adf408259f4b9640afd20d13aecc1430f3ad + languageName: node + linkType: hard + "tslib@npm:^1.9.3": version: 1.14.1 resolution: "tslib@npm:1.14.1" @@ -21595,10 +24448,10 @@ __metadata: languageName: node linkType: hard -"tslib@npm:^2.0.1, tslib@npm:^2.0.3, tslib@npm:^2.1.0, tslib@npm:^2.4.0, tslib@npm:^2.5.0, tslib@npm:^2.6.0, tslib@npm:^2.6.2": - version: 2.6.2 - resolution: "tslib@npm:2.6.2" - checksum: 329ea56123005922f39642318e3d1f0f8265d1e7fcb92c633e0809521da75eeaca28d2cf96d7248229deb40e5c19adf408259f4b9640afd20d13aecc1430f3ad +"tslib@npm:^2.0.0, tslib@npm:^2.3.1": + version: 2.8.1 + resolution: "tslib@npm:2.8.1" + checksum: e4aba30e632b8c8902b47587fd13345e2827fa639e7c3121074d5ee0880723282411a8838f830b55100cbe4517672f84a2472667d355b81e8af165a55dc6203a languageName: node linkType: hard @@ -21938,7 +24791,22 @@ __metadata: languageName: node linkType: hard -"unified@npm:^9.2.2": +"unified@npm:^11.0.5": + version: 11.0.5 + resolution: "unified@npm:11.0.5" + dependencies: + "@types/unist": ^3.0.0 + bail: ^2.0.0 + devlop: ^1.0.0 + extend: ^3.0.0 + is-plain-obj: ^4.0.0 + trough: ^2.0.0 + vfile: ^6.0.0 + checksum: b3bf7fd6f568cc261e074dae21188483b0f2a8ab858d62e6e85b75b96cc655f59532906ae3c64d56a9b257408722d71f1d4135292b3d7ee02907c8b592fb3cf0 + languageName: node + linkType: hard + +"unified@npm:^9.0.0, unified@npm:^9.2.2": version: 9.2.2 resolution: "unified@npm:9.2.2" dependencies: @@ -22012,6 +24880,13 @@ __metadata: languageName: node linkType: hard +"unist-util-is@npm:^3.0.0": + version: 3.0.0 + resolution: "unist-util-is@npm:3.0.0" + checksum: d24a5dd80c670f763b2ae608651cf062317456aa81be51f66f45cbd7d440a2ab18356e4f48aeac6b5e3d391c69d3c3452ade5fe5aa9574bec4a2de0b10122ed5 + languageName: node + linkType: hard + "unist-util-is@npm:^4.0.0": version: 4.1.0 resolution: "unist-util-is@npm:4.1.0" @@ -22099,6 +24974,15 @@ __metadata: languageName: node linkType: hard +"unist-util-visit-parents@npm:^2.0.0": + version: 2.1.2 + resolution: "unist-util-visit-parents@npm:2.1.2" + dependencies: + unist-util-is: ^3.0.0 + checksum: 048edbb590a8c4bc0043eec9f50d3fe76faa58f1ac663a7e6dee5e895ddd0ce8bc52f2cfe2e633849fa93671e8de021070667acb1518e3d40220768c7f70a3d3 + languageName: node + linkType: hard + "unist-util-visit-parents@npm:^3.0.0": version: 3.1.1 resolution: "unist-util-visit-parents@npm:3.1.1" @@ -22130,6 +25014,15 @@ __metadata: languageName: node linkType: hard +"unist-util-visit@npm:^1.4.1": + version: 1.4.1 + resolution: "unist-util-visit@npm:1.4.1" + dependencies: + unist-util-visit-parents: ^2.0.0 + checksum: e9395205b6908c8d0fe71bc44e65d89d4781d1bb2d453a33cb67ed4124bad0b89d6b1d526ebaecb82a7c48e211bdf6f24351449b8cc115327b345f4617c18728 + languageName: node + linkType: hard + "unist-util-visit@npm:^5.0.0": version: 5.0.0 resolution: "unist-util-visit@npm:5.0.0" @@ -22162,6 +25055,13 @@ __metadata: languageName: node linkType: hard +"unraw@npm:^3.0.0": + version: 3.0.0 + resolution: "unraw@npm:3.0.0" + checksum: 19eee0bc500ce197d262b79723a2c8c81c1d716baaa2a62c48a4d0d6b9e1fd9d350c5df86262e51343d591ab9c8a47ed150317d0b867b2b65795cdc17ef69873 + languageName: node + linkType: hard + "unzipit@npm:^1.4.3": version: 1.4.3 resolution: "unzipit@npm:1.4.3" @@ -22305,6 +25205,46 @@ __metadata: languageName: node linkType: hard +"use-callback-ref@npm:^1.3.0, use-callback-ref@npm:^1.3.3": + version: 1.3.3 + resolution: "use-callback-ref@npm:1.3.3" + dependencies: + tslib: ^2.0.0 + peerDependencies: + "@types/react": "*" + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 4da1c82d7a2409cee6c882748a40f4a083decf238308bf12c3d0166f0e338f8d512f37b8d11987eb5a421f14b9b5b991edf3e11ed25c3bb7a6559081f8359b44 + languageName: node + linkType: hard + +"use-sidecar@npm:^1.1.2": + version: 1.1.3 + resolution: "use-sidecar@npm:1.1.3" + dependencies: + detect-node-es: ^1.1.0 + tslib: ^2.0.0 + peerDependencies: + "@types/react": "*" + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 || ^19.0.0-rc + peerDependenciesMeta: + "@types/react": + optional: true + checksum: 88664c6b2c5b6e53e4d5d987694c9053cea806da43130248c74ca058945c8caa6ccb7b1787205a9eb5b9d124633e42153848904002828acabccdc48cda026622 + languageName: node + linkType: hard + +"use-sync-external-store@npm:^1.2.0, use-sync-external-store@npm:^1.4.0": + version: 1.4.0 + resolution: "use-sync-external-store@npm:1.4.0" + peerDependencies: + react: ^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0 + checksum: dc3843a1b59ac8bd01417bd79498d4c688d5df8bf4801be50008ef4bfaacb349058c0b1605b5b43c828e0a2d62722d7e861573b3f31cea77a7f23e8b0fc2f7e3 + languageName: node + linkType: hard + "util-deprecate@npm:^1.0.1, util-deprecate@npm:^1.0.2, util-deprecate@npm:~1.0.1": version: 1.0.2 resolution: "util-deprecate@npm:1.0.2" @@ -22454,6 +25394,49 @@ __metadata: languageName: node linkType: hard +"victory-vendor@npm:^36.6.8": + version: 36.9.2 + resolution: "victory-vendor@npm:36.9.2" + dependencies: + "@types/d3-array": ^3.0.3 + "@types/d3-ease": ^3.0.0 + "@types/d3-interpolate": ^3.0.1 + "@types/d3-scale": ^4.0.2 + "@types/d3-shape": ^3.1.0 + "@types/d3-time": ^3.0.0 + "@types/d3-timer": ^3.0.0 + d3-array: ^3.1.6 + d3-ease: ^3.0.1 + d3-interpolate: ^3.0.1 + d3-scale: ^4.0.2 + d3-shape: ^3.1.0 + d3-time: ^3.0.0 + d3-timer: ^3.0.1 + checksum: a755110e287b700202d08ac81982093ab100edaa9d61beef1476d59e9705605bd8299a3aa41fa04b933a12bd66737f4c8f7d18448dd6488c69d4f72480023a2e + languageName: node + linkType: hard + +"viem@npm:2.9.16": + version: 2.9.16 + resolution: "viem@npm:2.9.16" + dependencies: + "@adraffy/ens-normalize": 1.10.0 + "@noble/curves": 1.2.0 + "@noble/hashes": 1.3.2 + "@scure/bip32": 1.3.2 + "@scure/bip39": 1.2.1 + abitype: 1.0.0 + isows: 1.0.3 + ws: 8.13.0 + peerDependencies: + typescript: ">=5.0.4" + peerDependenciesMeta: + typescript: + optional: true + checksum: aef2e4b4c906fbeed31b4fdf273cfccfc297c4f62c639334f3ba7c642c2374ac1f91531f19f6bbb7051af14dc5420d54f7b26990bbda079b485c9ab66ca09826 + languageName: node + linkType: hard + "vscode-languageserver-textdocument@npm:^1.0.11": version: 1.0.11 resolution: "vscode-languageserver-textdocument@npm:1.0.11" @@ -22530,6 +25513,13 @@ __metadata: languageName: node linkType: hard +"web-vitals@npm:^4.2.0": + version: 4.2.4 + resolution: "web-vitals@npm:4.2.4" + checksum: 5b3ffe1db33f23aebf8cc8560ac574401a95939baafde5841835c1bb1c01f9a2478442f319f77aa0d7914739fc2f6b020c5d5b128c16c5c77ca6be2f9dfbbde6 + languageName: node + linkType: hard + "webidl-conversions@npm:^3.0.0": version: 3.0.1 resolution: "webidl-conversions@npm:3.0.1" @@ -23017,6 +26007,21 @@ __metadata: languageName: node linkType: hard +"ws@npm:8.13.0": + version: 8.13.0 + resolution: "ws@npm:8.13.0" + peerDependencies: + bufferutil: ^4.0.1 + utf-8-validate: ">=5.0.2" + peerDependenciesMeta: + bufferutil: + optional: true + utf-8-validate: + optional: true + checksum: 53e991bbf928faf5dc6efac9b8eb9ab6497c69feeb94f963d648b7a3530a720b19ec2e0ec037344257e05a4f35bd9ad04d9de6f289615ffb133282031b18c61c + languageName: node + linkType: hard + "ws@npm:8.16.0, ws@npm:^8.16.0": version: 8.16.0 resolution: "ws@npm:8.16.0" @@ -23285,7 +26290,7 @@ __metadata: languageName: node linkType: hard -"zwitch@npm:^2.0.0": +"zwitch@npm:^2.0.0, zwitch@npm:^2.0.4": version: 2.0.4 resolution: "zwitch@npm:2.0.4" checksum: f22ec5fc2d5f02c423c93d35cdfa83573a3a3bd98c66b927c368ea4d0e7252a500df2a90a6b45522be536a96a73404393c958e945fdba95e6832c200791702b6