16#include <gtest/gtest.h>
39 static auto [cached_verified, cached_proof_result] = []() {
42 const auto public_inputs_cols = public_inputs.to_columns();
48 const bool verified = verifier.
verify_proof(proof, public_inputs_cols);
53 ASSERT_TRUE(cached_verified) <<
"native proof verification failed";
54 proof_result = cached_proof_result;
73 GTEST_SKIP() <<
"Skipping slow test";
76 const bool pad_proof = GetParam();
83 NativeProofResult proof_result;
85 std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
86 ASSERT_NO_FATAL_FAILURE({ create_and_verify_native_proof(proof_result); });
87 std::chrono::steady_clock::time_point end = std::chrono::steady_clock::now();
88 std::cout <<
"Time taken (native proof): " << std::chrono::duration_cast<std::chrono::seconds>(end - start).count()
91 auto [proof, public_inputs_cols] = proof_result;
102 OuterBuilder outer_circuit;
106 public_inputs_ct.reserve(public_inputs_cols.size());
107 for (
const auto& vec : public_inputs_cols) {
109 vec_ct.reserve(vec.size());
110 for (
const auto& val : vec) {
111 vec_ct.push_back(UltraFF::from_witness(&outer_circuit, val));
113 public_inputs_ct.push_back(vec_ct);
118 auto verifier_output = [&]() {
119 std::cout <<
"Constructing AvmRecursiveVerifier and verifying " << (pad_proof ?
"padded " :
"") <<
"proof..."
121 std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
123 auto result = avm_rec_verifier.
verify_proof(stdlib_proof, public_inputs_ct);
124 std::chrono::steady_clock::time_point end = std::chrono::steady_clock::now();
125 std::cout <<
"Time taken (recursive verification): "
126 << std::chrono::duration_cast<std::chrono::seconds>(end - start).count() <<
"s" <<
std::endl;
131 inputs.pairing_inputs = verifier_output.points_accumulator;
132 inputs.ipa_claim = verifier_output.ipa_claim;
134 outer_circuit.ipa_proof = verifier_output.ipa_proof.get_value();
137 ASSERT_TRUE(verifier_output.points_accumulator.check()) <<
"Pairing points (aggregation state) are not valid.";
138 ASSERT_FALSE(outer_circuit.failed()) <<
"Outer circuit has failed.";
140 vinfo(
"Recursive verifier",
141 (pad_proof ?
" (padded proof)" :
""),
142 ": finalized num gates = ",
143 outer_circuit.num_gates());
150 auto outer_proof = [&]() {
152 UltraRollupProver outer_prover(outer_proving_key, verification_key);
153 return outer_prover.construct_proof();
161 bool result = final_verifier.
verify_proof(outer_proof).result;
170 GTEST_SKIP() <<
"Skipping slow test";
175 const bool pad_proof = GetParam();
177 NativeProofResult proof_result;
179 std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
180 ASSERT_NO_FATAL_FAILURE({ create_and_verify_native_proof(proof_result); });
181 std::chrono::steady_clock::time_point end = std::chrono::steady_clock::now();
182 std::cout <<
"Time taken (native proof): " << std::chrono::duration_cast<std::chrono::seconds>(end - start).count()
185 auto [proof, public_inputs_cols] = proof_result;
200 public_inputs_ct.reserve(public_inputs_cols.size());
201 for (
const auto& vec : public_inputs_cols) {
202 std::vector<FF> vec_ct;
203 vec_ct.reserve(vec.size());
204 for (
const auto& val : vec) {
205 vec_ct.push_back(FF::from_witness(&
builder, val));
207 public_inputs_ct.push_back(vec_ct);
212 FF final_state_full_verification;
214 transcript->enable_manifest();
217 std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
219 [[maybe_unused]]
auto _result = avm_rec_verifier.
verify_proof(stdlib_proof, public_inputs_ct);
220 std::chrono::steady_clock::time_point end = std::chrono::steady_clock::now();
221 std::cout <<
"Time taken (recursive verification): "
222 << std::chrono::duration_cast<std::chrono::seconds>(end - start).count() <<
"s" <<
std::endl;
227 FF final_state_transcript_operations_only;
230 std::tie(final_state_transcript_operations_only, mocked_transcript) =
235 EXPECT_EQ(final_state_full_verification.get_value(), final_state_transcript_operations_only.get_value());
238 auto manifest = transcript->get_manifest();
239 auto mocked_manifest = mocked_transcript->get_manifest();
243 for (
size_t round = 0; round < manifest.size(); ++round) {
244 ASSERT_EQ(manifest[round], mocked_manifest[round])
245 << std::format(
"Real/Mocked manifest discrepancy in round {}", round);
249 final_state_transcript_operations_only.assert_equal(final_state_full_verification);
251 EXPECT_TRUE(!
builder.failed());
256 ::testing::Values(
false,
true),
257 [](
const auto&
info) {
return info.param ?
"Padded" :
"Unpadded"; });
263 GTEST_SKIP() <<
"Skipping slow test";
270 NativeProofResult proof_result;
272 std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
273 ASSERT_NO_FATAL_FAILURE({ create_and_verify_native_proof(proof_result); });
274 std::chrono::steady_clock::time_point end = std::chrono::steady_clock::now();
275 std::cout <<
"Time taken (native proof): " << std::chrono::duration_cast<std::chrono::seconds>(end - start).count()
278 auto [proof, public_inputs_cols] = proof_result;
281 OuterBuilder outer_circuit;
285 public_inputs_ct.reserve(public_inputs_cols.size());
286 for (
const auto& vec : public_inputs_cols) {
288 vec_ct.reserve(vec.size());
289 for (
const auto& val : vec) {
290 vec_ct.push_back(UltraFF::from_witness(&outer_circuit, val));
292 public_inputs_ct.push_back(vec_ct);
295 public_inputs_ct[1][5] += 1;
301 std::chrono::steady_clock::time_point start = std::chrono::steady_clock::now();
303 auto result = avm_rec_verifier.
verify_proof(stdlib_proof, public_inputs_ct);
304 std::chrono::steady_clock::time_point end = std::chrono::steady_clock::now();
305 std::cout <<
"Time taken (recursive verification): "
306 << std::chrono::duration_cast<std::chrono::seconds>(end - start).count() <<
"s" <<
std::endl;
309 ASSERT_TRUE(outer_circuit.failed()) <<
"Outer circuit SHOULD fail with bad PIs.";
#define BB_ASSERT_GT(left, right,...)
#define BB_ASSERT_EQ(actual, expected,...)
#define AVM_V2_PROOF_LENGTH_IN_FIELDS_PADDED
static bool check(const Builder &circuit)
Check the witness satisifies the circuit.
UltraCircuitBuilder CircuitBuilder
typename Curve::ScalarField FF
Output verify_proof(const Proof &proof)
Perform ultra verification.
Proof prove(tracegen::TraceContainer &&trace)
static std::pair< stdlib::field_t< Builder >, std::shared_ptr< TemplatedTranscript< Builder > > > hash_avm_transcript_for_testing(Builder &builder, const stdlib::Proof< Builder > &stdlib_proof, const std::vector< std::vector< stdlib::field_t< Builder > > > &public_inputs)
Testing method to hash the transcript after having replicated the operations performed on the AVM tra...
MegaCircuitBuilder CircuitBuilder
FF hash_avm_transcript(const StdlibProof &stdlib_proof)
Hash the transcript after verification is complete to produce a hash of the public inputs and proofs ...
PairingPoints verify_proof(const StdlibProof &stdlib_proof, const std::vector< std::vector< typename Flavor::FF > > &public_inputs)
Verify an AVM proof and return PairingPoints whose validity bears witness to successful verification ...
bool verify_proof(const HonkProof &proof, const std::vector< std::vector< FF > > &public_inputs)
Verify an AVM proof.
Recursive verifier of AVM2 proofs that utilizes the Goblin mechanism for efficient EC operations.
TwoLayerAvmRecursiveVerifierOutput verify_proof(const stdlib::Proof< UltraCircuitBuilder > &stdlib_proof, const std::vector< std::vector< UltraFF > > &public_inputs) const
Recursively verify an AVM proof using Goblin and two layers of recursive verification.
typename RecursiveFlavor::CircuitBuilder OuterBuilder
static void SetUpTestSuite()
static void create_and_verify_native_proof(NativeProofResult &proof_result)
A simple wrapper around a vector of stdlib field elements representing a proof.
The data that is propagated on the public inputs of a rollup circuit.
TEST_P(AvmRecursiveTestsParameterized, TwoLayerAvmRecursion)
A test of the Two Layer AVM recursive verifier.
INSTANTIATE_TEST_SUITE_P(PaddingVariants, AvmRecursiveTestsParameterized, ::testing::Values(false, true), [](const auto &info) { return info.param ? "Padded" :"Unpadded";})
TEST_F(AvmRecursiveTests, TwoLayerAvmRecursionFailsWithWrongPIs)
bool skip_slow_tests()
Check if slow tests should be skipped.
std::pair< tracegen::TraceContainer, PublicInputs > get_minimal_trace_with_pi()
std::filesystem::path bb_crs_path()
void init_file_crs_factory(const std::filesystem::path &path)
constexpr decltype(auto) get(::tuplet::tuple< T... > &&t) noexcept
std::vector< std::vector< FF > > public_inputs_cols