Skip to content

Latest commit

 

History

History
175 lines (105 loc) · 11.4 KB

experimental-features.md

File metadata and controls

175 lines (105 loc) · 11.4 KB

Experimental features

In privacy-scaling-explorations/halo2 fork we have implemented many experimental features to cover different use cases, especially for those with huge circuits. We collect them in this page for easier tracking and referencing.

Commitment scheme abstraction

To support different kinds of polynomial commitment schemes, we've added a trait CommitmentScheme to allow create/verify proofs with different commitment scheme implementations, currently there are 2 available implementations in this fork:

  • IPACommitmentScheme

    The original implementation from zcash/halo2 with the original multi-open strategy {Prover,Verifier}IPA

  • KZGCommitmentScheme

    KZG commitment scheme as in GWC19, which doesn't commit the instance columns, with 2 multi-open strategies available:

    • {Prover,Verifier}GWC - The original strategy in GWC19
    • {Prover,Verifier}SHPLONK - The strategy proposed in BDFG20

When using create_proof and verify_proof, we need to specify the commitment scheme and multi-open strategy like:

// Using IPA
create_proof<IPACommitmentScheme<_>, ProverIPA<_>, _, _, _, _>
verify_proof<IPACommitmentScheme<_>, ProverIPA<_>, _, _, _>

// Using KZG with GWC19 multi-open strategy
create_proof<KZGCommitmentScheme<_>, ProverGWC<_>, _, _, _, _>
verify_proof<KZGCommitmentScheme<_>, ProverGWC<_>, _, _, _>

// Using KZG with BDFG20 multi-open strategy
create_proof<KZGCommitmentScheme<_>, ProverSHPLONK<_>, _, _, _, _>
verify_proof<KZGCommitmentScheme<_>, ProverSHPLONK<_>, _, _, _>

ConstraintSystem extension

Dynamic lookup

ConstraintSystem::lookup_any is added for use cases that need to lookup dynamic table instead of fixed table.

Unlike ConstraintSystem::lookup which only allows TableColumn(s) as table, it allows any Expression(s) without simple selector.

Shuffle

ConstraintSystem::shuffle is added for use cases that only need shuffle without pre-defined mapping.

It allows us to prove any Expression(s) without simple selector is shuffled from the other. For example halo2_proofs/examples/shuffle_api.rs shows how to prove two lists of 2-tuples are shuffled of each other.

Multi-phase

ConstraintSystem::advice_column_in and ConstraintSystem::challenge_usable_after are added for use cases that build PIOP sub-routine themselves, currently it supports up-to 3 phases as {First,Second,Third}Phase.

It allows us to allocate advice column in different interactive phases with extra challenges squeezed in-between. For example in halo2_proofs/examples/shuffle.rs it shows how to build a customized shuffle argument with such API.

Unblinded advice column

ConstraintSystem::unblinded_advice_column is added for use cases that want to reuse advice column commitment among different proofs. For example in halo2_proofs/examples/vector-ops-unblinded.rs it shows with this API and same assignment, two advice commitment from different proof can be same.

Worth mentioning, re-using advice column commitment in different proofs will need more blinding factors than the amount that prover adds, otherwise some information will be leaked and it's no longer perfect zero-knowledge.

Expression extension

Circuit extension

  • Circuit::Params

    To use this, feature circuit-params needs to be turned on.

    A associated type added for configuring circuit with runtime parameter.

    It allows us to implement Circuit::params to return the parameter of a circuit, and implement Circuit::configure_with_params to configure circuit with runtime parameter retrieved earlier.

ProvingKey & VerifyingKey de/serialization and SerdeFormat

ProvingKey::{read,write} and VerifyingKey::{read,write} is added to serialize proving key and verifying key.

For field elements and elliptic curve points inside pk and vk, currently we have 3 different de/serialization format:

  • SerdeFormat::Processed

    It de/serializes them as PrimeField::Repr and GroupEncoding::Repr respectively, and checks all elements are valid.

  • SerdeFormat::RawBytes

    It de/serializes them as SerdeObject::{from_raw_bytes,to_raw_bytes} respectively, and checks all elements are valid.

  • SerdeFormat::RawBytesUnchecked

    It de/serializes them as SerdeObject::{from_raw_bytes,to_raw_bytes} respectively, without checking if elements are valid.

Also ParamsKZG::{read_custom,write_custom} follows the same rule, and by default ParamsKZG::{read,write} uses SerdeFormat::RawBytes for efficiency.

Thread safe Region

To use this, feature thread-safe-region needs to be turned on.

It constrains the RegionLayouter to be Send so we can have a Region in different threads. It's useful when we want to arrange witness computation and assignment in the same place, but still keep the function Send so the caller can parallelize multiple of them.

For example halo2_proofs/examples/vector-mul.rs shows how to parallelize region computation and assignment.

Optional selector compression

Currently keygen_vk changes configured ConstraintSystem to compresses simple selectors into smaller set of fixed columns to reduce cost.

For some use cases that want to keep configured ConstraintSystem unchanged they can do the verifying key generation by calling keygen_vk_custom with second argument as false instead, which disables the selector compression.

MockProver improvement

Evaluator and evaluate_h

They are introduced to improve quotient computation speed and memory usage for circuit with complicated Expression.

Modular design (frontend-backend split)

The halo2 implementation has been split into two separate parts: the frontend and the backend, following these definitions:

  • frontend: allows the user to specify the circuit logic and its satisfying witness. It must provide a way to translate this logic into a low level arithmetization format specified in the middleware module.
  • backend: the proving system implementation that receives the middleware circuit arithmetization and performs the following tasks:
    • Generate the setup (proving and verifying keys)
    • Generate a proof (with witness as input)
    • Verify a proof

A note on naming: "halo2" can mean different things:

  • halo2 proof system, the protocol
  • halo2 proof system implementation, the backend
  • halo2 circuit library, the frontend (includes the halo2 circuit API, the layouter, the selector to fixed column transformation, etc.)
  • halo2 full-stack, the proof system full stack (the combination of the backend and frontend)

Currently the backend implements the "original" halo2 proof system extended with the features discussed in this document. Nevertheless, the public interface that the backend uses is generic for plonkish arithmetization. This allows for alternative frontend implementations as well as alternative plonkish proof system implementations. The middleware contains the type definitions used to connect the frontend and backend.

Summary of crates:

  • halo2_frontend: library used to define circuits and calculate their witness.
  • halo2_backend: implementation of the halo2 proof system (the protocol).
  • halo2_middleware: type definitions used to interface the backend with the frontend.
  • halo2_proofs: legacy API built by re-exporting from the frontend and backend as well as function wrappers.