From ee89aa98551851650b22f817a5b8298c8074e9b4 Mon Sep 17 00:00:00 2001 From: ludamad Date: Thu, 15 Feb 2024 16:40:35 +0000 Subject: [PATCH 1/3] fix: g2.Serialize sporadic failure --- .../interactive/Dockerfile.msan.ubuntu | 4 +- .../ecc/groups/affine_element.hpp | 52 +++++-------------- .../ecc/groups/affine_element_impl.hpp | 6 +-- 3 files changed, 18 insertions(+), 44 deletions(-) diff --git a/barretenberg/cpp/dockerfiles/interactive/Dockerfile.msan.ubuntu b/barretenberg/cpp/dockerfiles/interactive/Dockerfile.msan.ubuntu index 725dac7689d..d330a2de5ba 100644 --- a/barretenberg/cpp/dockerfiles/interactive/Dockerfile.msan.ubuntu +++ b/barretenberg/cpp/dockerfiles/interactive/Dockerfile.msan.ubuntu @@ -35,8 +35,8 @@ RUN cmake -G Ninja -S llvm-project/runtimes -B llvm-project/build \ RUN ninja -C llvm-project/build cxx cxxabi RUN ninja -C llvm-project/build install-cxx install-cxxabi -ENV MSAN_CFLAGS="-std=c++20 -fsanitize=memory -nostdinc++ -I/opt/include -I/opt/include/c++/v1" -ENV MSAN_LFLAGS="-fsanitize=memory -stdlib=libc++ -L/opt/lib -lc++abi -Wl,-rpath,/opt/lib" +ENV MSAN_CFLAGS="-std=c++20 -fsanitize=memory -fsanitize-memory-track-origins -nostdinc++ -I/opt/include -I/opt/include/c++/v1" +ENV MSAN_LFLAGS="-fsanitize=memory -fsanitize-memory-track-origins -stdlib=libc++ -L/opt/lib -lc++abi -Wl,-rpath,/opt/lib" WORKDIR /usr/src/barretenberg/cpp diff --git a/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp b/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp index be823c3f176..f3d0e5499bc 100644 --- a/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp +++ b/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp @@ -2,6 +2,7 @@ #include "barretenberg/ecc/curves/bn254/fq2.hpp" #include "barretenberg/numeric/uint256/uint256.hpp" #include "barretenberg/serialize/msgpack.hpp" +#include #include #include @@ -21,7 +22,7 @@ template class alignas(64) affine_ affine_element() noexcept = default; ~affine_element() noexcept = default; - constexpr affine_element(const Fq& a, const Fq& b) noexcept; + constexpr affine_element(const Fq& x, const Fq& y) noexcept; constexpr affine_element(const affine_element& other) noexcept = default; @@ -103,13 +104,8 @@ template class alignas(64) affine_ static void serialize_to_buffer(const affine_element& value, uint8_t* buffer) { if (value.is_point_at_infinity()) { - if constexpr (Fq::modulus.get_msb() == 255) { - write(buffer, uint256_t(0)); - write(buffer, Fq::modulus); - } else { - write(buffer, uint256_t(0)); - write(buffer, uint256_t(1) << 255); - } + // if we are infinity, just set all buffer bits to 1 + memset(buffer, 255, sizeof(Fq) * 2); } else { Fq::serialize_to_buffer(value.y, buffer); Fq::serialize_to_buffer(value.x, buffer + sizeof(Fq)); @@ -130,38 +126,16 @@ template class alignas(64) affine_ */ static affine_element serialize_from_buffer(uint8_t* buffer) { - affine_element result; - - // need to read a raw uint256_t to avoid reductions so we can check whether the point is the point at infinity - auto raw_x = from_buffer(buffer + sizeof(Fq)); - - if constexpr (Fq::modulus.get_msb() == 255) { - if (raw_x == Fq::modulus) { - result.y = Fq::zero(); - result.x.data[0] = raw_x.data[0]; - result.x.data[1] = raw_x.data[1]; - result.x.data[2] = raw_x.data[2]; - result.x.data[3] = raw_x.data[3]; - } else { - result.y = Fq::serialize_from_buffer(buffer); - result.x = Fq(raw_x); - } - } else { - if (raw_x.get_msb() == 255) { - result.y = Fq::zero(); - result.x = Fq::zero(); - result.self_set_infinity(); - } else { - // conditional here to avoid reading the same data twice in case of a field of prime order - if constexpr (std::is_same::value) { - result.y = Fq::serialize_from_buffer(buffer); - result.x = Fq::serialize_from_buffer(buffer + sizeof(Fq)); - } else { - result.y = Fq::serialize_from_buffer(buffer); - result.x = Fq(raw_x); - } - } + // Does the buffer consist entirely of 255's? If so, we have a point at infinity + // Note that if it isn't, this loop should end early. + bool is_point_at_infinity = + std::all_of(buffer, buffer + sizeof(Fq) * 2, [](uint8_t val) { return val == 255; }); + if (is_point_at_infinity) { + return affine_element::infinity(); } + affine_element result; + result.y = Fq::serialize_from_buffer(buffer); + result.x = Fq::serialize_from_buffer(buffer + sizeof(Fq)); return result; } diff --git a/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element_impl.hpp b/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element_impl.hpp index a227ba27312..c1906e40374 100644 --- a/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element_impl.hpp +++ b/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element_impl.hpp @@ -5,9 +5,9 @@ namespace bb::group_elements { template -constexpr affine_element::affine_element(const Fq& a, const Fq& b) noexcept - : x(a) - , y(b) +constexpr affine_element::affine_element(const Fq& x, const Fq& y) noexcept + : x(x) + , y(y) {} template From 0193ca95ee166e3d20f0fb0282a8663f6269479e Mon Sep 17 00:00:00 2001 From: ludamad Date: Thu, 15 Feb 2024 16:44:43 +0000 Subject: [PATCH 2/3] consistency --- barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp b/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp index f3d0e5499bc..7bd8fc9ef48 100644 --- a/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp +++ b/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp @@ -126,7 +126,7 @@ template class alignas(64) affine_ */ static affine_element serialize_from_buffer(uint8_t* buffer) { - // Does the buffer consist entirely of 255's? If so, we have a point at infinity + // Does the buffer consist entirely of set bits? If so, we have a point at infinity // Note that if it isn't, this loop should end early. bool is_point_at_infinity = std::all_of(buffer, buffer + sizeof(Fq) * 2, [](uint8_t val) { return val == 255; }); From 8a34999e4292f34eb1d777e75240cf3fd242fd4f Mon Sep 17 00:00:00 2001 From: ludamad Date: Thu, 15 Feb 2024 16:54:08 +0000 Subject: [PATCH 3/3] comment++ --- barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp | 2 ++ 1 file changed, 2 insertions(+) diff --git a/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp b/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp index 7bd8fc9ef48..003478324e5 100644 --- a/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp +++ b/barretenberg/cpp/src/barretenberg/ecc/groups/affine_element.hpp @@ -105,6 +105,7 @@ template class alignas(64) affine_ { if (value.is_point_at_infinity()) { // if we are infinity, just set all buffer bits to 1 + // we only need this case because the below gets mangled converting from montgomery for infinity points memset(buffer, 255, sizeof(Fq) * 2); } else { Fq::serialize_to_buffer(value.y, buffer); @@ -128,6 +129,7 @@ template class alignas(64) affine_ { // Does the buffer consist entirely of set bits? If so, we have a point at infinity // Note that if it isn't, this loop should end early. + // We only need this case because the below gets mangled converting to montgomery for infinity points bool is_point_at_infinity = std::all_of(buffer, buffer + sizeof(Fq) * 2, [](uint8_t val) { return val == 255; }); if (is_point_at_infinity) {