9#include <botan/internal/ascon_aead128.h>
11#include <botan/exceptn.h>
12#include <botan/mem_ops.h>
13#include <botan/internal/concat_util.h>
14#include <botan/internal/loadstor.h>
20constexpr void xor2x64(std::span<uint64_t, 2> lhs, std::span<const uint64_t, 2> rhs) {
26constexpr auto as_array_of_uint64(std::span<const uint8_t> in) {
32constexpr Ascon_p initial_state_of_ascon_aead_permutation({
33 .init_and_final_rounds = 12,
34 .processing_rounds = 8,
40constexpr uint64_t ascon_aead_128_iv = 0x00001000808c0001;
43constexpr uint64_t ascon_aead_128_domain_sep = 0x8000000000000000;
56 m_ascon_p = initial_state_of_ascon_aead_permutation;
63 m_key = as_array_of_uint64<2>(key);
67 BOTAN_ARG_CHECK(idx == 0,
"Ascon-AEAD128: cannot handle non-zero index in set_associated_data_n");
68 m_ad.assign(ad.begin(), ad.end());
77 m_ascon_p.state() =
concat(std::array{ascon_aead_128_iv}, *
m_key, as_array_of_uint64<2>({nonce, nonce_len}));
93 m_ascon_p.state()[4] ^= ascon_aead_128_domain_sep;
112size_t Ascon_AEAD128_Encryption::process_msg(uint8_t buf[],
size_t size) {
124 const auto final_block_at_offset = std::span{final_block}.subspan(offset);
125 process_msg(final_block_at_offset.data(), final_block_at_offset.size());
127 final_block.insert(final_block.end(), tag.begin(), tag.end());
130size_t Ascon_AEAD128_Decryption::process_msg(uint8_t buf[],
size_t size) {
142 const auto final_block_at_offset = std::span{final_block}.subspan(offset);
144 const auto final_ciphertext_block = final_block_at_offset.first(final_block_at_offset.size() -
tag_size());
145 const auto expected_tag = final_block_at_offset.last(
tag_size());
147 process_msg(final_ciphertext_block.data(), final_ciphertext_block.size());
149 clear_mem(std::span{final_block}.subspan(offset, final_ciphertext_block.size()));
150 throw Invalid_Authentication_Tag(
"Ascon-AEAD128 tag check failed");
153 final_block.resize(offset + final_ciphertext_block.size());
#define BOTAN_DEBUG_ASSERT(expr)
#define BOTAN_STATE_CHECK(expr)
#define BOTAN_ARG_CHECK(expr, msg)
bool has_keying_material() const final
void key_schedule(std::span< const uint8_t > key) final
std::optional< std::array< uint64_t, 2 > > m_key
void maybe_absorb_associated_data()
void set_associated_data_n(size_t idx, std::span< const uint8_t > ad) final
size_t tag_size() const final
bool valid_nonce_length(size_t n) const final
std::array< uint8_t, 16 > calculate_tag_and_finish()
void start_msg(const uint8_t nonce[], size_t nonce_len) final
void percolate_in(std::span< uint8_t > data)
constexpr auto store_le(ParamTs &&... params)
constexpr auto concat(Rs &&... ranges)
constexpr auto load_le(ParamTs &&... params)
std::vector< T, secure_allocator< T > > secure_vector
bool constant_time_compare(std::span< const uint8_t > x, std::span< const uint8_t > y)
constexpr void clear_mem(T *ptr, size_t n)