7#include <botan/internal/ec_inner_data.h>
9#include <botan/der_enc.h>
10#include <botan/internal/ec_inner_pc.h>
11#include <botan/internal/fmt.h>
12#include <botan/internal/pcurves.h>
15#if defined(BOTAN_HAS_LEGACY_EC_POINT)
16 #include <botan/internal/ec_inner_bn.h>
17 #include <botan/internal/point_mul.h>
20#if defined(BOTAN_HAS_XMD)
21 #include <botan/internal/xmd.h>
48 m_monty(m_p, m_mod_field),
51 m_p_words(
p.sig_words()),
53 m_order_bits(
order.bits()),
54 m_order_bytes((m_order_bits + 7) / 8),
55 m_a_is_minus_3(
a ==
p - 3),
56 m_a_is_zero(
a.is_zero()),
57 m_has_cofactor(m_cofactor != 1),
58 m_order_is_less_than_p(m_order <
p),
62 DER_Encoder der(m_der_named_curve);
65 const std::string name = m_oid.human_name_or_empty();
68 m_pcurve = PCurve::PrimeOrderCurve::for_named_curve(name);
71 m_engine = EC_Group_Engine::Optimized;
76 if(!m_pcurve && !m_has_cofactor) {
85#if defined(BOTAN_HAS_LEGACY_EC_POINT)
87 m_a_r = m_monty.mul(a, m_monty.R2(), ws);
88 m_b_r = m_monty.mul(b, m_monty.R2(), ws);
90 m_engine = EC_Group_Engine::Legacy;
95 throw Not_Implemented(
"EC_Group this group is not supported in this build configuration");
97 throw Not_Implemented(
98 fmt(
"EC_Group the group {} is not supported in this build configuration", oid.to_string()));
115#if defined(BOTAN_HAS_LEGACY_EC_POINT)
116 group->m_curve =
CurveGFp(group.get());
118 if(!group->m_pcurve) {
119 group->m_base_mult = std::make_unique<EC_Point_Base_Point_Precompute>(group->m_base_point, group->m_mod_order);
142 if(
order != this->order()) {
148 if(
g_x != this->g_x()) {
151 if(
g_y != this->g_y()) {
161 std::span<const uint8_t> base_pt,
173 if(
order != this->order()) {
180 const size_t field_len = this->
p_bytes();
182 if(base_pt.size() == 1 + field_len && (base_pt[0] == 0x02 || base_pt[0] == 0x03)) {
185 const auto g_x = m_g_x.serialize(field_len);
186 const auto g_y = m_g_y.is_odd();
188 const auto sec1_x = base_pt.subspan(1, field_len);
189 const bool sec1_y = (base_pt[0] == 0x03);
191 if(!std::ranges::equal(sec1_x,
g_x)) {
200 }
else if(base_pt.size() == 1 + 2 * field_len && base_pt[0] == 0x04) {
201 const auto g_x = m_g_x.serialize(field_len);
202 const auto g_y = m_g_y.serialize(field_len);
204 const auto sec1_x = base_pt.subspan(1, field_len);
205 const auto sec1_y = base_pt.subspan(1 + field_len, field_len);
207 if(!std::ranges::equal(sec1_x,
g_x)) {
211 if(!std::ranges::equal(sec1_y,
g_y)) {
217 throw Decoding_Error(
"Invalid base point encoding in explicit group");
235 const size_t bit_length = 8 * bytes.size();
241 const size_t shift = bit_length -
order_bits();
243 const size_t new_length = bytes.size() - (shift / 8);
244 const size_t bit_shift = shift % 8;
250 std::vector<uint8_t> sbytes(new_length);
253 for(
size_t i = 0; i != new_length; ++i) {
254 const uint8_t w = bytes[i];
255 sbytes[i] = (w >> bit_shift) |
carry;
256 carry = w << (8 - bit_shift);
270 if(
auto s = m_pcurve->scalar_from_wide_bytes(bytes)) {
271 return std::make_unique<EC_Scalar_Data_PC>(shared_from_this(), std::move(*s));
276#if defined(BOTAN_HAS_LEGACY_EC_POINT)
277 return std::make_unique<EC_Scalar_Data_BN>(shared_from_this(), m_mod_order.reduce(
BigInt(bytes)));
279 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
286 return std::make_unique<EC_Scalar_Data_PC>(shared_from_this(), m_pcurve->random_scalar(rng));
288#if defined(BOTAN_HAS_LEGACY_EC_POINT)
289 return std::make_unique<EC_Scalar_Data_BN>(shared_from_this(),
292 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
299 return std::make_unique<EC_Scalar_Data_PC>(shared_from_this(), m_pcurve->scalar_one());
301#if defined(BOTAN_HAS_LEGACY_EC_POINT)
302 return std::make_unique<EC_Scalar_Data_BN>(shared_from_this(),
BigInt::one());
304 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
310 if(bn <= 0 || bn >= m_order) {
317#if defined(BOTAN_HAS_LEGACY_EC_POINT)
318 return std::make_unique<EC_Scalar_Data_BN>(shared_from_this(), bn);
320 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
329 auto gk_x_mod_order = m_pcurve->base_point_mul_x_mod_order(k.value(), rng);
330 return std::make_unique<EC_Scalar_Data_PC>(shared_from_this(),
gk_x_mod_order);
332#if defined(BOTAN_HAS_LEGACY_EC_POINT)
335 std::vector<BigInt> ws;
336 const auto pt = m_base_mult->mul(k.value(), rng, m_order, ws);
339 return std::make_unique<EC_Scalar_Data_BN>(shared_from_this(),
BigInt::zero());
341 return std::make_unique<EC_Scalar_Data_BN>(shared_from_this(), m_mod_order.reduce(pt.get_affine_x()));
344 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
350 if(bytes.size() != m_order_bytes) {
355 if(
auto s = m_pcurve->deserialize_scalar(bytes)) {
356 return std::make_unique<EC_Scalar_Data_PC>(shared_from_this(), *s);
361#if defined(BOTAN_HAS_LEGACY_EC_POINT)
364 if(r.
is_zero() || r >= m_order) {
368 return std::make_unique<EC_Scalar_Data_BN>(shared_from_this(), std::move(r));
370 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
378 if(bytes.size() >= 1 + 2 * 4 && (bytes[0] == 0x06 || bytes[0] == 0x07)) {
379 const bool hdr_y_is_even = bytes[0] == 0x06;
380 const bool y_is_even = (bytes.back() & 0x01) == 0;
382 if(hdr_y_is_even == y_is_even) {
383 std::vector<uint8_t> sec1(bytes.begin(), bytes.end());
391 if(
auto pt = m_pcurve->deserialize_point(bytes)) {
392 return std::make_unique<EC_AffinePoint_Data_PC>(shared_from_this(), std::move(*pt));
397#if defined(BOTAN_HAS_LEGACY_EC_POINT)
399 return std::make_unique<EC_AffinePoint_Data_BN>(shared_from_this(), std::move(pt));
401 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
411std::function<void(std::span<uint8_t>)> h2c_expand_message(std::string_view hash_fn,
412 std::span<const uint8_t> input,
413 std::span<const uint8_t> domain_sep) {
418 if(hash_fn.starts_with(
"SHAKE")) {
419 throw Not_Implemented(
"Hash to curve currently does not support expand_message_xof");
422 return [=](std::span<uint8_t> uniform_bytes) {
423#if defined(BOTAN_HAS_XMD)
424 expand_message_xmd(hash_fn, uniform_bytes, input, domain_sep);
426 BOTAN_UNUSED(hash_fn, uniform_bytes, input, domain_sep);
427 throw Not_Implemented(
"Hash to curve is not implemented due to XMD being disabled");
435 std::span<const uint8_t> input,
436 std::span<const uint8_t> domain_sep)
const {
438 auto pt = m_pcurve->hash_to_curve_ro(h2c_expand_message(hash_fn, input, domain_sep));
439 return std::make_unique<EC_AffinePoint_Data_PC>(shared_from_this(), m_pcurve->point_to_affine(pt));
441 throw Not_Implemented(
"Hash to curve is not implemented for this curve");
446 std::span<const uint8_t> input,
447 std::span<const uint8_t> domain_sep)
const {
449 auto pt = m_pcurve->hash_to_curve_nu(h2c_expand_message(hash_fn, input, domain_sep));
450 return std::make_unique<EC_AffinePoint_Data_PC>(shared_from_this(), std::move(pt));
452 throw Not_Implemented(
"Hash to curve is not implemented for this curve");
460 auto pt = m_pcurve->point_to_affine(m_pcurve->mul_by_g(k.value(), rng));
461 return std::make_unique<EC_AffinePoint_Data_PC>(shared_from_this(), std::move(pt));
463#if defined(BOTAN_HAS_LEGACY_EC_POINT)
464 const auto& group = scalar.
group();
468 std::vector<BigInt> ws;
469 auto pt = group->m_base_mult->mul(bn.value(), rng, m_order, ws);
470 return std::make_unique<EC_AffinePoint_Data_BN>(shared_from_this(), std::move(pt));
472 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
490 return std::make_unique<EC_AffinePoint_Data_PC>(shared_from_this(), m_pcurve->point_to_affine(*pt));
495#if defined(BOTAN_HAS_LEGACY_EC_POINT)
496 std::vector<BigInt> ws;
497 const auto& group =
p.group();
503 const auto order = group->order() * group->cofactor();
508 auto px_qy = px + qy;
510 if(!px_qy.is_zero()) {
511 px_qy.force_affine();
512 return std::make_unique<EC_AffinePoint_Data_BN>(shared_from_this(), std::move(px_qy));
517 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
528 return std::make_unique<EC_AffinePoint_Data_PC>(shared_from_this(), m_pcurve->point_to_affine(pt));
530#if defined(BOTAN_HAS_LEGACY_EC_POINT)
531 auto pt =
p.to_legacy_point() + q.to_legacy_point();
532 return std::make_unique<EC_AffinePoint_Data_BN>(shared_from_this(), std::move(pt));
534 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
542 return std::make_unique<EC_AffinePoint_Data_PC>(shared_from_this(), pt);
544#if defined(BOTAN_HAS_LEGACY_EC_POINT)
545 auto pt =
p.to_legacy_point();
547 return std::make_unique<EC_AffinePoint_Data_BN>(shared_from_this(), std::move(pt));
549 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
556 return std::make_unique<EC_Mul2Table_Data_PC>(h);
558#if defined(BOTAN_HAS_LEGACY_EC_POINT)
560 return std::make_unique<EC_Mul2Table_Data_BN>(g, h);
562 throw Not_Implemented(
"Legacy EC interfaces disabled in this build configuration");
#define BOTAN_STATE_CHECK(expr)
#define BOTAN_ARG_CHECK(expr, msg)
static BigInt random_integer(RandomNumberGenerator &rng, const BigInt &min, const BigInt &max)
T serialize(size_t len) const
DER_Encoder & encode(bool b)
static const EC_AffinePoint_Data_PC & checked_ref(const EC_AffinePoint_Data &data)
std::unique_ptr< EC_Scalar_Data > gk_x_mod_order(const EC_Scalar_Data &scalar, RandomNumberGenerator &rng) const
const BigInt & g_x() const
std::unique_ptr< EC_AffinePoint_Data > affine_neg(const EC_AffinePoint_Data &p) 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
std::unique_ptr< EC_Scalar_Data > scalar_random(RandomNumberGenerator &rng) 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
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)
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
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
EC_Group_Data(const EC_Group_Data &other)=delete
EC_Group_Source source() const
void set_oid(const OID &oid)
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
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
EC_Point mul(const BigInt &k, RandomNumberGenerator &rng, const BigInt &group_order, std::vector< BigInt > &ws) const
static const EC_Scalar_Data_BN & checked_ref(const EC_Scalar_Data &data)
static const EC_Scalar_Data_PC & checked_ref(const EC_Scalar_Data &data)
virtual const std::shared_ptr< const EC_Group_Data > & group() const =0
static std::shared_ptr< const PrimeOrderCurve > from_params(const BigInt &p, const BigInt &a, const BigInt &b, const BigInt &base_x, const BigInt &base_y, const BigInt &order)
#define BOTAN_HAS_LEGACY_EC_POINT
@ Generic
A generic implementation that handles many curves in one implementation.
void carry(int64_t &h0, int64_t &h1)
std::vector< T, secure_allocator< T > > secure_vector
EC_Point OS2ECP(std::span< const uint8_t > data, const CurveGFp &curve)