Barretenberg
The ZK-SNARK library at the core of Aztec
Loading...
Searching...
No Matches
affine_element_impl.hpp
Go to the documentation of this file.
1// === AUDIT STATUS ===
2// internal: { status: Planned, auditors: [], commit: }
3// external_1: { status: not started, auditors: [], commit: }
4// external_2: { status: not started, auditors: [], commit: }
5// =====================
6
7#pragma once
8#include "./element.hpp"
11
12namespace bb::group_elements {
13template <class Fq, class Fr, class T>
15 : x(x)
16 , y(y)
17{}
18
19template <class Fq, class Fr, class T>
20template <typename BaseField, typename CompileTimeEnabled>
22{
23 uint256_t x_coordinate = compressed;
24 x_coordinate.data[3] = x_coordinate.data[3] & (~UINT256_TOP_LIMB_MSB);
25 bool y_bit = compressed.get_bit(255);
26
27 Fq x = Fq(x_coordinate);
28 Fq y2 = (x.sqr() * x + T::b);
29 if constexpr (T::has_a) {
30 y2 += (x * T::a);
31 }
32 auto [is_quadratic_remainder, y] = y2.sqrt();
33 if (!is_quadratic_remainder) {
34 return affine_element(Fq::zero(), Fq::zero());
35 }
36 if (uint256_t(y).get_bit(0) != y_bit) {
37 y = -y;
38 }
39
40 return affine_element<Fq, Fr, T>(x, y);
41}
42
43template <class Fq, class Fr, class T>
44template <typename BaseField, typename CompileTimeEnabled>
46 const uint256_t& compressed) noexcept
47{
48 // Try x as a recovery candidate: check it is in [0, p), compute y² = x³ + ax + b,
49 // and return the point if y exists. Fq(x) reduces silently, so ensuring x is in [0, p) is necessary to prevent
50 // returning an incorrect pair (x mod q, y).
51 auto try_candidate = [](const uint256_t& x_coordinate) -> affine_element<Fq, Fr, T> {
52 if (x_coordinate >= Fq::modulus) {
53 return { Fq::zero(), Fq::zero() };
54 }
55 Fq x = Fq(x_coordinate);
56 Fq y2 = ((x.sqr() * x) + T::b);
57 if constexpr (T::has_a) {
58 y2 += (x * T::a);
59 }
60 auto [is_qr, y] = y2.sqrt();
62 };
63
64 return { try_candidate(compressed), try_candidate(compressed + Fr::modulus) };
65}
66
67template <class Fq, class Fr, class T>
73
74template <class Fq, class Fr, class T>
75constexpr affine_element<Fq, Fr, T> affine_element<Fq, Fr, T>::operator*(const Fr& exponent) const noexcept
76{
77 return bb::group_elements::element(*this) * exponent;
78}
79
80template <class Fq, class Fr, class T> constexpr affine_element<Fq, Fr, T> affine_element<Fq, Fr, T>::infinity()
81{
84 return e;
85}
86
87template <class Fq, class Fr, class T>
89{
90 affine_element result(*this);
91 result.self_set_infinity();
92 return result;
93}
94
95template <class Fq, class Fr, class T> constexpr void affine_element<Fq, Fr, T>::self_set_infinity() noexcept
96{
97 if constexpr (Fq::modulus.data[3] >= MODULUS_TOP_LIMB_LARGE_THRESHOLD) {
98 // We set the value of x equal to modulus to represent inifinty
99 x.data[0] = Fq::modulus.data[0];
100 x.data[1] = Fq::modulus.data[1];
101 x.data[2] = Fq::modulus.data[2];
102 x.data[3] = Fq::modulus.data[3];
103
104 // Clear y for memory hygiene
105 y = Fq::zero();
106 } else {
107 (*this).x = Fq::zero();
108 (*this).y = Fq::zero();
109 x.self_set_msb();
110 }
111}
112
113template <class Fq, class Fr, class T> constexpr bool affine_element<Fq, Fr, T>::is_point_at_infinity() const noexcept
114{
115 if constexpr (Fq::modulus.data[3] >= MODULUS_TOP_LIMB_LARGE_THRESHOLD) {
116 // We check if the value of x is equal to modulus to represent inifinty
117 return ((x.data[0] ^ Fq::modulus.data[0]) | (x.data[1] ^ Fq::modulus.data[1]) |
118 (x.data[2] ^ Fq::modulus.data[2]) | (x.data[3] ^ Fq::modulus.data[3])) == 0;
119
120 } else {
121 return (x.is_msb_set());
122 }
123}
124
125template <class Fq, class Fr, class T> constexpr bool affine_element<Fq, Fr, T>::on_curve() const noexcept
126{
127 if (is_point_at_infinity()) {
128 return true;
129 }
130 Fq xxx = x.sqr() * x + T::b;
131 Fq yy = y.sqr();
132 if constexpr (T::has_a) {
133 xxx += (x * T::a);
134 }
135 return (xxx == yy);
136}
137
138template <class Fq, class Fr, class T> bool affine_element<Fq, Fr, T>::is_in_prime_subgroup() const noexcept
139{
140 if (is_point_at_infinity()) {
141 return true;
142 }
143 // Weierstrass group law is unsound for off-curve coordinates, so the [r]·P trick can
144 // give a false positive on points that satisfy y² = x³ + b' for some b' ≠ b. Reject
145 // those up front.
146 if (!on_curve()) {
147 return false;
148 }
150
151 // To compute r * P, we convert modulus r to u256 and perform a left-to-right double-and-add.
152 constexpr uint256_t r = Fr::modulus;
153 const uint64_t r_msb = r.get_msb();
154
155 // Left-to-right double-and-add over the bits of r below the MSB. The MSB itself is consumed by
156 // initializing `acc` with `*this`. Loop terminates via unsigned underflow (i wraps past 0).
157 Element acc(*this);
158 for (uint64_t i = r_msb - 1; i < r_msb; --i) {
159 acc.self_dbl();
160 if (r.get_bit(i)) {
161 acc += *this;
162 }
163 }
164 return acc.is_point_at_infinity();
165}
166
167template <class Fq, class Fr, class T>
168constexpr bool affine_element<Fq, Fr, T>::operator==(const affine_element& other) const noexcept
169{
170 bool this_is_infinity = is_point_at_infinity();
171 bool other_is_infinity = other.is_point_at_infinity();
172 bool both_infinity = this_is_infinity && other_is_infinity;
173 bool only_one_is_infinity = this_is_infinity != other_is_infinity;
174 return !only_one_is_infinity && (both_infinity || ((x == other.x) && (y == other.y)));
175}
176
177template <class Fq, class Fr, class T>
178constexpr bool affine_element<Fq, Fr, T>::operator>(const affine_element& other) const noexcept
179{
180 if (is_point_at_infinity()) {
181 return false;
182 }
183 if (other.is_point_at_infinity()) {
184 return true;
185 }
186
187 if (x > other.x) {
188 return true;
189 }
190 if (x == other.x && y > other.y) {
191 return true;
192 }
193 return false;
194}
195
196template <class Fq, class Fr, class T>
198 const Fq& x, bool sign_bit) noexcept
199{
200 auto yy = x.sqr() * x + T::b;
201 if constexpr (T::has_a) {
202 yy += (x * T::a);
203 }
204 auto [found_root, y] = yy.sqrt();
205
206 if (found_root) {
207 if (uint256_t(y).get_bit(0) != sign_bit) {
208 y = -y;
209 }
210 return affine_element(x, y);
211 }
212 return std::nullopt;
213}
214
247template <class Fq, class Fr, class T>
249 uint8_t attempt_count) noexcept
251{
252 std::vector<uint8_t> target_seed(seed);
253 // expand by 2 bytes to cover incremental hash attempts
254 const size_t seed_size = seed.size();
255 for (size_t i = 0; i < 2; ++i) {
256 target_seed.push_back(0);
257 }
258 target_seed[seed_size] = attempt_count;
259 target_seed[seed_size + 1] = 0;
260 const auto hash_hi = blake3::blake3s_constexpr(&target_seed[0], target_seed.size());
261 target_seed[seed_size + 1] = 1;
262 const auto hash_lo = blake3::blake3s_constexpr(&target_seed[0], target_seed.size());
263 // custom serialize methods as common/serialize.hpp is not constexpr!
264 const auto read_uint256 = [](const uint8_t* in) {
265 const auto read_limb = [](const uint8_t* in, uint64_t& out) {
266 for (size_t i = 0; i < 8; ++i) {
267 out += static_cast<uint64_t>(in[i]) << ((7 - i) * 8);
268 }
269 };
270 uint256_t out = 0;
271 read_limb(&in[0], out.data[3]);
272 read_limb(&in[8], out.data[2]);
273 read_limb(&in[16], out.data[1]);
274 read_limb(&in[24], out.data[0]);
275 return out;
276 };
277 // interpret 64 byte hash output as a uint512_t, reduce to Fq element
278 //(512 bits of entropy ensures result is not biased as 512 >> Fq::modulus.get_msb())
279 Fq x(uint512_t(read_uint256(&hash_lo[0]), read_uint256(&hash_hi[0])));
280 bool sign_bit = hash_hi[0] > 127;
281 std::optional<affine_element> result = derive_from_x_coordinate(x, sign_bit);
282 if (result.has_value()) {
283 return result.value();
284 }
285 return hash_to_curve(seed, attempt_count + 1);
286}
287
288template <typename Fq, typename Fr, typename T>
290{
291 if (engine == nullptr) {
293 }
294
295 Fq x;
296 Fq y;
297 while (true) {
298 // Sample a random x-coordinate and check if it satisfies curve equation.
300 // Negate the y-coordinate based on a randomly sampled bit.
301 bool sign_bit = (engine->get_random_uint8() & 1) != 0;
302
303 std::optional<affine_element> result = derive_from_x_coordinate(x, sign_bit);
304
305 if (result.has_value()) {
306 return result.value();
307 }
308 }
309 throw_or_abort("affine_element::random_element error");
310 return affine_element<Fq, Fr, T>(x, y);
311}
312
313} // namespace bb::group_elements
bool is_in_prime_subgroup() const noexcept
Check that the point lies in the prime-order subgroup of size Fr::modulus.
static constexpr std::array< affine_element, 2 > from_compressed_unsafe(const uint256_t &compressed) noexcept
Reconstruct a point in affine coordinates from compressed form.
constexpr bool is_point_at_infinity() const noexcept
static affine_element random_element(numeric::RNG *engine=nullptr) noexcept
Samples a random point on the curve.
constexpr void self_set_infinity() noexcept
static constexpr affine_element infinity()
constexpr affine_element operator+(const affine_element &other) const noexcept
static constexpr affine_element from_compressed(const uint256_t &compressed) noexcept
Reconstruct a point in affine coordinates from compressed form.
static affine_element hash_to_curve(const std::vector< uint8_t > &seed, uint8_t attempt_count=0) noexcept
Hash a seed buffer into a point.
constexpr bool on_curve() const noexcept
constexpr bool operator==(const affine_element &other) const noexcept
static constexpr std::optional< affine_element > derive_from_x_coordinate(const Fq &x, bool sign_bit) noexcept
constexpr bool operator>(const affine_element &other) const noexcept
constexpr affine_element operator*(const Fr &exponent) const noexcept
constexpr affine_element set_infinity() const noexcept
element class. Implements ecc group arithmetic using Jacobian coordinates See https://hyperelliptic....
Definition element.hpp:35
constexpr bool get_bit(uint64_t bit_index) const
constexpr uint64_t get_msb() const
bb::curve::BN254::Element Element
numeric::RNG & engine
uint256_t read_uint256(const uint8_t *data, size_t buffer_size=32)
AffineElement const size_t Fq *scratch_space noexcept
uintx< uint256_t > uint512_t
Definition uintx.hpp:309
RNG & get_randomness()
Definition engine.cpp:258
constexpr std::array< uint8_t, BLAKE3_OUT_LEN > blake3s_constexpr(const uint8_t *input, size_t input_size)
constexpr decltype(auto) get(::tuplet::tuple< T... > &&t) noexcept
Definition tuple.hpp:13
grumpkin::fq Fq
static constexpr uint256_t modulus
static field random_element(numeric::RNG *engine=nullptr) noexcept
BB_INLINE constexpr field sqr() const noexcept
static constexpr field zero()
void throw_or_abort(std::string const &err)