Skip to content

Commit

Permalink
Introduce polynomial_sum for fast batched addition
Browse files Browse the repository at this point in the history
  • Loading branch information
x-mass committed May 20, 2024
1 parent 858646c commit 281c9e7
Show file tree
Hide file tree
Showing 5 changed files with 122 additions and 11 deletions.
16 changes: 8 additions & 8 deletions include/nil/crypto3/math/polynomial/polynomial.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,7 @@ namespace nil {

polynomial(const polynomial& x) : val(x.val) {
}

polynomial(const polynomial& x, const allocator_type& a) : val(x.val, a) {
}

Expand All @@ -117,13 +118,8 @@ namespace nil {
this->operator[](power) = value;
}

explicit polynomial(const container_type &c) : val(c) {
if (val.empty()) {
val.push_back(FieldValueType::zero());
}
}

explicit polynomial(container_type &&c) : val(c) {
template<typename T>
explicit polynomial(T&& c) : val(std::forward<T>(c)) {
if (val.empty()) {
val.push_back(FieldValueType::zero());
}
Expand Down Expand Up @@ -179,6 +175,10 @@ namespace nil {
return this->val.__alloc();
}

container_type& get_storage() {
return val;
}

iterator begin() BOOST_NOEXCEPT {
return val.begin();
}
Expand Down Expand Up @@ -383,7 +383,7 @@ namespace nil {
* Returns true if polynomial is a one polynomial.
*/
bool is_one() const {
return (*this->begin() == FieldValueType(1)) &&
return (*this->begin() == FieldValueType(1)) &&
std::all_of(++this->begin(), this->end(),
[](FieldValueType i) { return i == FieldValueType::zero(); });
}
Expand Down
38 changes: 36 additions & 2 deletions include/nil/crypto3/math/polynomial/polynomial_dfs.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -35,9 +35,10 @@
#include <iterator>
#include <unordered_map>

#include <nil/crypto3/math/polynomial/basic_operations.hpp>
#include <nil/crypto3/math/domains/evaluation_domain.hpp>
#include <nil/crypto3/math/algorithms/make_evaluation_domain.hpp>
#include <nil/crypto3/math/domains/evaluation_domain.hpp>
#include <nil/crypto3/math/polynomial/basic_operations.hpp>
#include <nil/crypto3/math/polynomial/polynomial.hpp>

namespace nil {
namespace crypto3 {
Expand Down Expand Up @@ -164,6 +165,10 @@ namespace nil {
return this->val.__alloc();
}

container_type& get_storage() {
return val;
}

iterator begin() BOOST_NOEXCEPT {
return val.begin();
}
Expand Down Expand Up @@ -761,6 +766,35 @@ namespace nil {
return os;
}

template<typename FieldType>
static inline polynomial_dfs<typename FieldType::value_type> polynomial_sum(
std::vector<math::polynomial_dfs<typename FieldType::value_type>> addends) {
using FieldValueType = typename FieldType::value_type;
std::size_t max_size = 0;
std::unordered_map<std::size_t, polynomial_dfs<FieldValueType>> size_to_part_sum;
for (auto& addend : addends) {
max_size = std::max(max_size, addend.size());
auto it = size_to_part_sum.find(addend.size());
if (it == size_to_part_sum.end()) {
size_to_part_sum[addend.size()] = std::move(addend);
} else {
it->second += addend;
// Free the memory we are not going to use anymore.
addend = math::polynomial_dfs<FieldValueType>();
}
}

auto coef_result = polynomial<FieldValueType>(max_size, FieldValueType::zero());
for (auto& [_, partial_sum] : size_to_part_sum) {
coef_result += polynomial<FieldValueType>(std::move(partial_sum.coefficients()));
}

polynomial_dfs<FieldValueType> dfs_result;
dfs_result.from_coefficients(coef_result.get_storage());

return dfs_result;
}

template<typename FieldType>
static inline polynomial_dfs<typename FieldType::value_type> polynomial_product(
std::vector<math::polynomial_dfs<typename FieldType::value_type>> multipliers) {
Expand Down
5 changes: 4 additions & 1 deletion test/benchmarks/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,10 @@
# http://www.boost.org/LICENSE_1_0.txt
#---------------------------------------------------------------------------#

find_package(Boost REQUIRED COMPONENTS unit_test_framework timer)
find_package(Boost REQUIRED COMPONENTS
timer
unit_test_framework
)

cm_test_link_libraries(${CMAKE_WORKSPACE_NAME}_${CURRENT_PROJECT_NAME}

Expand Down
38 changes: 38 additions & 0 deletions test/benchmarks/polynomial_dfs_benchmark.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -174,4 +174,42 @@ BENCHMARK_AUTO_TEST_CASE(polynomial_product_test, 20) {
STOP_TIMER("polynomial_product")
}

BENCHMARK_AUTO_TEST_CASE(polynomial_sum_real_test, 20) {
using Field = nil::crypto3::algebra::fields::bls12_fr<381>;

std::vector<polynomial_dfs<typename Field::value_type>> random_polynomials;
random_polynomials.reserve(8);
Field::value_type a = alg_rnd_engine();
std::vector<std::size_t> sizes = {23, 15, 21, 16, 22, 17, 18};
for (auto size : sizes) {
random_polynomials.emplace_back(
generate_random_polynomial<Field>(
1u << size,
alg_rnd_engine
)
);
}
auto random_polynomials_copy = random_polynomials;

START_TIMER("polynomial_naive")
std::size_t max_size = 0;
for (const auto& polynomial : random_polynomials_copy) {
max_size = std::max(max_size, polynomial.size());
}
auto max_domain = make_evaluation_domain<FieldType>(max_size);
for (auto& polynomial : random_polynomials_copy) {
polynomial.resize(max_size, nullptr, max_domain);
}
polynomial_dfs<typename Field::value_type> naive_res(0, max_size);
for (auto& polynomial : random_polynomials_copy) {
naive_res += std::move(polynomial);
}
STOP_TIMER("polynomial_naive")

START_TIMER("polynomial_sum")
const auto res = polynomial_sum<Field>(std::move(random_polynomials));
STOP_TIMER("polynomial_sum")
BOOST_CHECK_EQUAL(naive_res, res);
}

BOOST_AUTO_TEST_SUITE_END()
36 changes: 36 additions & 0 deletions test/polynomial_dfs.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -234,6 +234,42 @@ BOOST_AUTO_TEST_CASE(polynomial_dfs_addition_less_a) {
0x5a8fa2a6cb26338cef1cd76f6106a8baa60293c9c1c0d2c31e813ed413279c47_cppui_modular253}};
BOOST_CHECK_EQUAL(c_res, c);
}

BOOST_AUTO_TEST_CASE(polynomial_dfs_sum) {
std::vector<polynomial_dfs<typename FieldType::value_type>> polynomials = {
{
7,
{
0x37_cppui_modular253,
0x6C17ABF513DFFC886A7F49F970801792C825CFDD829870DC60E8DA51F53633_cppui_modular253,
0x73EDA753299D7D3ED0CB3E52336E8625A78AA3D929CB5BFEFFEEFFFEFFFFFFFD_cppui_modular253,
0x53B09574717196328488C7990499B10ABA0C038C321BF5B1C0D1C5A4E10C7330_cppui_modular253,
0x73EDA753299D7D483339D80809A1D80553BDA402FFFE5BFEFFFFFFFEFFFFFFEE_cppui_modular253,
0x73818FA734899D485AE48BE74C1F3B0838E37E245E69C38E23991724AE0AC9C4_cppui_modular253,
0x9626E99B5D63351DFAC330029D63300000010FFFFFFFFFFFC_cppui_modular253,
0x203D11DEB82BE718FE9BDD45C91A43E021C3A08591F4664D3F343A5A1EF38CC7_cppui_modular253
}
},
{
6,
{
0x4e_cppui_modular253,
0x2984a53ad76597710bc6e589547653b47dcce72f84e0617332e6e0761851f4a9_cppui_modular253,
0x73eda753299d7d4078c0a4bd957794986b93a3e08bd45bfefff1fffefffffffb_cppui_modular253,
0x195e04ac5e7749b26f0033b1486ae23bed8b1011de0d893be16ec12aecd863c2_cppui_modular253,
0x73eda753299d7d483339d80809a1d80553bda402fffe5bfefffffffeffffffff_cppui_modular253,
0x4a6902185237e5dffc8fbf66155bd15f9620bcfadb4dfa8bcd291f88e7ae0b60_cppui_modular253,
0x7ba79334a742a436ce82a0022742a0000000dfffffffffffa_cppui_modular253,
0x5a8fa2a6cb26338cef1cd76f6106a8baa60293c9c1c0d2c31e813ed413279c47_cppui_modular253
}
}
};
auto native_sum = polynomial_dfs<typename FieldType::value_type>::zero();
for (const auto& p : polynomials) {
native_sum += p;
}
BOOST_CHECK_EQUAL(native_sum, polynomial_sum<FieldType>(polynomials));
}
BOOST_AUTO_TEST_SUITE_END()

BOOST_AUTO_TEST_SUITE(polynomial_dfs_addition_eq_test_suite)
Expand Down

0 comments on commit 281c9e7

Please sign in to comment.