11#include <botan/internal/kex_to_kem_adapter.h>
13#include <botan/internal/fmt.h>
14#include <botan/internal/pk_ops_impl.h>
15#include <botan/internal/stl_util.h>
17#if defined(BOTAN_HAS_DIFFIE_HELLMAN)
19 #include <botan/dl_group.h>
22#if defined(BOTAN_HAS_ECDH)
23 #include <botan/ecdh.h>
26#if defined(BOTAN_HAS_X25519)
27 #include <botan/x25519.h>
30#if defined(BOTAN_HAS_X448)
31 #include <botan/x448.h>
48size_t kex_shared_key_length(
const Public_Key& kex_public_key) {
51#if defined(BOTAN_HAS_ECDH)
52 if(
const auto* ecdh =
dynamic_cast<const ECDH_PublicKey*
>(&kex_public_key)) {
53 return ecdh->domain().get_p_bytes();
57#if defined(BOTAN_HAS_DIFFIE_HELLMAN)
58 if(
const auto* dh =
dynamic_cast<const DH_PublicKey*
>(&kex_public_key)) {
59 return dh->group().p_bytes();
63#if defined(BOTAN_HAS_X25519)
64 if(
const auto* curve =
dynamic_cast<const X25519_PublicKey*
>(&kex_public_key)) {
70#if defined(BOTAN_HAS_X448)
71 if(
const auto* curve =
dynamic_cast<const X448_PublicKey*
>(&kex_public_key)) {
77 throw Not_Implemented(
78 fmt(
"Cannot get shared kex key length from unknown key agreement public key of type '{}' in the hybrid KEM key",
79 kex_public_key.algo_name()));
86std::unique_ptr<PK_Key_Agreement_Key> generate_key_agreement_private_key(
const Public_Key& kex_public_key,
87 RandomNumberGenerator& rng) {
90 auto new_kex_key = [&] {
91 auto new_private_key = kex_public_key.generate_another(rng);
92 const auto kex_key =
dynamic_cast<PK_Key_Agreement_Key*
>(new_private_key.get());
93 if(kex_key) [[likely]] {
97 (void)new_private_key.release();
99 return std::unique_ptr<PK_Key_Agreement_Key>(kex_key);
102 BOTAN_ASSERT(new_kex_key,
"Keys wrapped in this adapter are always key-agreement keys");
106std::unique_ptr<Public_Key> maybe_get_public_key(
const std::unique_ptr<PK_Key_Agreement_Key>& private_key) {
108 return private_key->public_key();
111class KEX_to_KEM_Adapter_Encryption_Operation
final :
public PK_Ops::KEM_Encryption_with_KDF {
113 KEX_to_KEM_Adapter_Encryption_Operation(
const Public_Key& key, std::string_view kdf, std::string_view provider) :
114 PK_Ops::KEM_Encryption_with_KDF(kdf), m_provider(provider), m_public_key(key) {}
116 size_t raw_kem_shared_key_length()
const override {
return kex_shared_key_length(m_public_key); }
118 size_t encapsulated_key_length()
const override {
124 return m_public_key.raw_public_key_bits().size();
127 void raw_kem_encrypt(std::span<uint8_t> out_encapsulated_key,
128 std::span<uint8_t> raw_shared_key,
130 const auto sk = generate_key_agreement_private_key(m_public_key, rng);
131 const auto shared_key = PK_Key_Agreement(*sk, rng,
"Raw", m_provider)
132 .derive_key(0 , m_public_key.raw_public_key_bits())
135 const auto public_value = sk->public_value();
141 out_encapsulated_key.size(),
142 "KEX-to-KEM Adapter: encapsulated key out-param has correct length");
144 shared_key.size(), raw_shared_key.size(),
"KEX-to-KEM Adapter: shared key out-param has correct length");
145 std::copy(public_value.begin(), public_value.end(), out_encapsulated_key.begin());
146 std::copy(shared_key.begin(), shared_key.end(), raw_shared_key.begin());
150 std::string m_provider;
151 const Public_Key& m_public_key;
154class KEX_to_KEM_Decryption_Operation
final :
public PK_Ops::KEM_Decryption_with_KDF {
156 KEX_to_KEM_Decryption_Operation(
const PK_Key_Agreement_Key& key,
157 RandomNumberGenerator& rng,
158 const std::string_view kdf,
159 const std::string_view provider) :
160 PK_Ops::KEM_Decryption_with_KDF(kdf),
161 m_operation(key, rng,
"Raw", provider),
162 m_encapsulated_key_length(key.public_value().size()) {}
164 void raw_kem_decrypt(std::span<uint8_t> out_shared_key, std::span<const uint8_t> encap_key)
override {
167 shared_secret.size(), out_shared_key.size(),
"KEX-to-KEM Adapter: shared key out-param has correct length");
168 std::copy(shared_secret.begin(), shared_secret.end(), out_shared_key.begin());
171 size_t encapsulated_key_length()
const override {
return m_encapsulated_key_length; }
173 size_t raw_kem_shared_key_length()
const override {
return m_operation.agreed_value_size(); }
176 PK_Key_Agreement m_operation;
177 size_t m_encapsulated_key_length;
183 m_public_key(std::move(public_key)) {
189 return fmt(
"KEX-to-KEM({})", m_public_key->algo_name());
193 return m_public_key->estimated_strength();
197 return m_public_key->key_length();
201 return m_public_key->check_key(rng, strong);
205 return m_public_key->algorithm_identifier();
209 return m_public_key->raw_public_key_bits();
213 throw Not_Implemented(
"The KEX-to-KEM adapter does not support ASN.1-based public key serialization");
217 return std::make_unique<KEX_to_KEM_Adapter_PrivateKey>(generate_key_agreement_private_key(*m_public_key, rng));
230 return m_private_key->private_key_bits();
234 return std::make_unique<KEX_to_KEM_Adapter_PublicKey>(m_private_key->public_key());
238 return m_private_key->check_key(rng, strong);
242 std::string_view kdf, std::string_view provider)
const {
243 return std::make_unique<KEX_to_KEM_Adapter_Encryption_Operation>(*m_public_key, kdf, provider);
248 return std::make_unique<KEX_to_KEM_Decryption_Operation>(*m_private_key, rng, kdf, provider);
#define BOTAN_ASSERT_NOMSG(expr)
#define BOTAN_ASSERT_EQUAL(expr1, expr2, assertion_made)
#define BOTAN_ARG_CHECK(expr, msg)
#define BOTAN_ASSERT(expr, assertion_made)
KEX_to_KEM_Adapter_PrivateKey(std::unique_ptr< PK_Key_Agreement_Key > private_key)
std::unique_ptr< PK_Ops::KEM_Decryption > create_kem_decryption_op(RandomNumberGenerator &rng, std::string_view kdf, std::string_view provider="base") const override
bool check_key(RandomNumberGenerator &rng, bool strong) const override
secure_vector< uint8_t > private_key_bits() const override
std::unique_ptr< Public_Key > public_key() const override
bool supports_operation(PublicKeyOperation op) const override
size_t estimated_strength() const override
std::unique_ptr< PK_Ops::KEM_Encryption > create_kem_encryption_op(std::string_view kdf, std::string_view provider="base") const override
bool check_key(RandomNumberGenerator &rng, bool strong) const override
std::vector< uint8_t > public_key_bits() const override
AlgorithmIdentifier algorithm_identifier() const override
size_t key_length() const override
std::unique_ptr< Private_Key > generate_another(RandomNumberGenerator &rng) const final
KEX_to_KEM_Adapter_PublicKey(std::unique_ptr< Public_Key > public_key)
std::string algo_name() const override
std::vector< uint8_t > raw_public_key_bits() const override
int(* final)(unsigned char *, CTX *)
std::string fmt(std::string_view format, const T &... args)
std::vector< T, secure_allocator< T > > secure_vector