Botan 3.11.0
Crypto and TLS for C&
ec_inner_data.h
Go to the documentation of this file.
1/*
2* (C) 2024 Jack Lloyd
3*
4* Botan is released under the Simplified BSD License (see license.txt)
5*/
6
7#ifndef BOTAN_EC_INNER_DATA_H_
8#define BOTAN_EC_INNER_DATA_H_
9
10#include <botan/ec_group.h>
11
12#include <botan/asn1_obj.h>
13#include <botan/bigint.h>
14#include <botan/internal/monty.h>
15#include <memory>
16#include <span>
17
18#if defined(BOTAN_HAS_LEGACY_EC_POINT)
19 #include <botan/internal/barrett.h>
20#endif
21
22namespace Botan {
23
24#if defined(BOTAN_HAS_LEGACY_EC_POINT)
26#endif
27
28namespace PCurve {
29
30class PrimeOrderCurve;
31
32}
33
34class EC_Group_Data;
35
36class EC_Scalar_Data /* NOLINT(*-special-member-functions) */ {
37 public:
38 virtual ~EC_Scalar_Data() = default;
39
40 virtual const std::shared_ptr<const EC_Group_Data>& group() const = 0;
41
42 virtual size_t bytes() const = 0;
43
44 virtual std::unique_ptr<EC_Scalar_Data> clone() const = 0;
45
46 virtual bool is_zero() const = 0;
47
48 virtual bool is_eq(const EC_Scalar_Data& y) const = 0;
49
50 virtual void assign(const EC_Scalar_Data& y) = 0;
51
52 virtual void zeroize() = 0;
53
54 virtual void square_self() = 0;
55
56 virtual std::unique_ptr<EC_Scalar_Data> negate() const = 0;
57
58 virtual std::unique_ptr<EC_Scalar_Data> invert() const = 0;
59
60 virtual std::unique_ptr<EC_Scalar_Data> invert_vartime() const = 0;
61
62 virtual std::unique_ptr<EC_Scalar_Data> add(const EC_Scalar_Data& other) const = 0;
63
64 virtual std::unique_ptr<EC_Scalar_Data> sub(const EC_Scalar_Data& other) const = 0;
65
66 virtual std::unique_ptr<EC_Scalar_Data> mul(const EC_Scalar_Data& other) const = 0;
67
68 virtual void serialize_to(std::span<uint8_t> bytes) const = 0;
69};
70
71class EC_AffinePoint_Data /* NOLINT(*-special-member-functions) */ {
72 public:
73 virtual ~EC_AffinePoint_Data() = default;
74
75 virtual const std::shared_ptr<const EC_Group_Data>& group() const = 0;
76
77 virtual std::unique_ptr<EC_AffinePoint_Data> clone() const = 0;
78
79 // Return size of a field element
80 virtual size_t field_element_bytes() const = 0;
81
82 // Return if this is the identity element
83 virtual bool is_identity() const = 0;
84
85 // Writes 1 field element worth of data to bytes
86 virtual void serialize_x_to(std::span<uint8_t> bytes) const = 0;
87
88 // Writes 1 field element worth of data to bytes
89 virtual void serialize_y_to(std::span<uint8_t> bytes) const = 0;
90
91 // Writes 2 field elements worth of data to bytes
92 virtual void serialize_xy_to(std::span<uint8_t> bytes) const = 0;
93
94 // Writes 1 byte + 1 field element worth of data to bytes
95 virtual void serialize_compressed_to(std::span<uint8_t> bytes) const = 0;
96
97 // Writes 1 byte + 2 field elements worth of data to bytes
98 virtual void serialize_uncompressed_to(std::span<uint8_t> bytes) const = 0;
99
100 virtual std::unique_ptr<EC_AffinePoint_Data> mul(const EC_Scalar_Data& scalar,
101 RandomNumberGenerator& rng) const = 0;
102
104
105#if defined(BOTAN_HAS_LEGACY_EC_POINT)
106 virtual EC_Point to_legacy_point() const = 0;
107#endif
108};
109
110class EC_Mul2Table_Data /* NOLINT(*-special-member-functions) */ {
111 public:
112 virtual ~EC_Mul2Table_Data() = default;
113
114 // Returns nullptr if g*x + h*y was point at infinity
115 virtual std::unique_ptr<EC_AffinePoint_Data> mul2_vartime(const EC_Scalar_Data& x,
116 const EC_Scalar_Data& y) const = 0;
117
118 // Check if v == (g*x + h*y).x % n
119 //
120 // Returns false if g*x + h*y was point at infinity
122 const EC_Scalar_Data& x,
123 const EC_Scalar_Data& y) const = 0;
124};
125
126class EC_Group_Data final : public std::enable_shared_from_this<EC_Group_Data> {
127 public:
128 static std::shared_ptr<EC_Group_Data> create(const BigInt& p,
129 const BigInt& a,
130 const BigInt& b,
131 const BigInt& g_x,
132 const BigInt& g_y,
133 const BigInt& order,
134 const BigInt& cofactor,
135 const OID& oid,
137
139
140 EC_Group_Data(const EC_Group_Data& other) = delete;
141 EC_Group_Data(EC_Group_Data&& other) = delete;
142 EC_Group_Data& operator=(const EC_Group_Data& other) = delete;
144
145 bool params_match(const BigInt& p,
146 const BigInt& a,
147 const BigInt& b,
148 const BigInt& g_x,
149 const BigInt& g_y,
150 const BigInt& order,
151 const BigInt& cofactor) const;
152
153 // Like the other params_match but accepting the base point in encoded form
154 bool params_match(const BigInt& p,
155 const BigInt& a,
156 const BigInt& b,
157 std::span<const uint8_t> base_pt,
158 const BigInt& order,
159 const BigInt& cofactor) const;
160
161 bool params_match(const EC_Group_Data& other) const;
162
163 void set_oid(const OID& oid);
164
165 const OID& oid() const { return m_oid; }
166
167 const std::vector<uint8_t>& der_named_curve() const { return m_der_named_curve; }
168
169 const BigInt& p() const { return m_p; }
170
171 const BigInt& a() const { return m_a; }
172
173 const BigInt& b() const { return m_b; }
174
175 const BigInt& order() const { return m_order; }
176
177 const BigInt& cofactor() const { return m_cofactor; }
178
179#if defined(BOTAN_HAS_LEGACY_EC_POINT)
180 const CurveGFp& curve() const { return m_curve; }
181
182 const EC_Point& base_point() const { return m_base_point; }
183
184 const Montgomery_Params& monty() const { return m_monty; }
185
186 const BigInt& monty_a() const { return m_a_r; }
187
188 const BigInt& monty_b() const { return m_b_r; }
189
190 const Barrett_Reduction& mod_order() const { return m_mod_order; }
191#endif
192
193 bool order_is_less_than_p() const { return m_order_is_less_than_p; }
194
195 bool has_cofactor() const { return m_has_cofactor; }
196
197 const BigInt& g_x() const { return m_g_x; }
198
199 const BigInt& g_y() const { return m_g_y; }
200
201 size_t p_words() const { return m_p_words; }
202
203 size_t p_bits() const { return m_p_bits; }
204
205 size_t p_bytes() const { return (m_p_bits + 7) / 8; }
206
207 size_t order_bits() const { return m_order_bits; }
208
209 size_t order_bytes() const { return m_order_bytes; }
210
211 bool a_is_minus_3() const { return m_a_is_minus_3; }
212
213 bool a_is_zero() const { return m_a_is_zero; }
214
215 EC_Group_Source source() const { return m_source; }
216
217 EC_Group_Engine engine() const { return m_engine; }
218
219 /// Scalar from bytes
220 ///
221 /// This returns a value only if the bytes represent (in big-endian encoding) an integer
222 /// that is less than n, where n is the group order. It requires that the fixed length
223 /// encoding (with zero prefix) be used. It also rejects inputs that encode zero.
224 /// Thus the accepted range is [1,n)
225 ///
226 /// If the input is rejected then nullptr is returned
227 std::unique_ptr<EC_Scalar_Data> scalar_deserialize(std::span<const uint8_t> bytes) const;
228
229 /// Scalar from bytes with ECDSA style truncation
230 ///
231 /// This should always succeed
232 std::unique_ptr<EC_Scalar_Data> scalar_from_bytes_with_trunc(std::span<const uint8_t> bytes) const;
233
234 /// Scalar from bytes with modular reduction
235 ///
236 /// This returns a value only if bytes represents (in big-endian encoding) an integer
237 /// that is at most the square of the scalar group size. Otherwise it returns nullptr.
238 std::unique_ptr<EC_Scalar_Data> scalar_from_bytes_mod_order(std::span<const uint8_t> bytes) const;
239
240 /// Scalar from BigInt
241 ///
242 /// This returns a value only if bn is in [1,n) where n is the group order.
243 /// Otherwise it returns nullptr
244 std::unique_ptr<EC_Scalar_Data> scalar_from_bigint(const BigInt& bn) const;
245
246 /// Return a random scalar
247 ///
248 /// This will be in the range [1,n) where n is the group order
249 std::unique_ptr<EC_Scalar_Data> scalar_random(RandomNumberGenerator& rng) const;
250
251 std::unique_ptr<EC_Scalar_Data> scalar_one() const;
252
253 std::unique_ptr<EC_Scalar_Data> gk_x_mod_order(const EC_Scalar_Data& scalar, RandomNumberGenerator& rng) const;
254
255 /// Deserialize a point
256 ///
257 /// Returns nullptr if the point encoding was invalid or not on the curve
258 std::unique_ptr<EC_AffinePoint_Data> point_deserialize(std::span<const uint8_t> bytes) const;
259
260 std::unique_ptr<EC_AffinePoint_Data> point_hash_to_curve_ro(std::string_view hash_fn,
261 std::span<const uint8_t> input,
262 std::span<const uint8_t> domain_sep) const;
263
264 std::unique_ptr<EC_AffinePoint_Data> point_hash_to_curve_nu(std::string_view hash_fn,
265 std::span<const uint8_t> input,
266 std::span<const uint8_t> domain_sep) const;
267
268 std::unique_ptr<EC_AffinePoint_Data> point_g_mul(const EC_Scalar_Data& scalar, RandomNumberGenerator& rng) const;
269
270 std::unique_ptr<EC_AffinePoint_Data> mul_px_qy(const EC_AffinePoint_Data& p,
271 const EC_Scalar_Data& x,
272 const EC_AffinePoint_Data& q,
273 const EC_Scalar_Data& y,
274 RandomNumberGenerator& rng) const;
275
276 std::unique_ptr<EC_AffinePoint_Data> affine_add(const EC_AffinePoint_Data& p, const EC_AffinePoint_Data& q) const;
277
278 std::unique_ptr<EC_AffinePoint_Data> affine_neg(const EC_AffinePoint_Data& p) const;
279
280 std::unique_ptr<EC_Mul2Table_Data> make_mul2_table(const EC_AffinePoint_Data& pt) const;
281
283 BOTAN_ASSERT_NONNULL(m_pcurve);
284 return *m_pcurve;
285 }
286
287 /**
288 * Note this constructor should *only* be called by EC_Group_Data::create.
289 *
290 * It is only public to allow use of std::make_shared
291 */
292 EC_Group_Data(const BigInt& p,
293 const BigInt& a,
294 const BigInt& b,
295 const BigInt& g_x,
296 const BigInt& g_y,
297 const BigInt& order,
298 const BigInt& cofactor,
299 const OID& oid,
301
302 private:
303 // Possibly nullptr (if pcurves not available or not a standard curve)
304 std::shared_ptr<const PCurve::PrimeOrderCurve> m_pcurve;
305
306#if defined(BOTAN_HAS_LEGACY_EC_POINT)
307 // Set only if m_pcurve is nullptr
308 std::unique_ptr<EC_Point_Base_Point_Precompute> m_base_mult;
309#endif
310
311 BigInt m_p;
312 BigInt m_a;
313 BigInt m_b;
314 BigInt m_g_x;
315 BigInt m_g_y;
316 BigInt m_order;
317 BigInt m_cofactor;
318
319#if defined(BOTAN_HAS_LEGACY_EC_POINT)
320 CurveGFp m_curve;
321 EC_Point m_base_point;
322
323 Barrett_Reduction m_mod_field;
324 Barrett_Reduction m_mod_order;
325
326 // Montgomery parameters (only used for legacy EC_Point)
327 Montgomery_Params m_monty;
328
329 BigInt m_a_r; // (a*r) % p
330 BigInt m_b_r; // (b*r) % p
331#endif
332
333 OID m_oid;
334 std::vector<uint8_t> m_der_named_curve;
335 size_t m_p_words;
336 size_t m_p_bits;
337 size_t m_order_bits;
338 size_t m_order_bytes;
339 bool m_a_is_minus_3;
340 bool m_a_is_zero;
341 bool m_has_cofactor;
342 bool m_order_is_less_than_p;
343 EC_Group_Source m_source;
344 EC_Group_Engine m_engine;
345};
346
347} // namespace Botan
348
349#endif
#define BOTAN_ASSERT_NONNULL(ptr)
Definition assert.h:114
virtual void serialize_compressed_to(std::span< uint8_t > bytes) const =0
virtual ~EC_AffinePoint_Data()=default
virtual std::unique_ptr< EC_AffinePoint_Data > mul(const EC_Scalar_Data &scalar, RandomNumberGenerator &rng) const =0
virtual secure_vector< uint8_t > mul_x_only(const EC_Scalar_Data &scalar, RandomNumberGenerator &rng) const =0
virtual void serialize_y_to(std::span< uint8_t > bytes) const =0
virtual std::unique_ptr< EC_AffinePoint_Data > clone() const =0
virtual size_t field_element_bytes() const =0
virtual void serialize_uncompressed_to(std::span< uint8_t > bytes) const =0
virtual void serialize_x_to(std::span< uint8_t > bytes) const =0
virtual const std::shared_ptr< const EC_Group_Data > & group() const =0
virtual bool is_identity() const =0
virtual void serialize_xy_to(std::span< uint8_t > bytes) const =0
std::unique_ptr< EC_Scalar_Data > gk_x_mod_order(const EC_Scalar_Data &scalar, RandomNumberGenerator &rng) const
const BigInt & p() const
const BigInt & g_x() const
std::unique_ptr< EC_AffinePoint_Data > affine_neg(const EC_AffinePoint_Data &p) const
const BigInt & a() const
std::unique_ptr< EC_Scalar_Data > scalar_from_bytes_mod_order(std::span< const uint8_t > bytes) const
std::unique_ptr< EC_AffinePoint_Data > point_deserialize(std::span< const uint8_t > bytes) const
bool a_is_minus_3() const
std::unique_ptr< EC_Scalar_Data > scalar_random(RandomNumberGenerator &rng) const
bool has_cofactor() const
std::unique_ptr< EC_Scalar_Data > scalar_deserialize(std::span< const uint8_t > bytes) const
bool params_match(const BigInt &p, const BigInt &a, const BigInt &b, const BigInt &g_x, const BigInt &g_y, const BigInt &order, const BigInt &cofactor) const
const PCurve::PrimeOrderCurve & pcurve() const
static std::shared_ptr< EC_Group_Data > create(const BigInt &p, const BigInt &a, const BigInt &b, const BigInt &g_x, const BigInt &g_y, const BigInt &order, const BigInt &cofactor, const OID &oid, EC_Group_Source source)
EC_Group_Data & operator=(EC_Group_Data &&other)=delete
std::unique_ptr< EC_AffinePoint_Data > affine_add(const EC_AffinePoint_Data &p, const EC_AffinePoint_Data &q) const
std::unique_ptr< EC_Scalar_Data > scalar_from_bytes_with_trunc(std::span< const uint8_t > bytes) const
std::unique_ptr< EC_Mul2Table_Data > make_mul2_table(const EC_AffinePoint_Data &pt) const
std::unique_ptr< EC_AffinePoint_Data > mul_px_qy(const EC_AffinePoint_Data &p, const EC_Scalar_Data &x, const EC_AffinePoint_Data &q, const EC_Scalar_Data &y, RandomNumberGenerator &rng) const
std::unique_ptr< EC_Scalar_Data > scalar_one() const
const OID & oid() const
EC_Group_Data & operator=(const EC_Group_Data &other)=delete
std::unique_ptr< EC_AffinePoint_Data > point_g_mul(const EC_Scalar_Data &scalar, RandomNumberGenerator &rng) const
size_t order_bits() const
const BigInt & cofactor() const
const std::vector< uint8_t > & der_named_curve() const
bool order_is_less_than_p() const
EC_Group_Data(EC_Group_Data &&other)=delete
EC_Group_Data(const EC_Group_Data &other)=delete
size_t p_words() const
size_t p_bytes() const
EC_Group_Source source() const
void set_oid(const OID &oid)
EC_Group_Engine engine() const
const BigInt & g_y() const
size_t order_bytes() const
std::unique_ptr< EC_Scalar_Data > scalar_from_bigint(const BigInt &bn) const
std::unique_ptr< EC_AffinePoint_Data > point_hash_to_curve_ro(std::string_view hash_fn, std::span< const uint8_t > input, std::span< const uint8_t > domain_sep) const
size_t p_bits() const
const BigInt & b() const
std::unique_ptr< EC_AffinePoint_Data > point_hash_to_curve_nu(std::string_view hash_fn, std::span< const uint8_t > input, std::span< const uint8_t > domain_sep) const
const BigInt & order() const
virtual ~EC_Mul2Table_Data()=default
virtual bool mul2_vartime_x_mod_order_eq(const EC_Scalar_Data &v, const EC_Scalar_Data &x, const EC_Scalar_Data &y) const =0
virtual std::unique_ptr< EC_AffinePoint_Data > mul2_vartime(const EC_Scalar_Data &x, const EC_Scalar_Data &y) const =0
virtual void assign(const EC_Scalar_Data &y)=0
virtual const std::shared_ptr< const EC_Group_Data > & group() const =0
virtual std::unique_ptr< EC_Scalar_Data > invert_vartime() const =0
virtual void zeroize()=0
virtual size_t bytes() const =0
virtual std::unique_ptr< EC_Scalar_Data > sub(const EC_Scalar_Data &other) const =0
virtual std::unique_ptr< EC_Scalar_Data > invert() const =0
virtual void serialize_to(std::span< uint8_t > bytes) const =0
virtual std::unique_ptr< EC_Scalar_Data > negate() const =0
virtual std::unique_ptr< EC_Scalar_Data > clone() const =0
virtual std::unique_ptr< EC_Scalar_Data > add(const EC_Scalar_Data &other) const =0
virtual void square_self()=0
virtual bool is_eq(const EC_Scalar_Data &y) const =0
virtual bool is_zero() const =0
virtual ~EC_Scalar_Data()=default
virtual std::unique_ptr< EC_Scalar_Data > mul(const EC_Scalar_Data &other) const =0
EC_Group_Engine
Definition ec_group.h:48
std::vector< T, secure_allocator< T > > secure_vector
Definition secmem.h:68
EC_Group_Source
Definition ec_group.h:38