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
Crate | Versions supported | Patch Statement Example | Requires Unstable Flag? |
---|---|---|---|
sha2 | 0.10.8, 0.10.7, 0.10.6, 0.9.9 | sha2 = { git = "https://github.com/risc0/RustCrypto-hashes", tag = "sha2-v0.10.8-risczero.0" } | No |
tiny-keccak | 2.0.2 | tiny-keccak = { git = "https://github.com/risc0/tiny-keccak", tag = "tiny-keccak/v2.0.2-risczero.0" } | Yes |
ECDSA
Crate | Versions supported | Patch Statement Example | Requires Unstable Flag? |
---|---|---|---|
k256 | 0.13.4, 0.13.3, 0.13.2, 0.13.1 | k256 = { git = "https://github.com/risc0/RustCrypto-elliptic-curves", tag = "k256/v0.13.3-risczero.1" } | Yes |
p256 | 0.13.2 | p256 = { git = "https://github.com/risc0/RustCrypto-elliptic-curves", tag = "p256/v0.13.2-risczero.0" } | Yes |
EDDSA
Crate | Versions supported | Patch Statement Example | Requires Unstable Flag? |
---|---|---|---|
curve25519-dalek | 4.1.2, 4.1.1, 4.1.0 | ed25519-dalek = { git = "https://github.com/risc0/ed25519-dalek", tag = "curve25519-4.1.2-risczero.0" } | No |
RSA
Crate | Versions supported | Patch Statement Example | Requires Unstable Flag? |
---|---|---|---|
rsa | 0.9.6 | rsa = { git = "https://github.com/risc0/RustCrypto-RSA", tag = "v0.9.6-risczero.0" } | Yes |
Other Patched Crates
Crate | Versions supported | Patch Statement Example | Requires Unstable Flag? |
---|---|---|---|
crypto-bigint | 0.5.5, 0.5.4, 0.5.3, 0.5.2 | crypto-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 theCargo.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
-
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. ↩