0%
Welcome to the RISC Zero documentation! RISC Zero is creating the infrastructure & tooling necessary for developers around the globe to build software that leverages ZK technology.
ZK technology is staged to re-shape the way we interact digitally. Historically, the only method for confirming the correct execution of a software application was through redundant computation. ZK introduces a new option: verifiable computation.
It's now possible to pair a program's output with a self-certifying receipt, allowing a skeptical third party to verify correct execution — and the verifier doesn't need to repeat the original computation or even see the inputs to the program!
Ready to start building? Check out our Getting Started page.
These instructions guide you through installing or updating RISC Zero tools to build your own RISC Zero zkVM projects.
The RISC Zero zkVM requires Rust. If you don't already have Rust and rustup installed, start by installing Rust and rustup.
curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh
Follow the onscreen instructions to complete the installation. If you're using Windows Subsystem for Linux (WSL), run the command above in your terminal.
Use rzup
, the RISC Zero toolchain installer, to install the necessary tools:
curl -L https://risczero.com/install | bash
rzup install
This will install the latest version of the RISC Zero toolchain. For a specific version, use:
rzup install cargo-risczero <version>
Replace <version>
with the desired release tag (e.g., v1.1.1
).
For manual installation or systems not supported by rzup
(e.g., x86-64 macOS, arm64 Linux), follow these steps:
git clone https://github.com/risc0/risc0.git cd risc0 cargo install --path rzup rzup toolchain build rust
For x86-64 Linux and arm64 macOS, install the C++ toolchains by running:
rzup install cpp
To update your RISC Zero tools to the latest version, run:
rzup update
After updating, ensure your project's RISC Zero dependencies are also updated. Check the version with:
cargo risczero --version
As of March 18, 2025, the latest version is cargo-risczero 1.2.5
. Always check for updates before use.
After running curl -L https://risczero.com/install | bash
, you should see:
% Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed 100 15 0 15 0 0 57 0 --:--:-- --:--:-- --:--:-- 57 100 5947 100 5947 0 0 3926 0 0:00:01 0:00:01 --:--:-- 6980 → Starting rzup installation... → Downloading rzup... ######################################################################## 100.0% → Configuring shell... ✓ Updated shell profile ✓ Installation complete! rzup is installed in /home/zixine/.risc0/bin To get started: 1. Restart your shell or run: source '/home/zixine/.bashrc' 2. Verify the installation: rzup --help
To harness the power of ZK, you'll need to:
Welcome to the zkVM Quick Start page! Here are the steps to create your first proof:
rzup
.cargo-risczero
tool.dev-mode
.The zkVM provides a zero-knowledge proof of the correct execution of Rust-based programs. The host is responsible for launching the zkVM (or a prover), and the guest is the program that runs inside the zkVM. A proof of execution is a receipt; it contains a public part (a journal) and an encryption part (a seal).
Install rzup
by running the following command:
curl -L https://risczero.com/install | bash
Run rzup
to install the RISC Zero toolchain and cargo-risczero
:
rzup install
The cargo-risczero
tool takes a --guest-name
parameter, which is the guest program that the zkVM runs, to generate a proof of its execution:
cargo risczero new my_project --guest-name guest_code_for_zk_proof
There is a list of options in the feature flags.
The Hello World tutorial contains step-by-step instructions on how to:
During development, running your code can take a long time due to the proof generation process. To address this issue and allow for faster iterations, use dev-mode
. This mode bypasses the time-consuming proof generation process. To activate dev-mode
, set the environment variable RISC0_DEV_MODE=1
when executing your project:
RISC0_DEV_MODE=1 cargo run --release
Once you're ready to generate real proofs, set RISC0_DEV_MODE=0
. Generating proofs locally can be done by running:
RISC0_DEV_MODE=0 cargo run --release
Note that generating proofs will take significantly longer than running in dev-mode
. For optimal performance, we recommend at least 16 GB of RAM. Alternatively, consider using Bonsai to generate proofs remotely.
To gain insights into your application's performance, you can obtain executor statistics by setting the RUST_LOG
environment variable to info
:
RISC0_DEV_MODE=1 RUST_LOG=info RISC0_INFO=1 cargo run --release
The statistics include:
The following feature flags are present in one or more of the crates listed above:
Feature | Target(s) | Implies | Description | Crates |
---|---|---|---|---|
client | all except rv32im | std | Enables the client API. | risc0-zkvm |
cuda | prove, std | Enables CUDA GPU acceleration for the prover. Requires CUDA toolkit to be installed. | risc0-circuit-recursion, risc0-circuit-rv32im, risc0-zkp, risc0-zkvm | |
disable-dev-mode | all except rv32im | Disables dev mode so that proving and verifying may not be faked. Used to prevent a misplaced RISC0_DEV_MODE from breaking security in production systems. | risc0-zkvm | |
metal | macos | prove, std | Deprecated - Metal GPU acceleration is enabled by default on Apple Silicon. | risc0-circuit-recursion, risc0-circuit-rv32im, risc0-zkp, risc0-zkvm |
prove | all except rv32im | std | Enables the prover, incompatible within the zkvm guest. | risc0-circuit-recursion, risc0-circuit-rv32im, risc0-zkp, risc0-zkvm |
std | all | Support for the Rust stdlib. | risc0-circuit-recursion, risc0-circuit-rv32im, risc0-zkp, risc0-zkvm |
That's all it takes to build and run a minimal RISC Zero application.
Verifiable computation is a game changer for the resilience and economics of operating the computing infrastructure we all rely on. It creates a number of emergent use cases which we are excited to enable. Key among these are:
The RISC Zero zkVM can prove the correct execution of arbitrary code, allowing developers to build ZK applications in mature languages like Rust and C++.
This tutorial will walk you through building your first zkVM application. By following the steps in this guide, you will learn how to:
cargo-risczero
tool to create a new project.Firstly, visit the installation page for how to install the necessary software.
Once installed, clone the risc0
monorepo and change into the hello-world
example directory:
git clone https://github.com/risc0/risc0 cd risc0/examples/hello-world
Next, check the version of cargo risczero
installed:
cargo risczero --version
To match the example release version with your local installation, check out the corresponding branch of the example:
git checkout release-$MAJOR.$MINOR # e.g. release-1.2
Once on the correct release branch, run the example:
cargo run --release
The host code is located in hello-world/src/main.rs
and hello-world/src/lib.rs
. The host creates an executor environment ExecutorEnv
before constructing a prover. This executor environment is responsible for managing guest-readable memory.
// hello-world/src/lib.rs
use risc0_zkvm::{default_prover, ExecutorEnv, Receipt};
pub fn multiply(a: u64, b: u64) -> (Receipt, u64) {
let env = ExecutorEnv::builder()
// Send a & b to the guest
.write(&a)
.unwrap()
.write(&b)
.unwrap()
.build()
.unwrap();
// Obtain the default prover.
let prover = default_prover();
// Produce a receipt by proving the specified ELF binary.
let receipt = prover.prove(env, MULTIPLY_ELF).unwrap().receipt;
// Extract journal of receipt (i.e. output c, where c = a * b)
let c: u64 = receipt.journal.decode().expect(
"Journal output should deserialize into the same types (& order) that it was written",
);
// Report the product
println!("I know the factors of {}, and I can prove it!", c);
(receipt, c)
}
The guest code is located in methods/guest/src/main.rs
. This is the portion of the code that will be proven. The guest reads the two values a
and b
from the host and commits them to the journal portion of the receipt.
// hello-world/methods/guest/src/main.rs
use risc0_zkvm::guest::env;
risc0_zkvm::guest::entry!(main);
fn main() {
// Load the first number from the host
let a: u64 = env::read();
// Load the second number from the host
let b: u64 = env::read();
// Verify that neither of them are 1 (i.e. nontrivial factors)
if a == 1 || b == 1 {
panic!("Trivial factors")
}
// Compute the product while being careful with integer overflow
let product = a.checked_mul(b).expect("Integer overflow");
env::commit(&product);
}
The host generates a receipt and extracts the contents of the journal:
// hello-world/src/main.rs
use hello_world::multiply;
use hello_world_methods::MULTIPLY_ID;
fn main() {
tracing_subscriber::fmt()
.with_env_filter(tracing_subscriber::EnvFilter::from_default_env())
.init();
// Pick two numbers
let (receipt, _) = multiply(17, 23);
// Verify receipt, panic if it's wrong
receipt.verify(MULTIPLY_ID).expect(
"Code you have proven should successfully verify; did you specify the correct image ID?",
);
}
To run this example locally, and see if proving completed successfully, run:
cargo run --release
If proving is successful, the line 'I know the factors of 391, and I can prove it!'
is printed.
If you encounter errors during cloning or execution, try the following commands:
git config --global http.postBuffer 524288000 git config --global http.lowSpeedLimit 0 git config --global http.lowSpeedTime 999999 git config --global core.compression 0
If the issue persists, ensure you are on the correct release branch:
git checkout release-1.2 cargo clean cargo run --release
You can add more tests to verify the functionality of the multiply
function:
#[test]
fn test_multiply() {
let a: i64 = 5;
let b: i64 = 10;
let (_, result) = multiply(a as u64, b as u64);
assert_eq!(result, 50, "Multiply function failed");
}
#[test]
fn test_multiply_zero() {
let a: i64 = 0;
let b: i64 = 10;
let (_, result) = multiply(a as u64, b as u64);
assert_eq!(result, 0, "Multiply function failed with zero");
}
#[test]
fn test_multiply_two_negatives() {
let a: i64 = -7;
let b: i64 = -8;
let (_, result) = multiply(a.unsigned_abs(), b.unsigned_abs());
assert_eq!(result as i64, a * b, "Multiply function failed with two negatives");
}
#[test]
fn test_multiply_large_numbers() {
let a: i64 = 100000;
let b: i64 = 100000;
let (_, result) = multiply(a as u64, b as u64);
assert_eq!(result, 10000000000, "Multiply function failed with large numbers");
}
Here are some additional resources to help you get started with RISC Zero: