From b469ce781d4da6e9bf21a8e76091a84f9ecb6268 Mon Sep 17 00:00:00 2001 From: Iluvmagick Date: Mon, 13 Nov 2023 18:03:54 +0400 Subject: [PATCH] Added a basic packer test. Fixed a packing bug. --- .../plonk/lookup_table_definition.hpp | 9 +- test/CMakeLists.txt | 4 +- .../plonk/placeholder/lookup_table_packer.cpp | 140 ++++++++++++++++++ 3 files changed, 148 insertions(+), 5 deletions(-) create mode 100644 test/systems/plonk/placeholder/lookup_table_packer.cpp diff --git a/include/nil/crypto3/zk/snark/arithmetization/plonk/lookup_table_definition.hpp b/include/nil/crypto3/zk/snark/arithmetization/plonk/lookup_table_definition.hpp index 1bde80446..0c4283466 100644 --- a/include/nil/crypto3/zk/snark/arithmetization/plonk/lookup_table_definition.hpp +++ b/include/nil/crypto3/zk/snark/arithmetization/plonk/lookup_table_definition.hpp @@ -344,8 +344,9 @@ namespace nil { for (std::size_t column = layout_x + fold * column_amount; column < layout_x + (fold + 1) * column_amount; column++) { - for (std::size_t row = layout_y, table_row = 0; row < layout_y + rows_amount; - row++, table_row++) { + for (std::size_t row = layout_y, table_row = fold * rows_amount; + row < layout_y + rows_amount; + row++, table_row++) { if (table_row < table_rows_amount) { constant_columns[column][row] = @@ -366,7 +367,7 @@ namespace nil { continue; } // Check if the current rows are already included in some selector so that we reuse it - // Because folded table would not have a non-full subtable, this works + // This works because a folded table would not have a non-full subtable std::size_t next_selector; if (selector_ids.find(std::make_pair(subtable.begin, subtable.end)) != selector_ids.end()) { @@ -386,7 +387,7 @@ namespace nil { for(std::size_t fold = 0; fold < fold_count; fold++) { std::vector> option; for(const auto &column_index : subtable.column_indices) { - option.emplace_back( plonk_variable( + option.emplace_back(plonk_variable( constant_columns_ids[layout_x + fold * column_amount + column_index], 0, false, plonk_variable::column_type::constant )); diff --git a/test/CMakeLists.txt b/test/CMakeLists.txt index 6bfecd49a..508444a59 100644 --- a/test/CMakeLists.txt +++ b/test/CMakeLists.txt @@ -84,7 +84,7 @@ set(TESTS_NAMES "systems/plonk/pickles/oracles" "systems/plonk/pickles/to_field" "systems/plonk/pickles/to_group" - + "systems/plonk/placeholder/placeholder" "systems/plonk/placeholder/performance" @@ -105,6 +105,8 @@ set(TESTS_NAMES "transcript/transcript" "transcript/kimchi_transcript" + "systems/plonk/placeholder/lookup_table_packer" + "systems/plonk/plonk_constraint") foreach(TEST_NAME ${TESTS_NAMES}) diff --git a/test/systems/plonk/placeholder/lookup_table_packer.cpp b/test/systems/plonk/placeholder/lookup_table_packer.cpp new file mode 100644 index 000000000..0ad14c925 --- /dev/null +++ b/test/systems/plonk/placeholder/lookup_table_packer.cpp @@ -0,0 +1,140 @@ +//---------------------------------------------------------------------------// +// Copyright (c) 2023 Dmitrii Tabalin +// +// MIT License +// +// Permission is hereby granted, free of charge, to any person obtaining a copy +// of this software and associated documentation files (the "Software"), to deal +// in the Software without restriction, including without limitation the rights +// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +// copies of the Software, and to permit persons to whom the Software is +// furnished to do so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. +//---------------------------------------------------------------------------// + +#include +#define BOOST_TEST_MODULE zk_lookup_table_packer_test + +#include + +#include + +#include +#include + +#include +#include + +#include + +#include +#include +#include +#include + +using namespace nil::crypto3; +using namespace nil::crypto3::zk; +using namespace nil::crypto3::zk::snark; + +class test_lookup_table : public lookup_table_definition { +public: + std::size_t rows_amount; + std::size_t columns_amount; + + test_lookup_table(std::size_t _rows_amount, std::size_t _columns_amount) : + lookup_table_definition( + "test_table" + std::to_string(_rows_amount) + "_" + std::to_string(_columns_amount)), + rows_amount(_rows_amount), + columns_amount(_columns_amount) { + std::vector column_indices(columns_amount); + std::iota(column_indices.begin(), column_indices.end(), 0); + this->subtables["full"] = {column_indices, 0, rows_amount - 1}; + } + + void generate() override { + _table.resize(columns_amount); + for (std::size_t i = 0; i < columns_amount; i++) { + _table[i].resize(rows_amount); + } + for (std::size_t i = 0; i < columns_amount; ++i) { + for (std::size_t j = 0; j < rows_amount; j++) { + this->_table[i][j] = i * rows_amount + j; + } + } + } + + std::size_t get_columns_number() override { + return columns_amount; + } + virtual std::size_t get_rows_number() override { + return rows_amount; + } +}; + +BOOST_AUTO_TEST_SUITE(lookup_table_packer_test_suite) + +BOOST_AUTO_TEST_CASE(horizontal_lookup_table_packer_test) { + using curve_type = algebra::curves::pallas; + using FieldType = typename curve_type::base_field_type; + constexpr std::size_t WitnessColumns = 9; + constexpr std::size_t PublicInputColumns = 1; + constexpr std::size_t ConstantColumns = 33; + constexpr std::size_t SelectorColumns = 50; + using hash_type = nil::crypto3::hashes::keccak_1600<256>; + constexpr std::size_t Lambda = 1; + + using ArithmetizationParams = plonk_arithmetization_params; + using ArithmetizationType = plonk_constraint_system; + plonk_constraint_system bp; + plonk_assignment_table assignment; + + std::map>> lookup_tables; + std::map lookup_table_ids; + + std::vector constant_columns_ids(ConstantColumns); + std::iota(constant_columns_ids.begin(), constant_columns_ids.end(), 0); + + test_lookup_table table(4, 5); + lookup_tables[table.table_name] = std::make_shared(table); + lookup_table_ids[table.table_name + "/full"] = 1; + + const std::size_t rows_amount = pack_lookup_tables_horizontal( + lookup_table_ids, + lookup_tables, + bp, + assignment, + constant_columns_ids, + 0, + 2); + // Check that the folding scheme worked correctly + const auto &values = table.get_table(); + const auto &constants = assignment.constants(); + for (std::size_t column = 0; column < table.get_columns_number(); column++) { + for (std::size_t row = 0; row < table.get_rows_number(); row++) { + const std::size_t assignment_row = 1 + row % 2; + const std::size_t assignment_column = column + row / 2 * table.columns_amount; + BOOST_CHECK_EQUAL(constants[assignment_column][assignment_row], values[column][row]); + } + } + // Check that the selector is ther + const auto &selectors = assignment.selectors(); + BOOST_CHECK_EQUAL(selectors[1][0], 0); + for (std::size_t row = 1; row < 3; row++) { + BOOST_CHECK_EQUAL(selectors[1][row], 1); + } +} + +BOOST_AUTO_TEST_SUITE_END() + +