Botan 3.12.0
Crypto and TLS for C&
ghash.cpp
Go to the documentation of this file.
1/*
2* GCM GHASH
3* (C) 2013,2015,2017 Jack Lloyd
4* (C) 2016 Daniel Neus, Rohde & Schwarz Cybersecurity
5* (C) 2024 René Meusel, Rohde & Schwarz Cybersecurity
6*
7* Botan is released under the Simplified BSD License (see license.txt)
8*/
9
10#include <botan/internal/ghash.h>
11
12#include <botan/exceptn.h>
13#include <botan/internal/ct_utils.h>
14#include <botan/internal/loadstor.h>
15
16#if defined(BOTAN_HAS_CPUID)
17 #include <botan/internal/cpuid.h>
18#endif
19
20namespace Botan {
21
22std::string GHASH::provider() const {
23#if defined(BOTAN_HAS_GHASH_AVX512_CLMUL)
25 return *feat;
26 }
27#endif
28
29#if defined(BOTAN_HAS_GHASH_CLMUL_CPU)
30 if(auto feat = CPUID::check(CPUID::Feature::HW_CLMUL)) {
31 return *feat;
32 }
33#endif
34
35#if defined(BOTAN_HAS_GHASH_CLMUL_VPERM)
37 return *feat;
38 }
39#endif
40
41 return "base";
42}
43
44void GHASH::ghash_multiply(std::span<uint8_t, GCM_BS> x, std::span<const uint8_t> input, size_t blocks) {
45 BOTAN_ASSERT_NOMSG(input.size() % GCM_BS == 0);
46
47#if defined(BOTAN_HAS_GHASH_AVX512_CLMUL)
49 BOTAN_ASSERT_NOMSG(!m_H_pow.empty());
50 return ghash_multiply_avx512_clmul(x.data(), m_H_pow.data(), input.data(), blocks);
51 }
52#endif
53
54#if defined(BOTAN_HAS_GHASH_CLMUL_CPU)
56 BOTAN_ASSERT_NOMSG(!m_H_pow.empty());
57 return ghash_multiply_cpu(x.data(), m_H_pow, input.data(), blocks);
58 }
59#endif
60
61#if defined(BOTAN_HAS_GHASH_CLMUL_VPERM)
63 return ghash_multiply_vperm(x.data(), m_HM.data(), input.data(), blocks);
64 }
65#endif
66
67 auto scope = CT::scoped_poison(x);
68
70
71 BufferSlicer in(input);
72 for(size_t b = 0; b != blocks; ++b) {
73 const auto I = load_be<std::array<uint64_t, 2>>(in.take<GCM_BS>());
74 X[0] ^= I[0];
75 X[1] ^= I[1];
76
77 std::array<uint64_t, 2> Z{};
78
79 for(size_t i = 0; i != 64; ++i) {
80 const auto X0MASK = CT::Mask<uint64_t>::expand_top_bit(X[0]);
81 const auto X1MASK = CT::Mask<uint64_t>::expand_top_bit(X[1]);
82
83 X[0] <<= 1;
84 X[1] <<= 1;
85
86 Z[0] = X0MASK.select(Z[0] ^ m_HM[4 * i], Z[0]);
87 Z[1] = X0MASK.select(Z[1] ^ m_HM[4 * i + 1], Z[1]);
88
89 Z[0] = X1MASK.select(Z[0] ^ m_HM[4 * i + 2], Z[0]);
90 Z[1] = X1MASK.select(Z[1] ^ m_HM[4 * i + 3], Z[1]);
91 }
92
93 X[0] = Z[0];
94 X[1] = Z[1];
95 }
96
97 store_be(x, X);
98}
99
101 return !m_HM.empty() || !m_H_pow.empty();
102}
103
104void GHASH::key_schedule(std::span<const uint8_t> key) {
105 m_H_ad = {0};
106 m_ad_len = 0;
107 m_text_len = 0;
108
109 BOTAN_ASSERT_NOMSG(key.size() == GCM_BS);
110 auto H = load_be<std::array<uint64_t, 2>>(key.first<GCM_BS>());
111
112#if defined(BOTAN_HAS_GHASH_AVX512_CLMUL)
114 zap(m_HM);
115 if(m_H_pow.size() != 32) {
116 m_H_pow.resize(32);
117 }
118 ghash_precompute_avx512_clmul(key.data(), m_H_pow.data());
119 // m_HM left empty
120 return;
121 }
122#endif
123
124#if defined(BOTAN_HAS_GHASH_CLMUL_CPU)
126 zap(m_HM);
127 ghash_precompute_cpu(key.data(), m_H_pow);
128 // m_HM left empty
129 return;
130 }
131#endif
132
133 const uint64_t R = 0xE100000000000000;
134
135 if(m_HM.size() != 256) {
136 m_HM.resize(256);
137 }
138
139 // precompute the multiples of H
140 for(size_t i = 0; i != 2; ++i) {
141 for(size_t j = 0; j != 64; ++j) {
142 /*
143 we interleave H^1, H^65, H^2, H^66, H3, H67, H4, H68
144 to make indexing nicer in the multiplication code
145 */
146 m_HM[4 * j + 2 * i] = H[0];
147 m_HM[4 * j + 2 * i + 1] = H[1];
148
149 // GCM's bit ops are reversed so we carry out of the bottom
150 const uint64_t carry = CT::Mask<uint64_t>::expand(H[1] & 1).if_set_return(R);
151 H[1] = (H[1] >> 1) | (H[0] << 63);
152 H[0] = (H[0] >> 1) ^ carry;
153 }
154 }
155}
156
157void GHASH::start(std::span<const uint8_t> nonce) {
158 BOTAN_ARG_CHECK(nonce.size() == 16, "GHASH requires a 128-bit nonce");
159 auto& n = m_nonce.emplace();
160 copy_mem(n, nonce);
161 copy_mem(m_ghash, m_H_ad);
162 m_buffer.clear();
163 m_text_len = 0;
164}
165
166void GHASH::set_associated_data(std::span<const uint8_t> input) {
167 BOTAN_STATE_CHECK(!m_nonce);
168
170 m_H_ad = {0};
171 ghash_update(m_H_ad, input);
172 ghash_zeropad(m_H_ad);
173 m_ad_len = input.size();
174}
175
177 // This should only be called in GMAC context
178 BOTAN_STATE_CHECK(m_text_len == 0);
180 m_H_ad = {0};
181 m_ad_len = 0;
182}
183
184void GHASH::update_associated_data(std::span<const uint8_t> ad) {
186 ghash_update(m_ghash, ad);
187 m_ad_len += ad.size();
188}
189
190void GHASH::update(std::span<const uint8_t> input) {
192 BOTAN_STATE_CHECK(m_nonce);
193 ghash_update(m_ghash, input);
194 m_text_len += input.size();
195
196 // NIST SP 800-38D limits plaintext/ciphertext to 2^39 - 256 bits
197 constexpr uint64_t GHASH_MAX_BYTES = (((static_cast<uint64_t>(1) << 39)) - 256) / 8;
198 if(m_text_len > GHASH_MAX_BYTES) {
199 throw Invalid_State("GCM message length limit exceeded");
200 }
201}
202
203void GHASH::final(std::span<uint8_t> mac) {
204 BOTAN_ARG_CHECK(!mac.empty() && mac.size() <= GCM_BS, "GHASH output length");
205 BOTAN_STATE_CHECK(m_nonce);
207
208 ghash_zeropad(m_ghash);
209 ghash_final_block(m_ghash, m_ad_len, m_text_len);
210
211 xor_buf(mac, std::span{m_ghash}.first(mac.size()), std::span{*m_nonce}.first(mac.size()));
212
213 secure_scrub_memory(m_ghash);
214 m_text_len = 0;
215 m_nonce.reset();
216}
217
218void GHASH::nonce_hash(std::span<uint8_t, GCM_BS> y0, std::span<const uint8_t> nonce) {
220 BOTAN_STATE_CHECK(!m_nonce);
221
222 ghash_update(y0, nonce);
223 ghash_zeropad(y0);
224 ghash_final_block(y0, 0, nonce.size());
225}
226
228 zap(m_HM);
229 zap(m_H_pow);
230 this->reset_state();
231}
232
234 m_H_ad = {0};
235 secure_scrub_memory(m_ghash);
236 if(m_nonce) {
237 secure_scrub_memory(m_nonce.value());
238 m_nonce.reset();
239 }
240 m_buffer.clear();
241 m_text_len = 0;
242 m_ad_len = 0;
243}
244
245void GHASH::ghash_update(std::span<uint8_t, GCM_BS> x, std::span<const uint8_t> input) {
246 BufferSlicer in(input);
247 while(!in.empty()) {
248 if(const auto one_block = m_buffer.handle_unaligned_data(in)) {
249 ghash_multiply(x, one_block.value(), 1);
250 }
251
252 if(m_buffer.in_alignment()) {
253 const auto [aligned_data, full_blocks] = m_buffer.aligned_data_to_process(in);
254 if(full_blocks > 0) {
255 ghash_multiply(x, aligned_data, full_blocks);
256 }
257 }
258 }
259 BOTAN_ASSERT_NOMSG(in.empty());
260}
261
262void GHASH::ghash_zeropad(std::span<uint8_t, GCM_BS> x) {
263 if(!m_buffer.in_alignment()) {
264 m_buffer.fill_up_with_zeros();
265 ghash_multiply(x, m_buffer.consume(), 1);
266 }
267}
268
269void GHASH::ghash_final_block(std::span<uint8_t, GCM_BS> x, uint64_t ad_len, uint64_t text_len) {
270 BOTAN_STATE_CHECK(m_buffer.in_alignment());
271 const auto final_block = store_be(8 * ad_len, 8 * text_len);
272 ghash_multiply(x, final_block, 1);
273}
274
275} // namespace Botan
#define BOTAN_ASSERT_NOMSG(expr)
Definition assert.h:75
#define BOTAN_STATE_CHECK(expr)
Definition assert.h:49
#define BOTAN_ARG_CHECK(expr, msg)
Definition assert.h:33
std::tuple< std::span< const uint8_t >, size_t > aligned_data_to_process(BufferSlicer &slicer) const
std::optional< std::span< const T > > handle_unaligned_data(BufferSlicer &slicer)
static std::optional< std::string > check(CPUID::Feature feat)
Definition cpuid.h:67
static bool has(CPUID::Feature feat)
Definition cpuid.h:94
static constexpr Mask< T > expand(T v)
Definition ct_utils.h:392
static constexpr Mask< T > expand_top_bit(T v)
Definition ct_utils.h:415
void update_associated_data(std::span< const uint8_t > ad)
Incremental update of associated data used in the GMAC use-case.
Definition ghash.cpp:184
std::string provider() const
Definition ghash.cpp:22
void final(std::span< uint8_t > out)
Definition ghash.cpp:203
void nonce_hash(std::span< uint8_t, GCM_BS > y0, std::span< const uint8_t > nonce)
Hashing of non-default length nonce values for both GCM and GMAC use-cases.
Definition ghash.cpp:218
void reset_associated_data()
Reset the AAD state without resetting the key (used in GMAC::final_result).
Definition ghash.cpp:176
void reset_state()
Definition ghash.cpp:233
void clear() override
Definition ghash.cpp:227
void update(std::span< const uint8_t > in)
Definition ghash.cpp:190
void start(std::span< const uint8_t > nonce)
Definition ghash.cpp:157
bool has_keying_material() const override
Definition ghash.cpp:100
void set_associated_data(std::span< const uint8_t > ad)
Monolithic setting of associated data usid in the GCM use-case.
Definition ghash.cpp:166
void assert_key_material_set() const
Definition sym_algo.h:145
constexpr auto scoped_poison(const Ts &... xs)
Definition ct_utils.h:222
void zap(std::vector< T, Alloc > &vec)
Definition secmem.h:157
constexpr void copy_mem(T *out, const T *in, size_t n)
Definition mem_ops.h:144
void secure_scrub_memory(void *ptr, size_t n)
Definition mem_utils.cpp:25
void carry(int64_t &h0, int64_t &h1)
constexpr void xor_buf(ranges::contiguous_output_range< uint8_t > auto &&out, ranges::contiguous_range< uint8_t > auto &&in)
Definition mem_ops.h:341
constexpr auto store_be(ParamTs &&... params)
Definition loadstor.h:745
constexpr auto load_be(ParamTs &&... params)
Definition loadstor.h:504