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: Planned, auditors: [], 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>>(builder, constraint);
50 } else if (constraint.proof_type == HONK) {
51 honk_recursion_constraint =
52 create_honk_recursion_constraints<UltraRecursiveFlavor_<MegaCircuitBuilder>>(builder, constraint);
53 } else if (constraint.proof_type == ROLLUP_HONK || constraint.proof_type == ROOT_ROLLUP_HONK) {
54 bb::assert_failure("Rollup Honk proof type not supported on MegaBuilder");
55 } else {
56 bb::assert_failure("Invalid Honk proof type");
57 }
58
59 output.update(honk_recursion_constraint, /*update_ipa_data=*/false); // Update output
60 gate_counter.track_diff(gates_per_opcode, opcode_idx); // Track gate count
61 }
62
63 if (has_hn_recursion_constraints) {
64 process_hn_recursion_constraints(builder, gate_counter, gates_per_opcode, hn_recursion_data, ivc_base);
65 }
66
67 return output;
68}
69
70template <>
74 std::vector<size_t>& gates_per_opcode,
75 [[maybe_unused]] const std::shared_ptr<IVCBase>& ivc_base,
76 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& honk_recursion_data,
77 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& avm_recursion_data,
78 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& hn_recursion_data,
79 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& chonk_recursion_data)
80{
81 bool has_honk_recursion_constraints = !honk_recursion_data.first.empty();
82 bool has_avm_recursion_constraints = !avm_recursion_data.first.empty();
83 bool has_hn_recursion_constraints = !hn_recursion_data.first.empty();
84 bool has_chonk_recursion_constraints = !chonk_recursion_data.first.empty();
85
86 // Schema invariants: validate constraint type combinations for UltraBuilder
88 !has_hn_recursion_constraints,
89 "create_recursion_constraints: invalid circuit - HN recursion constraints not supported with UltraBuilder");
90 BB_ASSERT(!(has_chonk_recursion_constraints && has_honk_recursion_constraints),
91 "create_recursion_constraints: invalid circuit - both honk and chonk recursion constraints present");
92 if (has_chonk_recursion_constraints && has_avm_recursion_constraints) {
93 vinfo("WARNING: both chonk and avm recursion constraints are present. While we support this combination, we "
94 "expect to see it only in a mock circuit.");
95 }
96
98
99 for (const auto& [constraint, opcode_idx] : zip_view(honk_recursion_data.first, honk_recursion_data.second)) {
101
102 if (constraint.proof_type == HONK_ZK) {
103 honk_recursion_constraint =
104 create_honk_recursion_constraints<UltraZKRecursiveFlavor_<UltraCircuitBuilder>>(builder, constraint);
105 } else if (constraint.proof_type == HONK) {
106 honk_recursion_constraint =
107 create_honk_recursion_constraints<UltraRecursiveFlavor_<UltraCircuitBuilder>>(builder, constraint);
108 } else if (constraint.proof_type == ROLLUP_HONK || constraint.proof_type == ROOT_ROLLUP_HONK) {
109 honk_recursion_constraint =
110 create_honk_recursion_constraints<UltraRollupRecursiveFlavor_<UltraCircuitBuilder>>(builder,
111 constraint);
112 } else {
113 bb::assert_failure("Invalid Honk proof type");
114 }
115
116 // Update output
117 output.update(honk_recursion_constraint,
118 /*update_ipa_data=*/constraint.proof_type == ROLLUP_HONK ||
119 constraint.proof_type == ROOT_ROLLUP_HONK);
120 output.is_root_rollup = constraint.proof_type == ROOT_ROLLUP_HONK;
121
122 gate_counter.track_diff(gates_per_opcode, opcode_idx);
123 }
124 BB_ASSERT(!(output.is_root_rollup && output.nested_ipa_claims.size() != 2),
125 "Root rollup must accumulate two IPA proofs.");
126
127 for (const auto& [constraint, opcode_idx] : zip_view(chonk_recursion_data.first, chonk_recursion_data.second)) {
130
131 // Update the output
132 output.update(honk_output, /*update_ipa_data=*/true);
133
134 gate_counter.track_diff(gates_per_opcode, opcode_idx);
135 }
136
137 for (const auto& [constraint, opcode_idx] : zip_view(avm_recursion_data.first, avm_recursion_data.second)) {
140
141 // Update the output
142 output.update(honk_output, /*update_ipa_data=*/true);
143
144 gate_counter.track_diff(gates_per_opcode, opcode_idx);
145 }
146
147 return output;
148}
149
153 std::vector<size_t>& gates_per_opcode,
154 const std::pair<std::vector<RecursionConstraint>, std::vector<size_t>>& hn_recursion_data,
155 const std::shared_ptr<IVCBase>& ivc_base)
156{
157 using StdlibVerificationKey = Chonk::RecursiveVerificationKey;
158 using StdlibVKAndHash = Chonk::RecursiveVKAndHash;
159 using StdlibFF = Chonk::RecursiveFlavor::FF;
160
161 // Validate hn_recursion_data constraints/indices size match
162 BB_ASSERT_EQ(hn_recursion_data.first.size(),
163 hn_recursion_data.second.size(),
164 "process_hn_recursion_constraints: hn_recursion_data constraints/indices size mismatch");
165
166 // Lambda template to handle both Chonk and Chonk with the same code
167 auto process_with_ivc = [&]<typename IVCType>(const std::shared_ptr<IVCType>& ivc) {
168 // We expect the length of the internal verification queue to match the number of ivc recursion constraints
169 BB_ASSERT_EQ(hn_recursion_data.first.size(),
170 ivc->verification_queue.size(),
171 "process_hn_recursion_constraints: mismatch in number of recursive verifications during kernel "
172 "creation!");
173
174 // If no witness is provided, populate the VK and public inputs in the recursion constraint with dummy values so
175 // that the present kernel circuit is constructed correctly. (Used for constructing VKs without witnesses).
176 if (builder.is_write_vk_mode()) {
177 // Create stdlib representations of each {proof, vkey} pair to be recursively verified
178 for (auto [constraint, queue_entry] : zip_view(hn_recursion_data.first, ivc->verification_queue)) {
179 auto key_fields = fields_from_witnesses(builder, constraint.key);
180 populate_fields(builder, key_fields, queue_entry.honk_vk->to_field_elements());
181 builder.set_variable(constraint.key_hash, queue_entry.honk_vk->hash());
182 }
183 }
184
185 // Construct a stdlib verification key for each constraint based on the verification key witness indices
186 // therein
188 stdlib_vk_and_hashs.reserve(hn_recursion_data.first.size());
189 for (const auto& constraint : hn_recursion_data.first) {
190 stdlib_vk_and_hashs.push_back(std::make_shared<StdlibVKAndHash>(
192 StdlibVerificationKey::from_witness_indices(builder, constraint.key)),
193 StdlibFF::from_witness_index(&builder, constraint.key_hash)));
194 }
195 // Create stdlib representations of each {proof, vkey} pair to be recursively verified
196 ivc->instantiate_stdlib_verification_queue(builder, stdlib_vk_and_hashs);
197
198 // Verify stdlib queue size matches after instantiation (invariant check)
199 BB_ASSERT_EQ(ivc->stdlib_verification_queue.size(),
200 hn_recursion_data.first.size(),
201 "process_hn_recursion_constraints: stdlib_verification_queue size mismatch after instantiation");
202
203 // Validate constraints against stdlib verification queue entries
204 for (auto [constraint, queue_entry] : zip_view(hn_recursion_data.first, ivc->stdlib_verification_queue)) {
205 // Validate ACIR constraint proof_type matches IVC queue type
206 BB_ASSERT(proof_type_to_chonk_queue_type(constraint.proof_type) == queue_entry.type,
207 "process_hn_recursion_constraints: ACIR constraint proof_type does not match IVC queue type");
208
209 // HN recursion constraints from Noir always have empty public_inputs - the public inputs are handled
210 // entirely by the IVC (KernelIO/AppIO). If this changes in the future, we need to implement binding
211 // between ACIR public inputs and proof public inputs.
212 BB_ASSERT(constraint.public_inputs.empty(),
213 "process_hn_recursion_constraints: unexpected non-empty public_inputs in HN constraint - "
214 "Noir HN constraints should have empty public_inputs (public inputs are handled by IVC IO)");
215
216 // Validate public input layout: IO region size must match VK's num_public_inputs
217 size_t expected_io_size =
218 queue_entry.is_kernel ? IVCType::KernelIO::PUBLIC_INPUTS_SIZE : IVCType::AppIO::PUBLIC_INPUTS_SIZE;
219 size_t vk_num_public_inputs =
220 static_cast<size_t>(uint64_t(queue_entry.honk_vk_and_hash->vk->num_public_inputs.get_value()));
221 BB_ASSERT_EQ(expected_io_size,
222 vk_num_public_inputs,
223 "process_hn_recursion_constraints: IO size mismatch with VK num_public_inputs");
224
225 // Sanity check: proof vector should have at least num_public_inputs elements
226 // (HN proofs store public inputs at the start of the proof vector)
227 BB_ASSERT_GTE(queue_entry.proof.size(),
228 vk_num_public_inputs,
229 "process_hn_recursion_constraints: proof vector smaller than num_public_inputs - malformed "
230 "proof");
231 }
232
233 // Complete the kernel circuit with all required recursive verifications, databus consistency checks etc.
234 ivc->complete_kernel_circuit_logic(builder);
235
236 // Note: we can't easily track the gate contribution from each individual hn_recursion_constraint since they
237 // are handled simultaneously in the above function call; instead we track the total contribution
238 gate_counter.track_diff(gates_per_opcode, hn_recursion_data.second.at(0));
239 };
240
241 // If an ivc instance is not provided, we mock one with the state required to construct the recursion
242 // constraints present in the program. This is for when we write_vk.
243 if (ivc_base == nullptr) {
244 auto mock_ivc = create_mock_chonk_from_constraints(hn_recursion_data.first);
245 process_with_ivc(mock_ivc);
246 } else {
247 auto chonk = std::dynamic_pointer_cast<Chonk>(ivc_base);
248 if (!chonk) {
249 throw_or_abort("process_hn_recursion_constraints: ivc_base is not a Chonk instance");
250 }
251 process_with_ivc(chonk);
252 }
253}
254
255} // 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
#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)