From 58e6aa30477eaba634fdeacf133462eb46004f27 Mon Sep 17 00:00:00 2001 From: "xander.z" <162873981+xander42280@users.noreply.github.com> Date: Mon, 16 Dec 2024 12:38:24 +0800 Subject: [PATCH] feat: support zkm builder (#41) * Add guest build * Update README * Fix fmt & ci * Fix cargo fmt * ZKM_SKIP_PROGRAM_BUILD * Update README * Set ELF_PATH * Fix ELF_PATH fmt * Set default elf_path * Update default elf path * Update guest target path --- .github/workflows/ci.yml | 1 + README.md | 165 +------------------------ guest-program/README.md | 2 + host-program/mem-alloc-vec/Cargo.toml | 1 + host-program/mem-alloc-vec/build.rs | 14 +++ host-program/mem-alloc-vec/src/main.rs | 2 +- host-program/revme/Cargo.toml | 1 + host-program/revme/build.rs | 11 ++ host-program/revme/src/main.rs | 2 +- host-program/run-proving.sh | 7 +- host-program/sha2-go/src/main.rs | 7 +- host-program/sha2-rust/Cargo.toml | 1 + host-program/sha2-rust/build.rs | 14 +++ host-program/sha2-rust/src/main.rs | 4 +- 14 files changed, 57 insertions(+), 175 deletions(-) create mode 100644 host-program/mem-alloc-vec/build.rs create mode 100644 host-program/revme/build.rs create mode 100644 host-program/sha2-rust/build.rs diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index f5d83a73..682b9557 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -10,6 +10,7 @@ on: env: CARGO_TERM_COLOR: always + ZKM_SKIP_PROGRAM_BUILD: true jobs: build_and_test: diff --git a/README.md b/README.md index aacf54ea..fef4645f 100644 --- a/README.md +++ b/README.md @@ -16,59 +16,6 @@ There are two ways to prove the guest program: > [!NOTE] > The SDK has a libary(libsnark) which supports local proving. If the libsnark is required, please specify the features = ["snark"] in your Cargo.toml. To disable libsnark, set the environment variable NO_USE_SNARK to true when compiling the SDK. -``` -├── Cargo.toml -├── LICENSE -├── Makefile -├── README.md -├── assets -│   └── temp-run-diagram.png -├── clippy.toml -├── contracts //Use Foundry to manage the verifier contract -│   ├── README.md -│   ├── foundry.toml -│   //... -├── guest-program //Include Go and Rust examples -│   ├── README.md -│   ├── mem-alloc-vec -│   ├── sha2-go -│   └── sha2-rust -│   -├── host-program //Generate the proof and verifier contracts for the guest programs -│   ├── mem-alloc-vec -│   │   ├── Cargo.toml -│   │   ├── run-proving.sh -│   │   └── src -│   │   └── main.rs -│   ├── revme -│   │   ├── Cargo.toml -│   │   ├── run-proving.sh -│   │   └── src -│   │   └── main.rs -│   ├── sha2-go -│   │   ├── Cargo.toml -│   │   ├── run-proving.sh -│   │   └── src -│   │   └── main.rs -│   ├── sha2-rust -│   │   ├── Cargo.toml -│   │   ├── run-proving.sh -│   │   └── src -│   │   └── main.rs -├── install_mips_rust_tool -├── rust-toolchain.toml -├── sdk //Support proof network and local proof -    ├── Cargo.toml -  ├── build.rs -    └── src -   ├── lib.rs -   ├── local //Generate the proof locally using the libsnark library. -   ├── network //Generate the proof using ZKM Proof Network. -   ├── proto -   │   └── stage.proto -   └── prover.rs //interface -``` - ## Local Proving Requirements - Hardware: X86_64 CPU, 32 cores, 13GB memory (minimum) @@ -115,6 +62,8 @@ If successfully, it will generate the binary files in `target/release`/{`sha2-ru > [!NOTE] > You can run the guest program without generating a proof by setting the environmental variable `EXECUTE_ONLY` to "true".https://github.com/zkMIPS/zkm/issues/152 +> You can set the `ZKM_SKIP_PROGRAM_BUILD` environment variable to `true` to skip building the guest program when use `zkm_build::build_program`. + ### 3. Generate groth16 proof and verifier contract > [!NOTE] @@ -143,59 +92,11 @@ This host program sends the private input pri_input = vec![5u8; 1024] and its ha Make any edits to [`run-proving.sh`](host-program/run-proving.sh) and run the program: - ```sh cd zkm-project-template/host-program/sha2-rust ./run-proving.sh sha2-rust ``` -If successful, it will output a similar log: - -##### **`sha2-rust-local-proving.log`** - -``` -[2024-11-23T13:12:33Z INFO sha2_rust] new prover client,ok. -[2024-11-23T13:12:33Z INFO zkm_sdk] excuting the setup. -[2024-11-23T13:12:33Z INFO zkm_emulator::utils] Split done 66446 : 89443 -[2024-11-23T13:21:05Z INFO zkm_sdk::local::stark] [the seg_num is:2 ] -[2024-11-23T13:21:55Z INFO zkm_sdk::local::util] Process segment /mnt/data/gavin/zkm-project-template/host-program/sha2-rust/../test-vectors/input/segments/0 -[2024-11-23T13:21:59Z INFO zkm_prover::cpu::bootstrap_kernel] Bootstrapping took 3228 cycles -[2024-11-23T13:21:59Z INFO zkm_prover::generation] CPU halted after 64762 cycles -[2024-11-23T13:21:59Z INFO zkm_prover::generation] CPU trace padded to 65536 cycles -[2024-11-23T13:21:59Z INFO zkm_prover::generation] Trace lengths (before padding): TraceCheckpoint { arithmetic_len: 18057, cpu_len: 65536, poseidon_len: 3227, poseidon_sponge_len: 3227, logic_len: 15372, memory_len: 390244 } -[2024-11-23T13:22:14Z INFO plonky2::util::timing] 19.1346s to prove root first -[2024-11-23T13:22:15Z INFO zkm_sdk::local::util] Process segment /mnt/data/gavin/zkm-project-template/host-program/sha2-rust/../test-vectors/input/segments/1 -[2024-11-23T13:22:18Z INFO zkm_prover::cpu::bootstrap_kernel] Bootstrapping took 2583 cycles -[2024-11-23T13:22:18Z INFO zkm_prover::generation] CPU halted after 7530 cycles -[2024-11-23T13:22:18Z INFO zkm_prover::generation] CPU trace padded to 8192 cycles -[2024-11-23T13:22:18Z INFO zkm_prover::generation] Trace lengths (before padding): TraceCheckpoint { arithmetic_len: 1511, cpu_len: 8192, poseidon_len: 2582, poseidon_sponge_len: 2582, logic_len: 1157, memory_len: 123358 } -[2024-11-23T13:22:29Z INFO plonky2::util::timing] 14.4273s to prove root second -[2024-11-23T13:22:30Z INFO plonky2::util::timing] 1.3184s to prove aggression -[2024-11-23T13:22:32Z INFO zkm_sdk::local::util] proof size: 413003 -[2024-11-23T13:22:38Z INFO zkm_sdk::local::util] build finish -[2024-11-23T13:22:45Z INFO plonky2x::backend::wrapper::wrap] Succesfully wrote common circuit data to common_circuit_data.json -[2024-11-23T13:22:45Z INFO plonky2x::backend::wrapper::wrap] Succesfully wrote verifier data to verifier_only_circuit_data.json -[2024-11-23T13:22:45Z INFO plonky2x::backend::wrapper::wrap] Succesfully wrote proof to proof_with_public_inputs.json -[2024-11-23T13:22:45Z INFO plonky2::util::timing] 99.1857s to prove total time -Generating witness 2024-11-23 13:24:17.684495304 +0000 UTC m=+703.805549132 -frontend.NewWitness cost time: 143 ms -Creating proof 2024-11-23 13:24:17.828400392 +0000 UTC m=+703.949454200 -13:24:23 DBG constraint system solver done nbConstraints=5815132 took=5606.081295 -13:24:30 DBG prover done acceleration=none backend=groth16 curve=bn254 nbConstraints=5815132 took=7527.912415 -groth16.Prove cost time: 13134 ms -Verifying proof 2024-11-23 13:24:30.962722326 +0000 UTC m=+717.083776144 -13:24:30 DBG verifier done backend=groth16 curve=bn254 took=1.441281 -groth16.Verify cost time: 1 ms -before len of publicWitness:1 -after len of publicWitness:2 -13:24:30 DBG verifier done backend=groth16 curve=bn254 took=1.288129 -[2024-11-23T13:24:30Z INFO sha2_rust] Proof: successfully written 1268 bytes. -[2024-11-23T13:24:30Z INFO sha2_rust] hash(bincode(pulic_input))1: [197, 85, 237, 192, 203, 240, 69, 67, 7, 62, 23, 5, 10, 112, 210, 80, 40, 245, 196, 9, 255, 152, 190, 127, 32, 148, 73, 249, 212, 64, 168, 103] -[2024-11-23T13:24:30Z INFO sha2_rust] Contract: successfully written 12961 bytes. -[2024-11-23T13:24:30Z INFO sha2_rust] Generating proof successfully .The proof file and verifier contract are in the the path /mnt/data/gavin/zkm-project-template/host-program/sha2-rust/../../contracts/{verifier,src} . -[2024-11-23T13:24:30Z INFO sha2_rust] Elapsed time: 205 secs -``` - The result proof and contract file will be in the contracts/verifier and contracts/src respectively. #### Network Proving @@ -211,24 +112,6 @@ Must set the `PRIVATE_KEY` and `ZKM_PROVER=network` in [`run-proving.sh`](host-p ./run-proving.sh sha2-rust ``` -If successful, it will output a similar log: - -##### **`sha2-rust-network-proving.log`** - -``` -[2024-11-23T10:13:04Z INFO sha2_rust] new prover client. -[2024-11-23T10:13:05Z INFO sha2_rust] new prover client,ok. -[2024-11-23T10:13:05Z INFO zkm_sdk::network::prover] calling request_proof. -[2024-11-23T10:13:08Z INFO zkm_sdk::network::prover] calling wait_proof, proof_id=31134979-8e99-4de6-988c-46211df28c80 -[2024-11-23T10:13:08Z INFO zkm_sdk::network::prover] generate_proof : proving the task. -[2024-11-23T10:13:38Z INFO zkm_sdk::network::prover] generate_proof : finalizing the proof. -[2024-11-23T10:14:11Z INFO sha2_rust] Proof: successfully written 1263 bytes. -[2024-11-23T10:14:11Z INFO sha2_rust] hash(bincode(pulic_input))1: [197, 85, 237, 192, 203, 240, 69, 67, 7, 62, 23, 5, 10, 112, 210, 80, 40, 245, 196, 9, 255, 152, 190, 127, 32, 148, 73, 249, 212, 64, 168, 103] -[2024-11-23T10:14:11Z INFO sha2_rust] Contract: successfully written 12960 bytes. -[2024-11-23T10:14:11Z INFO sha2_rust] Generating proof successfully .The proof file and verifier contract are in the the path /mnt/data/gavin/zkm-project-template/host-program/sha2-rust/../../contracts/{verifier,src} . -[2024-11-23T10:14:11Z INFO sha2_rust] Elapsed time: 66 secs -``` - The result proof and contract file will be in the contracts/verifier and contracts/src. ### 4. Deploy the Verifier Contract @@ -245,20 +128,6 @@ cd zkm-project-template/contracts forge test ``` -If successful, it will output a similar log: - -``` -[⠊] Compiling... -No files changed, compilation skipped - -Ran 2 tests for test/verifier.t.sol:VerifierTest -[PASS] test_ValidProof() (gas: 287072) -[PASS] test_ValidPublicInputs() (gas: 67184) -Suite result: ok. 2 passed; 0 failed; 0 skipped; finished in 8.51ms (8.64ms CPU time) - -Ran 1 test suite in 9.28ms (8.51ms CPU time): 2 tests passed, 0 failed, 0 skipped (2 total tests) -``` - #### Deploy the contract generateing in the step 3 Please edit the following parameters according your aim blockchain. @@ -267,36 +136,6 @@ Please edit the following parameters according your aim blockchain. forge script script/verifier.s.sol:VerifierScript --rpc-url https://eth-sepolia.g.alchemy.com/v2/RH793ZL_pQkZb7KttcWcTlOjPrN0BjOW --private-key df4bc5647fdb9600ceb4943d4adff3749956a8512e5707716357b13d5ee687d9 ``` -If successful, it will output a similar log: - -``` -[⠊] Compiling... -[⠘] Compiling 2 files with Solc 0.8.26 -[⠊] Solc 0.8.26 finished in 699.26ms -Compiler run successful! -Script ran successfully. - -## Setting up 1 EVM. - -========================== - -Chain 11155111 - -Estimated gas price: 0.000035894 gwei - -Estimated total gas used for script: 1228147 - -Estimated amount required: 0.000000044083108418 ETH - -========================== - -SIMULATION COMPLETE. To broadcast these transactions, add --broadcast and wallet configuration(s) to the previous command. See forge script --help for more. - -Transactions saved to: /mnt/data/zkm-project-template/contracts/broadcast/verifier.s.sol/11155111/dry-run/run-latest.json - -Sensitive values saved to: /mnt/data/zkm-project-template/contracts/cache/verifier.s.sol/11155111/dry-run/run-latest.json -``` - For more details, please refer to [this](contracts/README.md) guide. ### Example 2 : `revme` diff --git a/guest-program/README.md b/guest-program/README.md index 33674c09..b207c772 100644 --- a/guest-program/README.md +++ b/guest-program/README.md @@ -60,6 +60,8 @@ cargo build --target=mips-unknown-linux-musl --release The compiled mips ELF is in the zkm-project-template/guest-program/{sha2-rust,mem-alloc-vec,revme}/target/mips-unknown-linux-musl/release/ . +You can also integrate `zkm_build::build_program` into the compilation process of the host program. + ## Remarks If the guest program need outputing some messages , it must use the runtime::commit(). Then, the messages can be catched in the host program: [`fn print_guest_execution_output() or print_guest_execution_output_struct()`](../sdk/src/lib.rs) diff --git a/host-program/mem-alloc-vec/Cargo.toml b/host-program/mem-alloc-vec/Cargo.toml index a7f80ce5..b048f7e6 100644 --- a/host-program/mem-alloc-vec/Cargo.toml +++ b/host-program/mem-alloc-vec/Cargo.toml @@ -14,6 +14,7 @@ env_logger = "0.10.0" anyhow = "1.0.75" [build-dependencies] tonic-build = "0.8.0" +zkm-build = { git = "https://github.com/zkMIPS/zkm", branch = "main", default-features = false } [features] test = [] diff --git a/host-program/mem-alloc-vec/build.rs b/host-program/mem-alloc-vec/build.rs new file mode 100644 index 00000000..044ace5b --- /dev/null +++ b/host-program/mem-alloc-vec/build.rs @@ -0,0 +1,14 @@ +fn main() { + let guest_path = format!( + "{}/../../guest-program/mem-alloc-vec", + env!("CARGO_MANIFEST_DIR") + ); + zkm_build::build_program(&guest_path); + let guest_target_path = format!( + "{}/{}/{}", + guest_path, + zkm_build::DEFAULT_OUTPUT_DIR, + zkm_build::BUILD_TARGET + ); + println!("cargo:rustc-env=GUEST_TARGET_PATH={}", guest_target_path); +} diff --git a/host-program/mem-alloc-vec/src/main.rs b/host-program/mem-alloc-vec/src/main.rs index 0120ed00..565aa7c7 100644 --- a/host-program/mem-alloc-vec/src/main.rs +++ b/host-program/mem-alloc-vec/src/main.rs @@ -26,7 +26,7 @@ async fn main() -> Result<()> { .and_then(|seg| seg.parse::().ok()) .unwrap_or(false); - let elf_path = env::var("ELF_PATH").expect("ELF PATH is missed"); + let elf_path = env::var("ELF_PATH").unwrap_or(env!("GUEST_TARGET_PATH").to_string()); let proof_results_path = env::var("PROOF_RESULTS_PATH").unwrap_or("../contracts".to_string()); let vk_path = env::var("VERIFYING_KEY_PATH").unwrap_or("/tmp/input".to_string()); diff --git a/host-program/revme/Cargo.toml b/host-program/revme/Cargo.toml index f629a1b9..a60f714d 100644 --- a/host-program/revme/Cargo.toml +++ b/host-program/revme/Cargo.toml @@ -14,6 +14,7 @@ env_logger = "0.10.0" anyhow = "1.0.75" [build-dependencies] tonic-build = "0.8.0" +zkm-build = { git = "https://github.com/zkMIPS/zkm", branch = "main", default-features = false } [features] test = [] diff --git a/host-program/revme/build.rs b/host-program/revme/build.rs new file mode 100644 index 00000000..8ce911c4 --- /dev/null +++ b/host-program/revme/build.rs @@ -0,0 +1,11 @@ +fn main() { + let guest_path = format!("{}/../../guest-program/revme", env!("CARGO_MANIFEST_DIR")); + zkm_build::build_program(&guest_path); + let guest_target_path = format!( + "{}/{}/{}", + guest_path, + zkm_build::DEFAULT_OUTPUT_DIR, + zkm_build::BUILD_TARGET + ); + println!("cargo:rustc-env=GUEST_TARGET_PATH={}", guest_target_path); +} diff --git a/host-program/revme/src/main.rs b/host-program/revme/src/main.rs index f20b7440..ca532dcf 100644 --- a/host-program/revme/src/main.rs +++ b/host-program/revme/src/main.rs @@ -26,7 +26,7 @@ async fn main() -> Result<()> { .and_then(|seg| seg.parse::().ok()) .unwrap_or(false); - let elf_path = env::var("ELF_PATH").expect("ELF PATH is missed"); + let elf_path = env::var("ELF_PATH").unwrap_or(env!("GUEST_TARGET_PATH").to_string()); let json_path = env::var("JSON_PATH").expect("JSON PATH is missing"); let proof_results_path = env::var("PROOF_RESULTS_PATH").unwrap_or("../contracts".to_string()); let vk_path = env::var("VERIFYING_KEY_PATH").unwrap_or("/tmp/input".to_string()); diff --git a/host-program/run-proving.sh b/host-program/run-proving.sh index e92b8e4e..7290aa4f 100755 --- a/host-program/run-proving.sh +++ b/host-program/run-proving.sh @@ -12,7 +12,6 @@ export LD_LIBRARY_PATH=$BASEDIR/../sdk/src/local/libsnark:$LD_LIBRARY_PATH ##Mo export RUST_LOG=info export SEG_SIZE=262144 export ARGS="711e9609339e92b03ddc0a211827dba421f38f9ed8b9d806e1ffdd8c15ffa03d world!" -export ELF_PATH=${BASEDIR}/../guest-program/$program/target/mips-unknown-linux-musl/release/$program export JSON_PATH=${BASEDIR}/test-vectors/test.json export PROOF_RESULTS_PATH=${BASEDIR}/../contracts export EXECUTE_ONLY=false @@ -28,12 +27,8 @@ echo "Compile guest-program ${program}" if [[ "$program" =~ .*go$ ]];then cd $BASEDIR/../guest-program/$program GOOS=linux GOARCH=mips GOMIPS=softfloat go build -o $program - export ELF_PATH=${BASEDIR}/../guest-program/$program/$program -else - cd $BASEDIR/../guest-program/$program - cargo build -r --target=mips-unknown-linux-musl + cd - fi -cd - if [ "$program" == "sha2-rust" ];then export SEG_SIZE=65536 diff --git a/host-program/sha2-go/src/main.rs b/host-program/sha2-go/src/main.rs index 33610e28..90d4f19a 100644 --- a/host-program/sha2-go/src/main.rs +++ b/host-program/sha2-go/src/main.rs @@ -9,6 +9,11 @@ use zkm_sdk::{prover::ClientCfg, prover::ProverInput, ProverClient}; pub const DEFAULT_PROVER_NETWORK_RPC: &str = "https://152.32.186.45:20002"; pub const DEFALUT_PROVER_NETWORK_DOMAIN: &str = "stage"; +const GUEST_TARGET_PATH: &str = concat!( + env!("CARGO_MANIFEST_DIR"), + "/../../guest-program/sha2-go/sha2-go" +); + #[tokio::main] async fn main() -> Result<()> { env_logger::try_init().unwrap_or_default(); @@ -26,7 +31,7 @@ async fn main() -> Result<()> { .and_then(|seg| seg.parse::().ok()) .unwrap_or(false); - let elf_path = env::var("ELF_PATH").expect("ELF PATH is missed"); + let elf_path = env::var("ELF_PATH").unwrap_or(GUEST_TARGET_PATH.to_string()); let args_parameter = env::var("ARGS").unwrap_or("data-to-hash".to_string()); //let json_path = env::var("JSON_PATH").expect("JSON PATH is missing"); let proof_results_path = env::var("PROOF_RESULTS_PATH").unwrap_or("../contracts".to_string()); diff --git a/host-program/sha2-rust/Cargo.toml b/host-program/sha2-rust/Cargo.toml index 530b0447..1b47a7fc 100644 --- a/host-program/sha2-rust/Cargo.toml +++ b/host-program/sha2-rust/Cargo.toml @@ -19,6 +19,7 @@ env_logger = "0.10.0" [build-dependencies] tonic-build = "0.8.0" +zkm-build = { git = "https://github.com/zkMIPS/zkm", branch = "main", default-features = false } [features] test = [] diff --git a/host-program/sha2-rust/build.rs b/host-program/sha2-rust/build.rs new file mode 100644 index 00000000..b7a85aa1 --- /dev/null +++ b/host-program/sha2-rust/build.rs @@ -0,0 +1,14 @@ +fn main() { + let guest_path = format!( + "{}/../../guest-program/sha2-rust", + env!("CARGO_MANIFEST_DIR") + ); + zkm_build::build_program(&guest_path); + let guest_target_path = format!( + "{}/{}/{}", + guest_path, + zkm_build::DEFAULT_OUTPUT_DIR, + zkm_build::BUILD_TARGET + ); + println!("cargo:rustc-env=GUEST_TARGET_PATH={}", guest_target_path); +} diff --git a/host-program/sha2-rust/src/main.rs b/host-program/sha2-rust/src/main.rs index 8c523f7e..1e36bbee 100644 --- a/host-program/sha2-rust/src/main.rs +++ b/host-program/sha2-rust/src/main.rs @@ -27,9 +27,7 @@ async fn main() -> Result<()> { .and_then(|seg| seg.parse::().ok()) .unwrap_or(false); - let elf_path = env::var("ELF_PATH").expect("ELF PATH is missed"); - //let args_parameter = env::var("ARGS").unwrap_or("data-to-hash".to_string()); - //let json_path = env::var("JSON_PATH").expect("JSON PATH is missing"); + let elf_path = env::var("ELF_PATH").unwrap_or(env!("GUEST_TARGET_PATH").to_string()); let proof_results_path = env::var("PROOF_RESULTS_PATH").unwrap_or("../contracts".to_string()); let vk_path = env::var("VERIFYING_KEY_PATH").unwrap_or("/tmp/input".to_string());