Skip to content

Commit

Permalink
chore: make first iteration of protogalaxy more efficient (#4630)
Browse files Browse the repository at this point in the history
This work makes the first iteration of protogalaxy more efficient by removing the perturbator computation in just the first iteration. We can do this because in the first iteration, there is no accumulator, so we're just folding instances together. Because of this, we can assume that f_i(\omega) is 0 for all i. This seems suspicious, but this assumption is fine because if the instances were not valid (some f_i(\omega) is not 0), folding invalid instances would yield an invalid accumlator so it wouldn't matter in the end. Normally, we can't do this because f_i(\omega) for a correct accumulator is not all 0s, as \omega is some folded thing.

Fixes #740.
  • Loading branch information
lucasxia01 authored and AztecBot committed Feb 21, 2024
1 parent 71c35f6 commit 7fa36d6
Show file tree
Hide file tree
Showing 4 changed files with 22 additions and 38 deletions.
20 changes: 8 additions & 12 deletions cpp/src/barretenberg/protogalaxy/protogalaxy_prover.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -154,17 +154,9 @@ template <class ProverInstances> void ProtoGalaxyProver_<ProverInstances>::prepa
send_accumulator(instance, domain_separator);
} else {
// This is the first round of folding and we need to generate some gate challenges.
// TODO(https://github.com/AztecProtocol/barretenberg/issues/740): implement option 2 to make this more
// efficient by avoiding the computation of the perturbator
finalise_and_send_instance(instance, domain_separator);
instance->target_sum = 0;
auto beta = transcript->template get_challenge<FF>(domain_separator + "_initial_gate_challenge");
std::vector<FF> gate_challenges(instance->log_instance_size);
gate_challenges[0] = beta;
for (size_t i = 1; i < instance->log_instance_size; i++) {
gate_challenges[i] = gate_challenges[i - 1].sqr();
}
instance->gate_challenges = gate_challenges;
instance->gate_challenges = std::vector<FF>(instance->log_instance_size, 0);
}

idx++;
Expand Down Expand Up @@ -315,9 +307,13 @@ template <class ProverInstances> void ProtoGalaxyProver_<ProverInstances>::pertu
state.accumulator = get_accumulator();
FF delta = transcript->template get_challenge<FF>("delta");
state.deltas = compute_round_challenge_pows(state.accumulator->log_instance_size, delta);
state.perturbator = compute_perturbator(state.accumulator, state.deltas);
for (size_t idx = 0; idx <= state.accumulator->log_instance_size; idx++) {
transcript->send_to_verifier("perturbator_" + std::to_string(idx), state.perturbator[idx]);
state.perturbator = Polynomial<FF>(state.accumulator->log_instance_size + 1); // initialize to all zeros
// compute perturbator only if this is not the first round and has an accumulator
if (state.accumulator->is_accumulator) {
state.perturbator = compute_perturbator(state.accumulator, state.deltas);
for (size_t idx = 0; idx <= state.accumulator->log_instance_size; idx++) {
transcript->send_to_verifier("perturbator_" + std::to_string(idx), state.perturbator[idx]);
}
}
};

