Debugging
Currently, ExpanderCompilerCollection also supports debugging circuits. To use this feature, you need to make a few adjustments to your circuit definition:
At the beginning, in addition to frontend
, you also need to include the contents of the frontend::extra
module:
use expander_compiler::frontend::*;
use extra::*;
If you are using Sub-Circuit functions, you need to ensure their parameters are RootAPI
, rather than the old API
.
fn sub_circuit_fn<C: Config, B: RootAPI<C>>(api: &mut B, x: &Vec<Variable>) -> Vec<Variable>
Similarly, you need to ensure the circuit itself is defined using GenericDefine
:
impl GenericDefine<GF2Config> for YourCircuit<Variable> {
fn define<Builder: RootAPI<GF2Config>>(&self, api: &mut Builder) {
// definition of your circuit
}
}
Finally, for compilation (similarly, compile_generic
):
let compile_result = compile_generic(&YourCircuit::default()).unwrap();
When you need to debug, modify your main function to include the following code:
let mut assignment = YourCircuit::<GF2>::default();
// assign values in the assignment
debug_eval(&YourCircuit::default(), &assignment, EmptyHintCaller::new());
We also have an API that is only available in debug mode:
fn display(&self, _label: &str, _x: impl ToVariableOrValue<C::CircuitField>) {}
Within debug_eval
, you can use this API to display the value of a variable and print it for debugging purposes. These calls will be ignored during normal compilation.
For a complete example, please refer to keccak_gf2.rs.