Barretenberg
The ZK-SNARK library at the core of Aztec
Loading...
Searching...
No Matches
recursion_constraint.cpp
Go to the documentation of this file.
1// === AUDIT STATUS ===
2// internal: { status: Completed, auditors: [Federico], commit: }
3// external_1: { status: not started, auditors: [], commit: }
4// external_2: { status: not started, auditors: [], commit: }
5// =====================
6
13
14namespace acir_format {
15
16template <>
20 std::vector<size_t>& gates_per_opcode,
21 [[maybe_unused]] const std::shared_ptr<IVCBase>& ivc_base,
22 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& honk_recursion_data,
23 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& avm_recursion_data,
24 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& hn_recursion_data,
25 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& chonk_recursion_data)
26{
27 bool has_honk_recursion_constraints = !honk_recursion_data.first.empty();
28 bool has_avm_recursion_constraints = !avm_recursion_data.first.empty();
29 bool has_hn_recursion_constraints = !hn_recursion_data.first.empty();
30 bool has_chonk_recursion_constraints = !chonk_recursion_data.first.empty();
31
32 // Schema invariants: validate constraint type combinations for MegaBuilder
33 BB_ASSERT(!(has_honk_recursion_constraints && has_hn_recursion_constraints),
34 "create_recursion_constraints: invalid circuit - both honk and HN recursion constraints present");
36 !has_avm_recursion_constraints,
37 "create_recursion_constraints: invalid circuit - AVM recursion constraints not supported with MegaBuilder");
38 BB_ASSERT(!has_chonk_recursion_constraints,
39 "create_recursion_constraints: invalid circuit - Chonk recursion constraints not supported with "
40 "MegaBuilder");
41
43
44 for (const auto& [constraint, opcode_idx] : zip_view(honk_recursion_data.first, honk_recursion_data.second)) {
46
47 if (constraint.proof_type == HONK_ZK) {
48 honk_recursion_constraint =
49 create_honk_recursion_constraints<UltraZKRecursiveFlavor_<MegaCircuitBuilder>,
51 constraint);
52 } else if (constraint.proof_type == HONK) {
53 honk_recursion_constraint =
54 create_honk_recursion_constraints<UltraRecursiveFlavor_<MegaCircuitBuilder>,
56 constraint);
57 } else if (constraint.proof_type == ROLLUP_HONK || constraint.proof_type == ROOT_ROLLUP_HONK) {
58 bb::assert_failure("Rollup Honk proof type not supported on MegaBuilder");
59 } else {
60 bb::assert_failure("Invalid Honk proof type");
61 }
62
63 output.update(honk_recursion_constraint, /*update_ipa_data=*/false); // Update output
64 gate_counter.track_diff(gates_per_opcode, opcode_idx); // Track gate count
65 }
66
67 if (has_hn_recursion_constraints) {
68 process_hn_recursion_constraints(builder, gate_counter, gates_per_opcode, hn_recursion_data, ivc_base);
69 }
70
71 return output;
72}
73
74template <>
78 std::vector<size_t>& gates_per_opcode,
79 [[maybe_unused]] const std::shared_ptr<IVCBase>& ivc_base,
80 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& honk_recursion_data,
81 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& avm_recursion_data,
82 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& hn_recursion_data,
83 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& chonk_recursion_data)
84{
85 bool has_honk_recursion_constraints = !honk_recursion_data.first.empty();
86 bool has_avm_recursion_constraints = !avm_recursion_data.first.empty();
87 bool has_hn_recursion_constraints = !hn_recursion_data.first.empty();
88 bool has_chonk_recursion_constraints = !chonk_recursion_data.first.empty();
89
90 // Schema invariants: validate constraint type combinations for UltraBuilder
92 !has_hn_recursion_constraints,
93 "create_recursion_constraints: invalid circuit - HN recursion constraints not supported with UltraBuilder");
94 BB_ASSERT(!(has_chonk_recursion_constraints && has_honk_recursion_constraints),
95 "create_recursion_constraints: invalid circuit - both honk and chonk recursion constraints present");
96 if (has_chonk_recursion_constraints && has_avm_recursion_constraints) {
97 vinfo("WARNING: both chonk and avm recursion constraints are present. While we support this combination, we "
98 "expect to see it only in a mock circuit.");
99 }
100
102
103 for (const auto& [constraint, opcode_idx] : zip_view(honk_recursion_data.first, honk_recursion_data.second)) {
105
106 if (constraint.proof_type == HONK_ZK) {
107 honk_recursion_constraint =
108 create_honk_recursion_constraints<UltraZKRecursiveFlavor_<UltraCircuitBuilder>,
110 constraint);
111 } else if (constraint.proof_type == HONK) {
112 honk_recursion_constraint =
113 create_honk_recursion_constraints<UltraRecursiveFlavor_<UltraCircuitBuilder>,
115 constraint);
116 } else if (constraint.proof_type == ROLLUP_HONK || constraint.proof_type == ROOT_ROLLUP_HONK) {
117 // Use UltraRecursiveFlavor with RollupIO for rollup proofs (IO determines IPA handling)
118 honk_recursion_constraint =
119 create_honk_recursion_constraints<UltraRecursiveFlavor_<UltraCircuitBuilder>,
121 } else {
122 bb::assert_failure("Invalid Honk proof type");
123 }
124
125 // Update output
126 output.update(honk_recursion_constraint,
127 /*update_ipa_data=*/constraint.proof_type == ROLLUP_HONK ||
128 constraint.proof_type == ROOT_ROLLUP_HONK);
129 output.is_root_rollup = constraint.proof_type == ROOT_ROLLUP_HONK;
130
131 gate_counter.track_diff(gates_per_opcode, opcode_idx);
132 }
133 BB_ASSERT(!(output.is_root_rollup && output.nested_ipa_claims.size() != 2),
134 "Root rollup must accumulate two IPA proofs.");
135
136 for (const auto& [constraint, opcode_idx] : zip_view(chonk_recursion_data.first, chonk_recursion_data.second)) {
139
140 // Update the output
141 output.update(honk_output, /*update_ipa_data=*/true);
142
143 gate_counter.track_diff(gates_per_opcode, opcode_idx);
144 }
145
146 for (const auto& [constraint, opcode_idx] : zip_view(avm_recursion_data.first, avm_recursion_data.second)) {
149
150 // Update the output
151 output.update(honk_output, /*update_ipa_data=*/true);
152
153 gate_counter.track_diff(gates_per_opcode, opcode_idx);
154 }
155
156 return output;
157}
158
162 std::vector<size_t>& gates_per_opcode,
163 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& hn_recursion_data,
164 const std::shared_ptr<IVCBase>& ivc_base)
165{
166 using StdlibVerificationKey = Chonk::RecursiveVerificationKey;
167 using StdlibVKAndHash = Chonk::RecursiveVKAndHash;
168 using StdlibFF = Chonk::RecursiveFlavor::FF;
169
170 // Validate hn_recursion_data constraints/indices size match
171 BB_ASSERT_EQ(hn_recursion_data.first.size(),
172 hn_recursion_data.second.size(),
173 "process_hn_recursion_constraints: hn_recursion_data constraints/indices size mismatch");
174
175 // Lambda template to handle both Chonk and Chonk with the same code
176 auto process_with_ivc = [&]<typename IVCType>(const std::shared_ptr<IVCType>& ivc) {
177 // We expect the length of the internal verification queue to match the number of ivc recursion constraints
178 BB_ASSERT_EQ(hn_recursion_data.first.size(),
179 ivc->verification_queue.size(),
180 "process_hn_recursion_constraints: mismatch in number of recursive verifications during kernel "
181 "creation!");
182
183 // If no witness is provided, populate the VK and public inputs in the recursion constraint with dummy values so
184 // that the present kernel circuit is constructed correctly. (Used for constructing VKs without witnesses).
185 if (builder.is_write_vk_mode()) {
186 // Create stdlib representations of each {proof, vkey} pair to be recursively verified
187 for (auto [constraint, queue_entry] : zip_view(hn_recursion_data.first, ivc->verification_queue)) {
188 auto key_fields = fields_from_witnesses(builder, constraint.key);
189 populate_fields(builder, key_fields, queue_entry.honk_vk->to_field_elements());
190 builder.set_variable(constraint.key_hash, queue_entry.honk_vk->hash());
191 }
192 }
193
194 // Construct a stdlib verification key for each constraint based on the verification key witness indices
195 // therein
197 stdlib_vk_and_hashs.reserve(hn_recursion_data.first.size());
198 for (const auto& constraint : hn_recursion_data.first) {
199 stdlib_vk_and_hashs.push_back(std::make_shared<StdlibVKAndHash>(
201 StdlibVerificationKey::from_witness_indices(builder, constraint.key)),
202 StdlibFF::from_witness_index(&builder, constraint.key_hash)));
203 }
204 // Create stdlib representations of each {proof, vkey} pair to be recursively verified
205 ivc->instantiate_stdlib_verification_queue(builder, stdlib_vk_and_hashs);
206
207 // Verify stdlib queue size matches after instantiation (invariant check)
208 BB_ASSERT_EQ(ivc->stdlib_verification_queue.size(),
209 hn_recursion_data.first.size(),
210 "process_hn_recursion_constraints: stdlib_verification_queue size mismatch after instantiation");
211
212 // Validate constraints against stdlib verification queue entries
213 for (auto [constraint, queue_entry] : zip_view(hn_recursion_data.first, ivc->stdlib_verification_queue)) {
214 // Validate ACIR constraint proof_type matches IVC queue type
215 BB_ASSERT(proof_type_to_chonk_queue_type(constraint.proof_type) == queue_entry.type,
216 "process_hn_recursion_constraints: ACIR constraint proof_type does not match IVC queue type");
217
218 // HN recursion constraints from Noir always have empty public_inputs - the public inputs are handled
219 // entirely by the IVC (KernelIO/AppIO). If this changes in the future, we need to implement binding
220 // between ACIR public inputs and proof public inputs.
221 BB_ASSERT(constraint.public_inputs.empty(),
222 "process_hn_recursion_constraints: unexpected non-empty public_inputs in HN constraint - "
223 "Noir HN constraints should have empty public_inputs (public inputs are handled by IVC IO)");
224
225 // Validate public input layout: IO region size must match VK's num_public_inputs
226 size_t expected_io_size =
227 queue_entry.is_kernel ? IVCType::KernelIO::PUBLIC_INPUTS_SIZE : IVCType::AppIO::PUBLIC_INPUTS_SIZE;
228 size_t vk_num_public_inputs =
229 static_cast<size_t>(uint64_t(queue_entry.honk_vk_and_hash->vk->num_public_inputs.get_value()));
230 BB_ASSERT_EQ(expected_io_size,
231 vk_num_public_inputs,
232 "process_hn_recursion_constraints: IO size mismatch with VK num_public_inputs");
233
234 // Sanity check: proof vector should have at least num_public_inputs elements
235 // (HN proofs store public inputs at the start of the proof vector)
236 BB_ASSERT_GTE(queue_entry.proof.size(),
237 vk_num_public_inputs,
238 "process_hn_recursion_constraints: proof vector smaller than num_public_inputs - malformed "
239 "proof");
240 }
241
242 // Complete the kernel circuit with all required recursive verifications, databus consistency checks etc.
243 ivc->complete_kernel_circuit_logic(builder);
244
245 // Note: we can't easily track the gate contribution from each individual hn_recursion_constraint since they
246 // are handled simultaneously in the above function call; instead we track the total contribution
247 gate_counter.track_diff(gates_per_opcode, hn_recursion_data.second.at(0));
248 };
249
250 // If an ivc instance is not provided, we mock one with the state required to construct the recursion
251 // constraints present in the program. This is for when we write_vk.
252 if (ivc_base == nullptr) {
253 auto mock_ivc = create_mock_chonk_from_constraints(hn_recursion_data.first);
254 process_with_ivc(mock_ivc);
255 } else {
256 auto chonk = std::dynamic_pointer_cast<Chonk>(ivc_base);
257 if (!chonk) {
258 throw_or_abort("process_hn_recursion_constraints: ivc_base is not a Chonk instance");
259 }
260 process_with_ivc(chonk);
261 }
262}
263
264} // namespace acir_format
#define BB_ASSERT(expression,...)
Definition assert.hpp:70
#define BB_ASSERT_GTE(left, right,...)
Definition assert.hpp:128
#define BB_ASSERT_EQ(actual, expected,...)
Definition assert.hpp:83
Utility class for tracking the gate count of acir constraints.
void track_diff(std::vector< size_t > &gates_per_opcode, size_t opcode_index)
RecursiveFlavor::VerificationKey RecursiveVerificationKey
Definition chonk.hpp:62
RecursiveFlavor::VKAndHash RecursiveVKAndHash
Definition chonk.hpp:63
typename Curve::ScalarField FF
Manages the data that is propagated on the public inputs of an application/function circuit.
The data that is propagated on the public inputs of a rollup circuit.
#define vinfo(...)
Definition log.hpp:94
AluTraceBuilder builder
Definition alu.test.cpp:124
HonkRecursionConstraintOutput< bb::UltraCircuitBuilder > create_avm2_recursion_constraints_goblin(bb::UltraCircuitBuilder &builder, const RecursionConstraint &input)
Add constraints associated with recursive verification of an AVM2 proof using Goblin.
void process_hn_recursion_constraints(MegaCircuitBuilder &builder, GateCounter< MegaCircuitBuilder > &gate_counter, std::vector< size_t > &gates_per_opcode, const std::pair< std::vector< RecursionConstraint >, std::vector< size_t > > &hn_recursion_data, const std::shared_ptr< IVCBase > &ivc_base)
Process HyperNova recursion constraints and complete kernel logic.
void populate_fields(Builder &builder, const std::vector< field_t< Builder > > &fields, const std::vector< bb::fr > &values)
========== WRITE_VK UTILITIES ========== ///
Definition utils.cpp:78
std::vector< field_t< Builder > > fields_from_witnesses(Builder &builder, std::span< const uint32_t > witness_indices)
========== ACIR TO BARRETENBERG ========== ///
Definition utils.cpp:16
std::shared_ptr< Chonk > create_mock_chonk_from_constraints(const std::vector< RecursionConstraint > &constraints)
Create a Chonk instance with mocked state corresponding to a set of IVC recursion constraints.
HonkRecursionConstraintsOutput< MegaCircuitBuilder > create_recursion_constraints(MegaCircuitBuilder &builder, GateCounter< MegaCircuitBuilder > &gate_counter, std::vector< size_t > &gates_per_opcode, const std::shared_ptr< IVCBase > &ivc_base, const std::pair< std::vector< RecursionConstraint >, std::vector< size_t > > &honk_recursion_data, const std::pair< std::vector< RecursionConstraint >, std::vector< size_t > > &avm_recursion_data, const std::pair< std::vector< RecursionConstraint >, std::vector< size_t > > &hn_recursion_data, const std::pair< std::vector< RecursionConstraint >, std::vector< size_t > > &chonk_recursion_data)
Chonk::QUEUE_TYPE proof_type_to_chonk_queue_type(uint32_t proof_type)
HonkRecursionConstraintOutput< bb::UltraCircuitBuilder > create_chonk_recursion_constraints(bb::UltraCircuitBuilder &builder, const RecursionConstraint &input)
Add constraints associated with recursive verification of a Chonk proof.
void assert_failure(std::string const &err)
Definition assert.cpp:11
constexpr decltype(auto) get(::tuplet::tuple< T... > &&t) noexcept
Definition tuple.hpp:13
Container for the output of multiple recursive verifications.
std::vector< OpeningClaim< stdlib::grumpkin< Builder > > > nested_ipa_claims
void update(const HonkRecursionConstraintOutput< Builder > &other, bool update_ipa_data)
Update the current output with another recursion constraint output.
Output type for recursive ultra verification.
void throw_or_abort(std::string const &err)