Expand Down
2 changes: 0 additions & 2 deletions cpp/src/barretenberg/protogalaxy/protogalaxy_prover.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -134,8 +134,6 @@ template <class ProverInstances_> class ProtoGalaxyProver_ {

// Returns the accumulator, which is the first element in ProverInstances. The accumulator is assumed to have the
// FoldingParameters set and be the result of a previous round of folding.
// TODO(https://github.com/AztecProtocol/barretenberg/issues/740): handle the case when the accumulator is empty
// (i.e. we are in the first round of folding)/
std::shared_ptr<Instance> get_accumulator() { return instances[0]; }

/**
Expand Down
19 changes: 7 additions & 12 deletions cpp/src/barretenberg/protogalaxy/protogalaxy_verifier.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -159,17 +159,9 @@ void ProtoGalaxyVerifier_<VerifierInstances>::prepare_for_folding(const std::vec
receive_accumulator(inst, domain_separator);
} else {
// This is the first round of folding and we need to generate some gate challenges.
// TODO(https://github.com/AztecProtocol/barretenberg/issues/740): implement option 2 to make this more
// efficient by avoiding the computation of the perturbator
receive_and_finalise_instance(inst, domain_separator);
inst->target_sum = 0;
auto beta = transcript->template get_challenge<FF>(domain_separator + "_initial_gate_challenge");
std::vector<FF> gate_challenges(inst->log_instance_size);
gate_challenges[0] = beta;
for (size_t i = 1; i < inst->log_instance_size; i++) {
gate_challenges[i] = gate_challenges[i - 1].sqr();
}
inst->gate_challenges = gate_challenges;
inst->gate_challenges = std::vector<FF>(inst->log_instance_size, 0);
}
index++;

Expand All @@ -189,9 +181,12 @@ bool ProtoGalaxyVerifier_<VerifierInstances>::verify_folding_proof(const std::ve
auto accumulator = get_accumulator();
auto deltas = compute_round_challenge_pows(accumulator->log_instance_size, delta);

std::vector<FF> perturbator_coeffs(accumulator->log_instance_size + 1);
for (size_t idx = 0; idx <= accumulator->log_instance_size; idx++) {
perturbator_coeffs[idx] = transcript->template receive_from_prover<FF>("perturbator_" + std::to_string(idx));
std::vector<FF> perturbator_coeffs(accumulator->log_instance_size + 1, 0);
if (accumulator->is_accumulator) {
for (size_t idx = 0; idx <= accumulator->log_instance_size; idx++) {
perturbator_coeffs[idx] =
transcript->template receive_from_prover<FF>("perturbator_" + std::to_string(idx));
}
}

if (perturbator_coeffs[0] != accumulator->target_sum) {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -164,17 +164,9 @@ template <class VerifierInstances> void ProtoGalaxyRecursiveVerifier_<VerifierIn
receive_accumulator(inst, domain_separator);
} else {
// This is the first round of folding and we need to generate some gate challenges.
// TODO(https://github.com/AztecProtocol/barretenberg/issues/740): implement option 2 to make this more
// efficient by avoiding the computation of the perturbator
receive_and_finalise_instance(inst, domain_separator);
inst->target_sum = 0;
auto beta = transcript->template get_challenge<FF>(domain_separator + "_initial_gate_challenge");
std::vector<FF> gate_challenges(inst->log_instance_size);
gate_challenges[0] = beta;
for (size_t i = 1; i < inst->log_instance_size; i++) {
gate_challenges[i] = gate_challenges[i - 1].sqr();
}
inst->gate_challenges = gate_challenges;
inst->gate_challenges = std::vector<FF>(inst->log_instance_size, 0);
}
index++;

Expand All @@ -199,9 +191,12 @@ void ProtoGalaxyRecursiveVerifier_<VerifierInstances>::verify_folding_proof(cons
auto accumulator = get_accumulator();
auto deltas = compute_round_challenge_pows(accumulator->log_instance_size, delta);

std::vector<FF> perturbator_coeffs(accumulator->log_instance_size + 1);
for (size_t idx = 0; idx <= accumulator->log_instance_size; idx++) {
perturbator_coeffs[idx] = transcript->template receive_from_prover<FF>("perturbator_" + std::to_string(idx));
std::vector<FF> perturbator_coeffs(accumulator->log_instance_size + 1, 0);
if (accumulator->is_accumulator) {
for (size_t idx = 0; idx <= accumulator->log_instance_size; idx++) {
perturbator_coeffs[idx] =
transcript->template receive_from_prover<FF>("perturbator_" + std::to_string(idx));
}
}

// TODO(https://github.com/AztecProtocol/barretenberg/issues/833): As currently the stdlib transcript is not
Expand Down

0 comments on commit 7fa36d6

Please sign in to comment.