diff options
author | lloyd <[email protected]> | 2013-03-16 18:48:30 +0000 |
---|---|---|
committer | lloyd <[email protected]> | 2013-03-16 18:48:30 +0000 |
commit | e139778872b8e6be9725f8540a6dab846e93e657 (patch) | |
tree | 4fbb6bc7b8f4fa850cbd6b8946eebea148349cac /src/filters | |
parent | acef62145872b262d42f51e02db2fbdb53e64e1c (diff) |
Save most of the block cipher calls used when setting the nonce in OCB
by avoiding recomputing stretch if the previous value works. This
greatly reduces the cost of using OCB when only the lowest bits of the
nonce value are changing (as when using a counter).
Diffstat (limited to 'src/filters')
-rw-r--r-- | src/filters/modes/ocb/ocb.cpp | 92 | ||||
-rw-r--r-- | src/filters/modes/ocb/ocb.h | 4 |
2 files changed, 59 insertions, 37 deletions
diff --git a/src/filters/modes/ocb/ocb.cpp b/src/filters/modes/ocb/ocb.cpp index 5750e3da2..24f7d3d9c 100644 --- a/src/filters/modes/ocb/ocb.cpp +++ b/src/filters/modes/ocb/ocb.cpp @@ -52,18 +52,67 @@ class L_computer mutable std::vector<secure_vector<byte>> m_L; }; -#if 0 class Nonce_State { public: - secure_vector<byte> update_nonce(const byte nonce[], size_t nonce_len); + Nonce_State(const BlockCipher& cipher) : m_cipher(cipher) {} + + secure_vector<byte> update_nonce(const byte nonce[], + size_t nonce_len); bool fresh_nonce() { bool b = false; std::swap(b, m_fresh); return b; } private: + const BlockCipher& m_cipher; + secure_vector<byte> m_last_nonce; secure_vector<byte> m_stretch; bool m_fresh = false; }; -#endif + +secure_vector<byte> +Nonce_State::update_nonce(const byte nonce[], size_t nonce_len) + { + const size_t BS = 16; + + BOTAN_ASSERT(nonce_len < BS, "Nonce is less than 128 bits"); + + secure_vector<byte> nonce_buf(BS); + + copy_mem(&nonce_buf[BS - nonce_len], nonce, nonce_len); + nonce_buf[BS - nonce_len - 1] = 1; + + const byte bottom = nonce_buf[15] & 0x3F; + nonce_buf[15] &= 0xC0; + + const bool need_new_stretch = (m_last_nonce != nonce_buf); + + if(need_new_stretch) + { + m_last_nonce = nonce_buf; + + m_cipher.encrypt(nonce_buf); + + for(size_t i = 0; i != 8; ++i) + nonce_buf.push_back(nonce_buf[i] ^ nonce_buf[i+1]); + + m_stretch = nonce_buf; + } + + // now set the offset from stretch and bottom + + const size_t shift_bytes = bottom / 8; + const size_t shift_bits = bottom % 8; + + secure_vector<byte> offset(BS); + for(size_t i = 0; i != BS; ++i) + { + offset[i] = (m_stretch[i+shift_bytes] << shift_bits); + offset[i] |= (m_stretch[i+shift_bytes+1] >> (8-shift_bits)); + } + + m_fresh = true; + return offset; + } + namespace { @@ -146,6 +195,7 @@ void OCB_Mode::set_key(const SymmetricKey& key) { m_cipher->set_key(key); m_L.reset(new L_computer(*m_cipher)); + m_nonce_state.reset(new Nonce_State(*m_cipher)); } void OCB_Mode::set_nonce(const byte nonce[], size_t nonce_len) @@ -153,44 +203,14 @@ void OCB_Mode::set_nonce(const byte nonce[], size_t nonce_len) if(!valid_iv_length(nonce_len)) throw Invalid_IV_Length(name(), nonce_len); - byte bottom; - secure_vector<byte> stretch; - - if(1) // need to recompute stretch (save iv to compare) - { - secure_vector<byte> buf(BS); - - const size_t offset = BS - nonce_len; - - copy_mem(&buf[offset], nonce, nonce_len); - buf[offset-1] = 1; + BOTAN_ASSERT(m_nonce_state, "A key was set"); - bottom = buf[15] & 0x3F; - buf[15] &= 0xC0; - - m_cipher->encrypt(buf); - - for(size_t i = 0; i != 8; ++i) - buf.push_back(buf[i] ^ buf[i+1]); - - stretch = buf; - } - - // now set the offset from stretch and bottom - - const size_t shift_bytes = bottom / 8; - const size_t shift_bits = bottom % 8; - - for(size_t i = 0; i != BS; ++i) - { - m_offset[i] = (stretch[i+shift_bytes] << shift_bits); - m_offset[i] |= (stretch[i+shift_bytes+1] >> (8-shift_bits)); - } + m_offset = m_nonce_state->update_nonce(nonce, nonce_len); } void OCB_Mode::start_msg() { - //BOTAN_ASSERT(m_nonce_state.fresh_nonce(), "Nonce state is fresh"); + BOTAN_ASSERT(m_nonce_state->fresh_nonce(), "Nonce state is fresh"); } void OCB_Mode::set_associated_data(const byte ad[], size_t ad_len) diff --git a/src/filters/modes/ocb/ocb.h b/src/filters/modes/ocb/ocb.h index 5497b08a4..5e3c5cf0a 100644 --- a/src/filters/modes/ocb/ocb.h +++ b/src/filters/modes/ocb/ocb.h @@ -16,7 +16,7 @@ namespace Botan { class L_computer; -//class Nonce_State; +class Nonce_State; /** * OCB Mode (base class for OCB_Encryption and OCB_Decryption). Note @@ -73,6 +73,8 @@ class BOTAN_DLL OCB_Mode : public AEAD_Mode, void write(const byte input[], size_t input_length) override; void start_msg() override; void end_msg() override; + + std::unique_ptr<Nonce_State> m_nonce_state; }; class BOTAN_DLL OCB_Encryption : public OCB_Mode |