14#include "gmock/gmock.h"
17#include <gtest/gtest.h>
22using ::testing::ElementsAreArray;
24using ::testing::Property;
29template <
typename G1>
class TestAffineElement :
public testing::Test {
30 using element =
typename G1::element;
31 using affine_element =
typename G1::affine_element;
32 using Fr =
typename G1::Fr;
35 static void test_read_write_buffer()
39 affine_element P = affine_element(element::random_element());
42 std::vector<uint8_t> v(64);
43 uint8_t* ptr = v.data();
44 affine_element::serialize_to_buffer(P, ptr);
46 R = affine_element::serialize_from_buffer(ptr);
47 ASSERT_TRUE(R.on_curve());
53 affine_element P = affine_element(element::random_element());
54 P.self_set_infinity();
57 std::vector<uint8_t> v(64);
58 uint8_t* ptr = v.data();
59 affine_element::serialize_to_buffer(P, ptr);
61 R = affine_element::serialize_from_buffer(ptr);
62 ASSERT_TRUE(R.is_point_at_infinity());
68 static void test_deserialize_off_curve_throws()
70 using Fq =
typename G1::Fq;
73 affine_element P = affine_element(element::random_element());
74 affine_element off_curve;
78 std::vector<uint8_t> v(
sizeof(affine_element));
79 uint8_t* ptr = v.data();
80 affine_element::serialize_to_buffer(off_curve, ptr);
82 if (!off_curve.on_curve()) {
89 static void test_read_and_write()
93 affine_element P = affine_element(element::random_element());
94 [[maybe_unused]] affine_element R;
96 std::vector<uint8_t> v(
sizeof(R));
97 uint8_t* ptr = v.data();
99 ASSERT_TRUE(P.on_curve());
104 const uint8_t* read_ptr = v.data();
107 ASSERT_TRUE(R.on_curve());
112 static void test_msgpack_serialization()
116 affine_element P = affine_element(element::random_element());
119 msgpack::sbuffer sbuf;
120 msgpack::pack(sbuf, P);
123 msgpack::object_handle oh = msgpack::unpack(sbuf.data(), sbuf.size());
124 msgpack::object deserialized = oh.get();
127 deserialized.convert(R);
129 ASSERT_TRUE(R.on_curve() && !R.is_point_at_infinity());
135 affine_element P = affine_element(element::random_element());
136 P.self_set_infinity();
139 msgpack::sbuffer sbuf;
140 msgpack::pack(sbuf, P);
143 msgpack::object_handle oh = msgpack::unpack(sbuf.data(), sbuf.size());
144 msgpack::object deserialized = oh.get();
147 deserialized.convert(R);
149 ASSERT_TRUE(R.is_point_at_infinity());
154 static void test_point_compression()
156 for (
size_t i = 0; i < 10; i++) {
157 affine_element P = affine_element(element::random_element());
160 compressed.
data[3] |= group_elements::UINT256_TOP_LIMB_MSB;
162 affine_element Q = affine_element::from_compressed(compressed);
167 static void test_point_compression_unsafe()
169 for (
size_t i = 0; i < 100; i++) {
170 affine_element P = affine_element(element::random_element());
176 EXPECT_EQ(P, Q_points[0]);
180 static void test_add_affine()
182 element lhs = element::random_element();
183 affine_element lhs_affine(lhs);
186 affine_element rhs_affine(rhs);
189 affine_element result = lhs_affine + rhs_affine;
190 EXPECT_EQ(
element(result) == expected,
true);
198 static void test_mixed_add_infinity_regression()
200 const element P = element::random_element();
201 const affine_element Q_inf = affine_element::infinity();
204 EXPECT_EQ(P + Q_inf, P);
205 EXPECT_EQ(P - Q_inf, P);
218 EXPECT_EQ(Q_inf + P, P);
219 EXPECT_EQ(Q_inf - P, -P);
222 element inf_elem = element::zero();
223 ASSERT_TRUE(inf_elem.is_point_at_infinity());
224 EXPECT_TRUE((inf_elem + Q_inf).is_point_at_infinity());
225 EXPECT_TRUE((inf_elem - Q_inf).is_point_at_infinity());
228 EXPECT_TRUE((P + Q_inf).on_curve());
229 EXPECT_TRUE((P - Q_inf).on_curve());
234 static void test_infinity_regression()
237 P.self_set_infinity();
238 affine_element R(0, P.y);
239 ASSERT_FALSE(P == R);
241 static void test_infinity_ordering_regression()
243 affine_element P(0, 1);
244 affine_element Q(0, 1);
246 P.self_set_infinity();
247 EXPECT_NE(P < Q, Q < P);
251 static void test_point_compression_invalid_x()
253 using Fq =
typename G1::Fq;
254 size_t invalid_count = 0;
255 for (
size_t i = 0; i < 20; ++i) {
257 if (!result.on_curve()) {
265 EXPECT_GT(invalid_count, 0U);
272 static void test_batch_endomorphism_by_minus_one()
278 element::batch_mul_with_endomorphism(affine_points, -affine_element::Fr::one());
281 EXPECT_EQ(affine_points[i], -result[i]);
289 static void test_fixed_point_at_infinity()
291 using Fq = affine_element::Fq;
292 affine_element P = affine_element::infinity();
295 affine_element R = affine_element(element::random_element());
300 static void test_infinity_mul_by_scalar_is_infinity()
303 EXPECT_TRUE(result.is_point_at_infinity());
306 static void test_batch_mul_matches_non_batch_mul()
310 affine_points.push_back(affine_element::infinity());
313 std::transform(affine_points.begin(),
316 [exponent](
const auto& el) { return el * exponent; });
318 EXPECT_THAT(result, ElementsAreArray(expected));
321 static void test_infinity_batch_mul_by_scalar_is_infinity()
326 EXPECT_THAT(result, Each(Property(&affine_element::is_point_at_infinity, Eq(
true))));
329 static void test_batch_mul_endomorphism_even_scalars()
331 const affine_element P = affine_element::one();
333 for (
const Fr scalar : {
Fr(0),
Fr(2),
Fr(4),
Fr(6),
Fr(8) }) {
334 const auto result = element::batch_mul_with_endomorphism(points, scalar);
335 const affine_element expected(
element(P) * scalar);
336 for (
size_t i = 0; i < points.size(); ++i) {
337 EXPECT_EQ(result[i], expected);
345 static size_t k2_bit_length(
const Fr& scalar)
352 const auto& k2 = endo.second;
354 return 128 -
static_cast<size_t>(__builtin_clzll(k2[1]));
357 return 64 -
static_cast<size_t>(__builtin_clzll(k2[0]));
365 static Fr find_scalar_with_k2_bits(
size_t target_bits,
size_t max_attempts = 2000)
367 for (
size_t i = 0; i < max_attempts; ++i) {
369 if (k2_bit_length(s) == target_bits) {
373 throw_or_abort(
"could not find scalar with desired K2 bit-width");
378 static void check_batch_mul_against_naive(
size_t num_points,
const Fr& scalar)
381 points.reserve(num_points);
383 points.push_back(affine_element(element::random_element()));
386 expected.reserve(num_points);
387 for (
const auto& p : points) {
388 expected.push_back(affine_element(
element(p) * scalar));
391 ASSERT_EQ(result.size(), expected.size());
392 EXPECT_THAT(result, ElementsAreArray(expected));
398 static void test_batch_mul_zero_scalar()
402 points.reserve(num_points);
404 points.push_back(affine_element(element::random_element()));
407 ASSERT_EQ(result.size(), num_points);
408 for (
const auto& r : result) {
409 EXPECT_TRUE(r.is_point_at_infinity());
414 static void test_batch_mul_num_points_not_multiple_of_threads()
420 static void test_batch_mul_scalar_under_127_bits()
423 const Fr scalar(
uint256_t{ 0xdeadbeefcafef00dULL, 0x3edcba98765432f1ULL, 0, 0 });
424 ASSERT_EQ(k2_bit_length(scalar), 0U);
425 check_batch_mul_against_naive(64, scalar);
429 static void test_batch_mul_scalar_low_127_bits_zero()
432 check_batch_mul_against_naive(64, scalar);
436 static void test_batch_mul_k2_128_bits_never_occurs()
438 for (
size_t i = 0; i < 10000; ++i) {
440 const size_t bits = k2_bit_length(s);
441 ASSERT_LE(bits, 127U) <<
"GLV split must produce K2 ≤ 127 bits; got " << bits <<
" bits on sample " << i;
446 static void test_batch_mul_k2_127_bits()
448 const Fr scalar = find_scalar_with_k2_bits(127);
449 ASSERT_EQ(k2_bit_length(scalar), 127U);
450 check_batch_mul_against_naive(64, scalar);
454 static void test_batch_mul_k2_126_bits()
456 const Fr scalar = find_scalar_with_k2_bits(126);
457 ASSERT_EQ(k2_bit_length(scalar), 126U);
458 check_batch_mul_against_naive(64, scalar);
462 static void test_batch_mul_k2_125_bits()
464 const Fr scalar = find_scalar_with_k2_bits(125);
465 ASSERT_EQ(k2_bit_length(scalar), 125U);
466 check_batch_mul_against_naive(64, scalar);
470 static void test_batch_mul_empty_input()
474 EXPECT_TRUE(result.empty());
478 static void test_batch_mul_size_less_than_num_threads()
480 for (
size_t sz : {
size_t{ 1 },
size_t{ 2 },
size_t{ 3 } }) {
491 static void test_batch_mul_small_scalars_edge_predicate()
495 points.reserve(num_points);
497 points.push_back(affine_element(element::random_element()));
501 for (int64_t s = -32; s <= 32; ++s) {
502 const Fr scalar = (s >= 0) ?
Fr(
static_cast<uint64_t
>(s)) : -
Fr(static_cast<uint64_t>(-s));
504 expected.reserve(num_points);
505 for (
const auto& p : points) {
506 expected.push_back(affine_element(
element(p) * scalar));
509 ASSERT_EQ(result.size(), expected.size());
511 EXPECT_EQ(result[i], expected[i]) <<
"scalar = " << s <<
", point index = " << i;
516 static void test_batch_mul_randomized_matches_naive()
518 for (
size_t trial = 0; trial < 24; ++trial) {
521 if ((trial % 8) == 0) {
522 scalar =
Fr(
static_cast<uint64_t
>(trial + 1));
524 check_batch_mul_against_naive(num_points, scalar);
528 static void test_frc_codec_round_trip()
531 affine_element point = affine_element::random_element();
534 affine_element::PUBLIC_INPUTS_SIZE);
535 auto reconstructed = FrCodec::deserialize_from_fields<affine_element>(limbs);
536 EXPECT_EQ(reconstructed, point);
541 static void test_is_in_prime_subgroup_accepts_subgroup_points()
543 EXPECT_TRUE(affine_element::infinity().is_in_prime_subgroup());
544 EXPECT_TRUE(affine_element::one().is_in_prime_subgroup());
546 for (
size_t i = 0; i < 8; ++i) {
547 affine_element P = affine_element(element::random_element());
548 EXPECT_TRUE(P.is_in_prime_subgroup());
554using TestTypes = testing::Types<bb::g1, grumpkin::g1, secp256k1::g1, secp256r1::g1>;
561 TestFixture::test_add_affine();
568 TestFixture::test_mixed_add_infinity_regression();
573 TestFixture::test_read_and_write();
578 TestFixture::test_read_write_buffer();
579 TestFixture::test_msgpack_serialization();
584 if constexpr (TypeParam::Fq::modulus.data[3] >= MODULUS_TOP_LIMB_LARGE_THRESHOLD) {
587 TestFixture::test_point_compression();
593 if constexpr (TypeParam::Fq::modulus.data[3] >= MODULUS_TOP_LIMB_LARGE_THRESHOLD) {
596 TestFixture::test_fixed_point_at_infinity();
602 if constexpr (TypeParam::Fq::modulus.data[3] >= MODULUS_TOP_LIMB_LARGE_THRESHOLD) {
603 TestFixture::test_point_compression_unsafe();
611 TestFixture::test_infinity_ordering_regression();
620 template <
typename Element,
typename Scalar>
625 template <
typename Element,
typename Scalar>
634TYPED_TEST(TestAffineElement, MulWithEndomorphismMatchesMulWithoutEndomorphism)
636 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
639 using element_t =
typename TypeParam::element;
640 using Fr =
typename TypeParam::Fr;
641 for (
int i = 0; i < 100; i++) {
642 element_t x1(element_t::random_element());
651TYPED_TEST(TestAffineElement, MulWithEndomorphismEdgeCasesMatchMulWithoutEndomorphism)
653 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
656 using element_t =
typename TypeParam::element;
657 using Fr =
typename TypeParam::Fr;
659 const element_t point(element_t::random_element());
660 std::vector<Fr> scalars;
662 for (uint64_t i = 0; i <= 64; ++i) {
663 scalars.emplace_back(i);
666 for (
const size_t bit : { 125UL, 126UL, 127UL }) {
668 for (
const uint64_t delta : { 0UL, 1UL, 2UL, 7UL, 8UL, 15UL, 16UL }) {
669 scalars.emplace_back(power + delta);
671 scalars.emplace_back(power - delta);
676 for (
const Fr& scalar : scalars) {
679 EXPECT_EQ(point * scalar, expected);
680 EXPECT_EQ(point.mul_const_time(scalar), expected);
689 using element_t =
typename TypeParam::element;
690 using Fr =
typename TypeParam::Fr;
691 element_t
G(element_t::random_element());
695 EXPECT_EQ(
G.mul_const_time(s),
G * s);
698 for (
int i = 0; i < 50; ++i) {
700 EXPECT_EQ(
G.mul_const_time(s),
G * s);
708 TestFixture::test_frc_codec_round_trip();
716TYPED_TEST(TestAffineElement, BatchMulEndomorphismEvenScalars)
718 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
721 TestFixture::test_batch_mul_endomorphism_even_scalars();
728 TestFixture::test_infinity_mul_by_scalar_is_infinity();
734 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
737 TestFixture::test_batch_mul_matches_non_batch_mul();
742TYPED_TEST(TestAffineElement, InfinityBatchMulByScalarIsInfinity)
744 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
747 TestFixture::test_infinity_batch_mul_by_scalar_is_infinity();
753 if constexpr (TypeParam::USE_ENDOMORPHISM) {
754 TestFixture::test_batch_endomorphism_by_minus_one();
765 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
768 TestFixture::test_batch_mul_zero_scalar();
772TYPED_TEST(TestAffineElement, BatchMulNumPointsNotMultipleOfThreads)
774 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
777 TestFixture::test_batch_mul_num_points_not_multiple_of_threads();
783 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
786 TestFixture::test_batch_mul_scalar_under_127_bits();
792 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
795 TestFixture::test_batch_mul_scalar_low_127_bits_zero();
801 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
804 TestFixture::test_batch_mul_k2_128_bits_never_occurs();
810 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
813 TestFixture::test_batch_mul_k2_127_bits();
819 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
822 TestFixture::test_batch_mul_k2_126_bits();
828 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
831 TestFixture::test_batch_mul_k2_125_bits();
837 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
840 TestFixture::test_batch_mul_empty_input();
846 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
849 TestFixture::test_batch_mul_size_less_than_num_threads();
855 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
858 TestFixture::test_batch_mul_randomized_matches_naive();
862TYPED_TEST(TestAffineElement, BatchMulSmallScalarsEdgePredicate)
864 if constexpr (!TypeParam::USE_ENDOMORPHISM) {
867 TestFixture::test_batch_mul_small_scalars_edge_predicate();
874 TestFixture::test_deserialize_off_curve_throws();
878TYPED_TEST(TestAffineElement, IsInPrimeSubgroupAcceptsSubgroupPoints)
880 TestFixture::test_is_in_prime_subgroup_accepts_subgroup_points();
886 if constexpr (TypeParam::Fq::modulus.data[3] >= MODULUS_TOP_LIMB_LARGE_THRESHOLD) {
889 TestFixture::test_point_compression_invalid_x();
893TEST(AffineElement, HashToCurve)
898 fr(
uint256_t(
"24c4cb9c1206ab5470592f237f1698abe684dadf0ab4d7a132c32b2134e2c12e")),
899 fr(
uint256_t(
"0668b8d61a317fb34ccad55c930b3554f1828a0e5530479ecab4defe6bbc0b2e"))));
903 fr(
uint256_t(
"107f1b633c6113f3222f39f6256f0546b41a4880918c86864b06471afb410454")),
904 fr(
uint256_t(
"050cd3823d0c01590b6a50adcc85d2ee4098668fd28805578aa05a423ea938c6"))));
907 test_vectors.emplace_back(std::vector<uint8_t>{ 0x68, 0x65, 0x6c, 0x6c, 0x6f, 0x20, 0x77, 0x6f, 0x72, 0x6c, 0x64 },
909 fr(
uint256_t(
"037c5c229ae495f6e8d1b4bf7723fafb2b198b51e27602feb8a4d1053d685093")),
910 fr(
uint256_t(
"10cf9596c5b2515692d930efa2cf3817607e4796856a79f6af40c949b066969f"))));
913 auto result = grumpkin::g1::affine_element::hash_to_curve(
std::get<0>(test_case), 0);
#define EXPECT_THROW_OR_ABORT(statement, matcher)
static std::vector< fr > serialize_to_fields(const T &val)
Conversion from transcript values to bb::frs.
static Element mul_without_endomorphism(const Element &element, const Scalar &scalar)
static Element mul_with_endomorphism(const Element &element, const Scalar &scalar)
element class. Implements ecc group arithmetic using Jacobian coordinates See https://hyperelliptic....
element mul_with_endomorphism(const Fr &scalar) const noexcept
element mul_without_endomorphism(const Fr &scalar) const noexcept
group_elements::affine_element< Fq, Fr, Params > affine_element
constexpr bool get_bit(uint64_t bit_index) const
#define G(r, i, a, b, c, d)
test_vector test_vectors[]
std::conditional_t< IsGoblinBigGroup< C, Fq, Fr, G >, element_goblin::goblin_element< C, goblin_field< C >, Fr, G >, element_default::element< C, Fq, Fr, G > > element
element wraps either element_default::element or element_goblin::goblin_element depending on parametr...
Entry point for Barretenberg command-line interface.
void read(B &it, field2< base_field, Params > &value)
TYPED_TEST_SUITE(CommitmentKeyTest, Curves)
field< Bn254FrParams > fr
void write(B &buf, field2< base_field, Params > const &value)
TYPED_TEST(CommitmentKeyTest, CommitToZeroPoly)
TEST(BoomerangMegaCircuitBuilder, BasicCircuit)
constexpr decltype(auto) get(::tuplet::tuple< T... > &&t) noexcept
testing::Types< VKTestParams< UltraFlavor, stdlib::recursion::honk::DefaultIO< UltraCircuitBuilder > >, VKTestParams< UltraFlavor, stdlib::recursion::honk::RollupIO >, VKTestParams< UltraKeccakFlavor, stdlib::recursion::honk::DefaultIO< UltraCircuitBuilder > >, VKTestParams< MegaFlavor, stdlib::recursion::honk::DefaultIO< MegaCircuitBuilder > > > TestTypes
static constexpr field one()
static void split_into_endomorphism_scalars(const field &k, field &k1, field &k2)
Full-width endomorphism decomposition: k ≡ k1 - k2·λ (mod r). Modifies the field elements k1 and k2.
static field random_element(numeric::RNG *engine=nullptr) noexcept
BB_INLINE constexpr bool is_zero() const noexcept
BB_INLINE constexpr field from_montgomery_form() const noexcept
static constexpr field zero()
void throw_or_abort(std::string const &err)