Skip to main content
Version: 1.2

Precompiles

RISC Zero's rv32im implementation includes a number of specialized extension circuits, including "precompiles" for cryptographic and algebraic functions: SHA-256, RSA, elliptic curve, and modular multiplication operations. By implementing these operations directly in the "hardware" of the zkVM, programs that use these precompiles execute faster and can be proven with significantly less resources 1.

To see statistics on precompile usage in your guest program, use RISC0_INFO=1.

Patched Crates

We have patched several popular cryptographic Rust crates to create "accelerated" versions that integrate our precompiles.

For the most up to date tags to use for the following crates, see the /releases in each fork's repository on GitHub.

Hash Functions

CrateVersions supportedPatch Statement ExampleRequires Unstable Flag?
sha20.10.8, 0.10.7, 0.10.6, 0.9.9sha2 = { git = "https://github.com/risc0/RustCrypto-hashes", tag = "sha2-v0.10.8-risczero.0" }No
tiny-keccak2.0.2tiny-keccak = { git = "https://github.com/risc0/tiny-keccak", tag = "tiny-keccak/v2.0.2-risczero.0" }Yes

ECDSA

CrateVersions supportedPatch Statement ExampleRequires Unstable Flag?
k2560.13.4, 0.13.3, 0.13.2, 0.13.1k256 = { git = "https://github.com/risc0/RustCrypto-elliptic-curves", tag = "k256/v0.13.3-risczero.1" }Yes
p2560.13.2p256 = { git = "https://github.com/risc0/RustCrypto-elliptic-curves", tag = "p256/v0.13.2-risczero.0" }Yes

EDDSA

CrateVersions supportedPatch Statement ExampleRequires Unstable Flag?
curve25519-dalek4.1.2, 4.1.1, 4.1.0ed25519-dalek = { git = "https://github.com/risc0/ed25519-dalek", tag = "curve25519-4.1.2-risczero.0" }No

RSA

CrateVersions supportedPatch Statement ExampleRequires Unstable Flag?
rsa0.9.6rsa = { git = "https://github.com/risc0/RustCrypto-RSA", tag = "v0.9.6-risczero.0" }Yes

Other Patched Crates

CrateVersions supportedPatch Statement ExampleRequires Unstable Flag?
crypto-bigint0.5.5, 0.5.4, 0.5.3, 0.5.2crypto-bigint = { git = "https://github.com/risc0/RustCrypto-crypto-bigint", tag = "v0.5.5-risczero.0" }Yes

Make sure that your dependency gives the same patch version of the crate as listed in the git tag of the patch. If you need other patch versions or crates than listed here, please reach out to us on Discord or otherwise!

If using tag = "sha2-v0.10.8-risczero.0", the dependency should be defined as a git dependency:

[dependencies]
sha2 = "=0.10.8"

[patch.crates-io]
sha2 = { git = "https://github.com/risc0/RustCrypto-hashes", tag = "sha2-v0.10.8-risczero.0" }

In some situations, for example when a patch is used indirectly, you may need to update the version specifically to update your lockfile for your guest. For example:

cargo update -p sha2 --precise 0.10.8

Note: To ensure that the patch is being applied, search for the crate in the Cargo.lock file of your guest to ensure that it references the fork repository. It is generally good practice to commit the Cargo.lock file to git, to ensure versions don't get accidentally updated.

When using cryptography indirectly, e.g. via the cookie, oauth2, or revm, crates it may be possible to enable precompile support without code changes by applying a Cargo patch.

An example of how to use these crates to accelerate ECDSA signature verification can be in the ECDSA example. Note the use of the patched versions of sha2, crypto-bigint and k256 crates used in the guest's Cargo.toml.

Adding Precompile Support To Crates

It's possible to add precompile support for your own crates.

An example of how to do this can be found in this diff of RISC Zero's k256 crate fork, which shows the code changes needed to accelerate RustCrypto's secp256k1 ECDSA library. This fork starts from the base implementation, and changes the core operations to use the precompiled 256-bit elliptic curve instructions. E.g. lincomb.

Stability

Certain versions of patches for some crates (e.g. k256, rsa) depend on more optimized precompiles that are still undergoing revision and review, and so users must opt-in to these features by setting the "unstable" feature flag on the risc0-zkvm crate used by the zkVM guest and by the risc0-build crate used to build the guest. These also require using versions >=1.2.0 of risc0 crates. For users who need a stable, production-ready version we are working on stablizing these precompiles as soon as possible, and the "unstable" feature flag will no longer be required.

Footnotes

  1. This is similar to the cryptography support such as AES-NI or the SHA extensions for x86 processors. In both cases, the circuitry is extended to compute otherwise expensive operations in fewer instruction cycles.