10#include <botan/ber_dec.h>
11#include <botan/der_enc.h>
12#include <botan/ec_group.h>
13#include <botan/hash.h>
15#include <botan/mem_ops.h>
16#include <botan/pk_ops.h>
17#include <botan/internal/ct_utils.h>
18#include <botan/internal/fmt.h>
27 m_group(key.domain()), m_peer(key._public_ec_point()) {
32 size_t max_input_bits()
const override {
37 size_t ciphertext_length(
size_t ptext_len)
const override {
38 const size_t elem_size = m_group.get_order_bytes();
39 const size_t der_overhead = 16;
41 return der_overhead + 2 * elem_size + m_hash->output_length() + ptext_len;
44 std::vector<uint8_t>
encrypt(std::span<const uint8_t> msg, RandomNumberGenerator& rng)
override {
49 const EC_AffinePoint kPB = m_peer.mul(k, rng);
51 const auto x2_bytes = kPB.x_bytes();
52 const auto y2_bytes = kPB.y_bytes();
55 kdf_input += x2_bytes;
56 kdf_input += y2_bytes;
58 const auto kdf_output = m_kdf->derive_key(msg.size(), kdf_input);
60 std::vector<uint8_t> masked_msg(msg.size());
61 xor_buf(masked_msg, msg, kdf_output);
63 m_hash->update(x2_bytes);
65 m_hash->update(y2_bytes);
66 const auto C3 = m_hash->final<std::vector<uint8_t>>();
68 std::vector<uint8_t> ctext;
71 .encode(BigInt(C1.x_bytes()))
72 .encode(BigInt(C1.y_bytes()))
81 const EC_Group m_group;
82 const EC_AffinePoint m_peer;
83 std::unique_ptr<HashFunction> m_hash;
84 std::unique_ptr<KDF> m_kdf;
90 RandomNumberGenerator& rng,
91 std::string_view kdf_hash) :
92 m_group(key.domain()), m_x(key._private_key()), m_rng(rng) {
95 const std::string kdf_name =
fmt(
"KDF2({})", kdf_hash);
99 size_t plaintext_length(
size_t ptext_len)
const override {
104 const size_t elem_size = m_group.get_order_bytes();
106 if(ptext_len < 2 * elem_size + m_hash->output_length()) {
110 return ptext_len - (2 * elem_size + m_hash->output_length());
114 const size_t p_bytes = m_group.get_p_bytes();
119 if(ctext.size() < 1 + p_bytes * 2 + m_hash->output_length()) {
138 if(C3.size() != m_hash->output_length()) {
142 std::vector<uint8_t> recode_ctext;
143 DER_Encoder(recode_ctext)
162 const auto dbC1 = C1->mul(m_x, m_rng);
163 const auto x2_bytes = dbC1.x_bytes();
164 const auto y2_bytes = dbC1.y_bytes();
166 const auto kdf_output = m_kdf->derive_key(masked_msg.size(), dbC1.xy_bytes());
168 xor_buf(masked_msg.data(), kdf_output.data(), kdf_output.size());
170 m_hash->update(x2_bytes);
171 m_hash->update(masked_msg);
172 m_hash->update(y2_bytes);
173 const auto u = m_hash->final();
184 const EC_Group m_group;
186 RandomNumberGenerator& m_rng;
187 std::unique_ptr<HashFunction> m_hash;
188 std::unique_ptr<KDF> m_kdf;
194 std::string_view params,
195 std::string_view provider)
const {
198 if(provider ==
"base" || provider.empty()) {
200 return std::make_unique<SM2_Encryption_Operation>(*
this,
"SM3");
202 return std::make_unique<SM2_Encryption_Operation>(*
this, params);
210 std::string_view params,
211 std::string_view provider)
const {
212 if(provider ==
"base" || provider.empty()) {
214 return std::make_unique<SM2_Decryption_Operation>(*
this, rng,
"SM3");
216 return std::make_unique<SM2_Decryption_Operation>(*
this, rng, params);
static std::optional< EC_AffinePoint > from_bigint_xy(const EC_Group &group, const BigInt &x, const BigInt &y)
static EC_AffinePoint g_mul(const EC_Scalar &scalar, RandomNumberGenerator &rng)
Multiply by the group generator returning a complete point.
static EC_Scalar random(const EC_Group &group, RandomNumberGenerator &rng)
static std::unique_ptr< HashFunction > create_or_throw(std::string_view algo_spec, std::string_view provider="")
static std::unique_ptr< KDF > create_or_throw(std::string_view algo_spec, std::string_view provider="")
std::unique_ptr< PK_Ops::Decryption > create_decryption_op(RandomNumberGenerator &rng, std::string_view params, std::string_view provider) const override
std::unique_ptr< PK_Ops::Encryption > create_encryption_op(RandomNumberGenerator &rng, std::string_view params, std::string_view provider) const override
std::string algo_name() const override
constexpr CT::Mask< T > is_equal(const T x[], const T y[], size_t len)
std::string decrypt(std::span< const uint8_t > input, std::string_view passphrase)
std::string encrypt(const uint8_t input[], size_t input_len, std::string_view passphrase, RandomNumberGenerator &rng)
SM2_PublicKey SM2_Encryption_PublicKey
std::string fmt(std::string_view format, const T &... args)
SM2_PrivateKey SM2_Encryption_PrivateKey
constexpr void xor_buf(ranges::contiguous_output_range< uint8_t > auto &&out, ranges::contiguous_range< uint8_t > auto &&in)
std::vector< T, secure_allocator< T > > secure_vector