-
Notifications
You must be signed in to change notification settings - Fork 301
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
feat: MSM sorting #7351
Merged
Merged
feat: MSM sorting #7351
Changes from all commits
Commits
Show all changes
24 commits
Select commit
Hold shift + click to select a range
7a00672
wip
zac-williamson 33f90e2
batched denominators method and WiP batched add
ledwards2225 de4905d
batched affine add test is passing
ledwards2225 d8d5f0d
move logic and tests to new class
ledwards2225 1adaa41
reduce inputs functionality complete and passing tests
ledwards2225 a235afe
memory is owned by the class
ledwards2225 38d59cc
crs comment
ledwards2225 9f79c44
cleanup and remove old versions of methods
ledwards2225 23a7943
Merge branch 'master' into lde/msm_sorting
ledwards2225 584df9d
cleanup
ledwards2225 ced9f79
comments and naming
ledwards2225 2ecea89
test simplification
ledwards2225 1772b64
larger test
ledwards2225 a2967f3
mas cleanup
ledwards2225 196aa76
Merge branch 'master' into lde/msm_sorting
ledwards2225 6fb1e7d
class description
ledwards2225 853ba14
more comments
ledwards2225 3feb70f
Merge branch 'master' into lde/msm_sorting
ledwards2225 1132695
and more comments
ledwards2225 ebc195d
Merge branch 'master' into lde/msm_sorting
ledwards2225 adfcd9b
fix build
ledwards2225 6a2c1d7
one more
ledwards2225 1341c62
comments, multithreading sort, and inlining add method
ledwards2225 dc67d1f
Merge branch 'master' into lde/msm_sorting
ledwards2225 File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
218 changes: 218 additions & 0 deletions
218
barretenberg/cpp/src/barretenberg/ecc/scalar_multiplication/sorted_msm.cpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,218 @@ | ||
#include "barretenberg/ecc/scalar_multiplication/sorted_msm.hpp" | ||
|
||
namespace bb { | ||
|
||
/** | ||
* @brief Reduce MSM inputs such that the set of scalars contains no duplicates by summing points which share a scalar. | ||
* @details Since point addition is substantially cheaper than scalar multiplication, it is more efficient in some cases | ||
* to first sum all points which share a scalar then perform the MSM on the reduced set of inputs. This is achieved via | ||
* the following procedure: | ||
* | ||
* 1) Sort the input {points, scalars} by scalar in order to group points into 'addition sequences' i.e. sets of points | ||
* to be added together prior to performing the MSM. | ||
* | ||
* 2) For each sequence, perform pairwise addition on all points. (If the length of the sequence is odd, the unpaired | ||
* point is simply carried over to the next round). The inverses needed in the addition formula are batch computed in a | ||
* single go for all additions to be performed across all sequences in a given round. | ||
* | ||
* 3) Perform rounds of pair-wise addition until each sequence is reduced to a single point. | ||
* | ||
* @tparam Curve | ||
* @param scalars | ||
* @param points | ||
* @return MsmSorter<Curve>::ReducedMsmInputs | ||
*/ | ||
template <typename Curve> | ||
MsmSorter<Curve>::ReducedMsmInputs MsmSorter<Curve>::reduce_msm_inputs(std::span<Fr> scalars, std::span<G1> points) | ||
{ | ||
// Generate the addition sequences (sets of points sharing a scalar) | ||
AdditionSequences addition_sequences = construct_addition_sequences(scalars, points); | ||
|
||
// Perform rounds of pairwise addition until all sets of points sharing a scalar have been reduced to a single point | ||
batched_affine_add_in_place(addition_sequences); | ||
|
||
// The reduced MSM inputs are the unique scalars and the reduced points | ||
std::span<Fr> output_scalars(unique_scalars.data(), num_unique_scalars); | ||
std::span<G1> output_points(updated_points.data(), num_unique_scalars); | ||
return { output_scalars, output_points }; | ||
} | ||
|
||
/** | ||
* @brief Sort the MSM points by scalar so that points sharing a scalar can be summed prior to performing MSM | ||
* | ||
* @tparam Curve | ||
* @param scalars | ||
* @param points | ||
* @return MsmSorter<Curve>::AdditionSequences | ||
*/ | ||
template <typename Curve> | ||
MsmSorter<Curve>::AdditionSequences MsmSorter<Curve>::construct_addition_sequences(std::span<Fr> scalars, | ||
std::span<G1> points) | ||
{ | ||
// Create the array containing the indices of the scalars and points sorted by scalar value | ||
const size_t num_points = points.size(); | ||
std::iota(index.begin(), index.end(), 0); | ||
#ifdef NO_TBB | ||
std::sort(index.begin(), index.end(), [&](size_t idx_1, size_t idx_2) { return scalars[idx_1] < scalars[idx_2]; }); | ||
#else | ||
std::sort(std::execution::par_unseq, index.begin(), index.end(), [&](size_t idx_1, size_t idx_2) { | ||
return scalars[idx_1] < scalars[idx_2]; | ||
}); | ||
#endif | ||
|
||
// Store the unique scalar values, the input points sorted by scalar value, and the number of occurences of each | ||
// unique scalar (i.e. the size of each addition sequence) | ||
unique_scalars[0] = scalars[index[0]]; | ||
updated_points[0] = points[index[0]]; | ||
size_t seq_idx = 0; | ||
sequence_counts[seq_idx] = 1; | ||
for (size_t i = 1; i < scalars.size(); ++i) { | ||
const Fr& current_scalar = scalars[index[i]]; | ||
const Fr& prev_scalar = scalars[index[i - 1]]; | ||
|
||
// if the current scalar matches the previous, increment the count for this sequence | ||
if (current_scalar == prev_scalar) { | ||
sequence_counts[seq_idx]++; | ||
} else { // otherwise, a new sequence begins | ||
seq_idx++; | ||
sequence_counts[seq_idx]++; | ||
unique_scalars[seq_idx] = current_scalar; | ||
} | ||
|
||
updated_points[i] = points[index[i]]; | ||
} | ||
|
||
num_unique_scalars = seq_idx + 1; | ||
|
||
// Return the sorted points and the counts for each addition sequence | ||
std::span<uint64_t> seq_counts(sequence_counts.data(), num_unique_scalars); | ||
std::span<G1> sorted_points(updated_points.data(), num_points); | ||
return AdditionSequences{ seq_counts, sorted_points, {} }; | ||
} | ||
|
||
/** | ||
* @brief Batch compute inverses needed for a set of point addition sequences | ||
* @details Addition of points P_1, P_2 requires computation of a term of the form 1/(P_2.x - P_1.x). For efficiency, | ||
* these terms are computed all at once for a full set of addition sequences using batch inversion. | ||
* | ||
* @tparam Curve | ||
* @param add_sequences | ||
*/ | ||
template <typename Curve> | ||
void MsmSorter<Curve>::batch_compute_point_addition_slope_inverses(AdditionSequences& add_sequences) | ||
{ | ||
auto points = add_sequences.points; | ||
auto sequence_counts = add_sequences.sequence_counts; | ||
|
||
// Count the total number of point pairs to be added across all addition sequences | ||
size_t total_num_pairs{ 0 }; | ||
for (auto& count : sequence_counts) { | ||
total_num_pairs += count >> 1; | ||
} | ||
|
||
// Define scratch space for batched inverse computations and eventual storage of denominators | ||
std::span<Fq> scratch_space(denominators.data(), total_num_pairs); | ||
std::vector<Fq> differences; | ||
differences.resize(total_num_pairs); | ||
|
||
// Compute and store successive products of differences (x_2 - x_1) | ||
Fq accumulator = 1; | ||
size_t point_idx = 0; | ||
size_t pair_idx = 0; | ||
for (auto& count : sequence_counts) { | ||
const auto num_pairs = count >> 1; | ||
for (size_t j = 0; j < num_pairs; ++j) { | ||
const auto& x1 = points[point_idx++].x; | ||
const auto& x2 = points[point_idx++].x; | ||
|
||
// It is assumed that the input points are random and thus w/h/p do not share an x-coordinate | ||
ASSERT(x1 != x2); | ||
|
||
auto diff = x2 - x1; | ||
differences[pair_idx] = diff; | ||
|
||
// Store and update the running product of differences at each stage | ||
scratch_space[pair_idx++] = accumulator; | ||
accumulator *= diff; | ||
} | ||
// If number of points in the sequence is odd, we skip the last one since it has no pair | ||
point_idx += (count & 0x01ULL); | ||
} | ||
|
||
// Invert the full product of differences | ||
Fq inverse = accumulator.invert(); | ||
|
||
// Compute the individual point-pair addition denominators 1/(x2 - x1) | ||
for (size_t i = 0; i < total_num_pairs; ++i) { | ||
size_t idx = total_num_pairs - 1 - i; | ||
scratch_space[idx] *= inverse; | ||
inverse *= differences[idx]; | ||
} | ||
} | ||
|
||
/** | ||
* @brief In-place summation to reduce a set of addition sequences to a single point for each sequence | ||
* @details At each round, the set of points in each addition sequence is roughly halved by performing pairwise | ||
* additions. For sequences with odd length, the unpaired point is simply carried over to the next round. For | ||
* efficiency, the inverses needed in the point addition slope \lambda are batch computed for the full set of pairwise | ||
* additions in each round. The method is called recursively until the sequences have all been reduced to a single | ||
* point. | ||
* | ||
* @tparam Curve | ||
* @param addition_sequences Set of points and counts indicating number of points in each addition chain | ||
*/ | ||
template <typename Curve> void MsmSorter<Curve>::batched_affine_add_in_place(AdditionSequences addition_sequences) | ||
{ | ||
const size_t num_points = addition_sequences.points.size(); | ||
if (num_points == 0 || num_points == 1) { // nothing to do | ||
return; | ||
} | ||
|
||
// Batch compute terms of the form 1/(x2 -x1) for each pair to be added in this round | ||
batch_compute_point_addition_slope_inverses(addition_sequences); | ||
|
||
auto points = addition_sequences.points; | ||
auto sequence_counts = addition_sequences.sequence_counts; | ||
|
||
// Compute pairwise in-place additions for all sequences with more than 1 point | ||
size_t point_idx = 0; // index for points to be summed | ||
size_t result_point_idx = 0; // index for result points | ||
size_t pair_idx = 0; // index into array of denominators for each pair | ||
bool more_additions = false; | ||
for (auto& count : sequence_counts) { | ||
const auto num_pairs = count >> 1; | ||
const bool overflow = static_cast<bool>(count & 0x01ULL); | ||
// Compute the sum of all pairs in the sequence and store the result in the same points array | ||
for (size_t j = 0; j < num_pairs; ++j) { | ||
const auto& point_1 = points[point_idx++]; // first summand | ||
const auto& point_2 = points[point_idx++]; // second summand | ||
const auto& denominator = denominators[pair_idx++]; // denominator needed in add formula | ||
auto& result = points[result_point_idx++]; // target for addition result | ||
|
||
result = affine_add_with_denominator(point_1, point_2, denominator); | ||
} | ||
// If the sequence had an odd number of points, simply carry the unpaired point over to the next round | ||
if (overflow) { | ||
points[result_point_idx++] = points[point_idx++]; | ||
} | ||
|
||
// Update the sequence counts in place for the next round | ||
const uint64_t updated_sequence_count = static_cast<uint64_t>(num_pairs) + static_cast<uint64_t>(overflow); | ||
count = updated_sequence_count; | ||
|
||
// More additions are required if any sequence has not yet been reduced to a single point | ||
more_additions = more_additions || updated_sequence_count > 1; | ||
} | ||
|
||
// Recursively perform pairwise additions until all sequences have been reduced to a single point | ||
if (more_additions) { | ||
const size_t updated_point_count = result_point_idx; | ||
std::span<G1> updated_points(&points[0], updated_point_count); | ||
return batched_affine_add_in_place( | ||
AdditionSequences{ sequence_counts, updated_points, addition_sequences.scratch_space }); | ||
} | ||
} | ||
|
||
template class MsmSorter<curve::Grumpkin>; | ||
template class MsmSorter<curve::BN254>; | ||
} // namespace bb |
91 changes: 91 additions & 0 deletions
91
barretenberg/cpp/src/barretenberg/ecc/scalar_multiplication/sorted_msm.hpp
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,91 @@ | ||
#pragma once | ||
|
||
#include "./runtime_states.hpp" | ||
#include "barretenberg/ecc/curves/bn254/bn254.hpp" | ||
#include "barretenberg/ecc/curves/grumpkin/grumpkin.hpp" | ||
#include <cstddef> | ||
#include <cstdint> | ||
|
||
namespace bb { | ||
|
||
/** | ||
* @brief Reduce MSM inputs such that the set of scalars contains no duplicates by summing points which share a scalar. | ||
* | ||
* @warning This class is intended to reduce MSMs with EC points that are fully random, e.g. those from an SRS. It does | ||
* not necessarily handle the case where two adjacent points are equal or the inverse of one another (i.e. where x_i == | ||
* x_{i+1}) | ||
* | ||
* @tparam Curve | ||
*/ | ||
template <typename Curve> class MsmSorter { | ||
|
||
public: | ||
using G1 = typename Curve::AffineElement; | ||
using Fr = typename Curve::ScalarField; | ||
using Fq = typename Curve::BaseField; | ||
|
||
// Storage for a set of points to be sorted and reduced | ||
struct AdditionSequences { | ||
std::span<uint64_t> sequence_counts; | ||
std::span<G1> points; | ||
std::optional<std::span<Fq>> scratch_space; | ||
}; | ||
|
||
// Set of reduced MSM inputs where all scalars are unique | ||
struct ReducedMsmInputs { | ||
std::span<Fr> scalars; | ||
std::span<G1> points; | ||
}; | ||
|
||
size_t num_unique_scalars = 0; | ||
std::vector<uint64_t> sequence_counts; | ||
std::vector<Fr> unique_scalars; | ||
std::vector<G1> updated_points; | ||
std::vector<size_t> index; | ||
std::vector<Fq> denominators; | ||
|
||
MsmSorter(const size_t num_scalars = 0) | ||
{ | ||
sequence_counts.resize(num_scalars); | ||
unique_scalars.resize(num_scalars); | ||
updated_points.resize(num_scalars); | ||
index.resize(num_scalars); | ||
denominators.resize(num_scalars); | ||
} | ||
|
||
ReducedMsmInputs reduce_msm_inputs(std::span<Fr> scalars, std::span<G1> points); | ||
|
||
void batch_compute_point_addition_slope_inverses(AdditionSequences& add_sequences); | ||
|
||
void batched_affine_add_in_place(AdditionSequences addition_sequences); | ||
|
||
AdditionSequences construct_addition_sequences(std::span<Fr> scalars, std::span<G1> points); | ||
|
||
/** | ||
* @brief Add two affine elements with the inverse in the slope term \lambda provided as input | ||
* @details The sum of two points (x1, y1), (x2, y2) is given by x3 = \lambda^2 - x1 - x2, y3 = \lambda*(x1 - x3) - | ||
* y1, where \lambda = (y2 - y1)/(x2 - x1). When performing many additions at once, it is more efficient to batch | ||
* compute the inverse component of \lambda for each pair of points. This gives rise to the need for a method like | ||
* this one. | ||
* | ||
* @tparam Curve | ||
* @param point_1 (x1, y1) | ||
* @param point_2 (x2, y2) | ||
* @param denominator 1/(x2 - x1) | ||
* @return Curve::AffineElement | ||
*/ | ||
inline G1 affine_add_with_denominator(const G1& point_1, const G1& point_2, const Fq& denominator) | ||
{ | ||
const auto& x1 = point_1.x; | ||
const auto& y1 = point_1.y; | ||
const auto& x2 = point_2.x; | ||
const auto& y2 = point_2.y; | ||
|
||
const Fq lambda = denominator * (y2 - y1); | ||
Fq x3 = lambda.sqr() - x2 - x1; | ||
Fq y3 = lambda * (x1 - x3) - y1; | ||
return { x3, y3 }; | ||
} | ||
}; | ||
|
||
} // namespace bb |
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Probably would be good to parallelise both for larger ranges. For sort you can do similar to
aztec-packages/barretenberg/cpp/src/barretenberg/honk/proof_system/permutation_library.hpp
Lines 188 to 192 in 59b92ca
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good call thanks. BTW is there a typo in the code you linked? Seems like in
else
it should beextra_denominator_uint.end()
instead ofextra_denominator.end()
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@Rumata888 I got the following error when running
./bootstrap.sh
in cpp ( on v0.46.1 ).There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is it the same error on latest master?