Compiler APIs
For circuit developers, most of the necessary APIs are provided in the expander_compiler::frontend
module. If you need to perform more advanced development on top of the compiler, you may need to use other components. This document primarily discusses the contents of the frontend
module.
Introduction of other modules can be found in Rust Internal Modules.
The following items are defined in the frontend:
pub use circuit::declare_circuit;
pub type API<C> = builder::RootBuilder<C>;
pub use crate::circuit::config::*;
pub use crate::compile::CompileOptions;
pub use crate::field::{Field, FieldArith, FieldModulus, BN254, GF2, M31};
pub use crate::hints::registry::{EmptyHintCaller, HintCaller, HintRegistry};
pub use crate::utils::error::Error;
pub use api::{BasicAPI, RootAPI};
pub use builder::Variable;
pub use circuit::{Define, GenericDefine};
pub use witness::WitnessSolver;
pub mod internal {
pub use super::circuit::{
declare_circuit_default, declare_circuit_dump_into, declare_circuit_field_type,
declare_circuit_load_from, declare_circuit_num_vars,
};
pub use super::variables::{DumpLoadTwoVariables, DumpLoadVariables};
pub use crate::utils::serde::Serde;
}
pub mod extra {
pub use super::api::UnconstrainedAPI;
pub use super::debug::DebugBuilder;
pub use crate::hints::registry::{EmptyHintCaller, HintCaller, HintRegistry};
pub use crate::utils::serde::Serde;
pub fn debug_eval<
C: Config,
Cir: internal::DumpLoadTwoVariables<Variable> + GenericDefine<C> + Clone,
CA: internal::DumpLoadTwoVariables<C::CircuitField>,
H: HintCaller<C::CircuitField>,
>(
circuit: &Cir,
assignment: &CA,
hint_caller: H,
) {
// implementation
}
}
pub struct CompileResult<C: Config> {
pub witness_solver: WitnessSolver<C>,
pub layered_circuit: layered::Circuit<C, NormalInputType>,
}
pub struct CompileResultCrossLayer<C: Config> {
pub witness_solver: WitnessSolver<C>,
pub layered_circuit: layered::Circuit<C, CrossLayerInputType>,
}
pub fn compile_generic<
C: Config,
Cir: internal::DumpLoadTwoVariables<Variable> + GenericDefine<C> + Clone,
>(
circuit: &Cir,
options: CompileOptions,
) -> Result<CompileResult<C>, Error> {
// implementation
}
pub fn compile_generic_cross_layer<
C: Config,
Cir: internal::DumpLoadTwoVariables<Variable> + GenericDefine<C> + Clone,
>(
circuit: &Cir,
options: CompileOptions,
) -> Result<CompileResultCrossLayer<C>, Error> {
// implementation
}
Declaring a Circuit
The declare_circuit
macro helps define the structure of a circuit. For example:
declare_circuit!(Circuit {
x: Variable,
y: Variable,
});
You can use more complex structures, such as [[Variable; 256]; N_HASHES]
. The defined struct will look like this:
pub struct Circuit<T> {
pub x: T,
pub y: T,
}
Long Arrays
If you need a very long array, you might need to use Vec
instead of a fixed-size array in the structure. Due to Rust's limitations, the syntax for this definition is slightly different from that of Vec
:
declare_circuit!(Circuit {
x: [[[Variable]]],
y: [[Variable]],
z: [[[Variable]; 10]]
});
The actual structure definition will look like this:
pub struct Circuit<T> {
pub x: Vec<Vec<Vec<T>>>,
pub y: Vec<Vec<T>>,
pub z: Vec<[Vec<T>; 10]>,
}
Public Variables
If you need to define public variables, change Variable
to PublicVariable
.
API Overview
The API is similar to gnark
's frontend API. C
represents the configuration for the specified field.
Currently, the Config
and Field
types are one-to-one:
- Fields:
BN254
,GF2
,M31
- Configs:
BN254Config
,GF2Config
,M31Config
Many functions and structs use Config
as a template parameter.
Error Handling
The Error
type is returned by many functions and includes UserError
and InternalError
. UserError
typically indicates an issue with your circuit definition, while InternalError
suggests a problem within the compiler itself. Please contact us if you encounter an InternalError
.
Basic API
The BasicAPI
trait provides a set of operations similar to those in gnark
. The semantics of xor
, or
, and and
are consistent with gnark
.
pub trait BasicAPI<C: Config> {
// These basic functions are consistent with the API in gnark
fn add(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn sub(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn mul(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn div(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>, checked: bool) -> Variable;
fn neg(&mut self, x: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn inverse(&mut self, x: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn is_zero(&mut self, x: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn xor(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn or(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn and(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>) -> Variable;
fn assert_is_zero(&mut self, x: impl ToVariableOrValue<C::CircuitField>);
fn assert_is_non_zero(&mut self, x: impl ToVariableOrValue<C::CircuitField>);
fn assert_is_bool(&mut self, x: impl ToVariableOrValue<C::CircuitField>);
fn assert_is_equal(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>);
fn assert_is_different(&mut self, x: impl ToVariableOrValue<C::CircuitField>, y: impl ToVariableOrValue<C::CircuitField>);
// The way to define hints is different from gnark, requiring a custom string as the hint key
fn new_hint(&mut self, hint_key: &str, inputs: &[Variable], num_outputs: usize) -> Vec<Variable>;
// The get_random_value function generates a random number using Fiat-Shamir during proof generation
fn get_random_value(&mut self) -> Variable;
// Converts a constant to a Variable type
fn constant(&mut self, x: impl ToVariableOrValue<C::CircuitField>) -> Variable;
// Try to get the value of a compile-time constant variable
// This function behaves differently in normal and debug mode; in debug mode, it always returns Some(value)
fn constant_value(&mut self, x: impl ToVariableOrValue<C::CircuitField>) -> Option<C::CircuitField>;
// Used to display the value of a variable during debugging
fn display(&self, _label: &str, _x: impl ToVariableOrValue<C::CircuitField>) {}
}
For the usage of the hint system, please refer to Hints. For displaying values during debugging, please refer to Debugging.
Root API
The RootAPI
trait currently provides a single function for calling a sub-circuit.
pub trait RootAPI<C: Config>: Sized + BasicAPI<C> + UnconstrainedAPI<C> + 'static {
fn memorized_simple_call<F: Fn(&mut Self, &Vec<Variable>) -> Vec<Variable> + 'static>(
&mut self,
f: F,
inputs: &[Variable],
) -> Vec<Variable>;
}
The definition of sub-circuits is similar to that in the Go API. The memorized_simple_call
function accepts a function or closure f
. For an example of its definition or invocation, refer to keccak_gf2.
Variables
The Variable
type is similar to gnark
's frontend Variable
. However, Variable
is a fixed type, so constants need to be converted to variables through the builder.
Define Trait
The Define
and GenericDefine
traits, similar to gnark
's define
, need to be implemented for your circuit structure to call the compiling functions.
// Use this
pub trait GenericDefine<C: Config> {
fn define<Builder: RootAPI<C>>(&self, api: &mut Builder);
}
// This is deprecated
pub trait Define<C: Config> {
fn define(&self, api: &mut RootBuilder<C>);
}
Compiling the Circuit
We primarily provide two functions to compile circuits: compile_generic
and compile_generic_cross_layer
. These functions can compile circuits that implement the GenericDefine
trait. Generally, you should use compile_generic
. The other function is designed for Virgo++, which is still under development and not recommended for use at this time.
Additionally, there is a compile
function for compatibility with circuits that implement the older Define
trait, but it is currently deprecated.
WitnessSolver
The WitnessSolver
is similar to the InputSolver
in the Go version of ExpanderCompilerCollection. It primarily provides the solve_witness
method, along with its various variants, for solving witnesses in different scenarios.
Internal Module
The internal
module contains items for internal use, such as macros for proper expansion. Circuit developers typically do not need to handle these.
Extra Module
The extra
module includes additional items:
UnconstrainedAPI
: Provides operations with semantics consistent with Circom's operators. These operations do not generate constraints and are only called during the witness solving stage. Circuit developers need to manually constrain the results of these operations.Serde
: Definesserialize_into()
anddeserialize_from()
. These functions can be used to dump compilation results to a file.debug_eval
: This function can be used to debug circuits. For specific usage, please refer to Debugging.