Botan 3.11.0
Crypto and TLS for C&
fpe_fe1.cpp
Go to the documentation of this file.
1/*
2* Format Preserving Encryption (FE1 scheme)
3* (C) 2009,2018 Jack Lloyd
4*
5* Botan is released under the Simplified BSD License (see license.txt)
6*/
7
8#include <botan/fpe_fe1.h>
9
10#include <botan/exceptn.h>
11#include <botan/mac.h>
12#include <botan/numthry.h>
13#include <botan/internal/divide.h>
14#include <botan/internal/fmt.h>
15#include <botan/internal/loadstor.h>
16
17namespace Botan {
18
19namespace {
20
21// Normally FPE is for SSNs, CC#s, etc, nothing too big
22const size_t MAX_N_BYTES = 128 / 8;
23
24/*
25* Factor n into a and b which are as close together as possible.
26* Assumes n is composed mostly of small factors which is the case for
27* typical uses of FPE (typically, n is a power of 10)
28*/
29void factor(BigInt n, BigInt& a, BigInt& b) {
30 BOTAN_ARG_CHECK(n >= 2, "Invalid FPE modulus");
31
32 a = BigInt::one();
33 b = BigInt::one();
34
35 /*
36 * This algorithm was poorly designed. It should have fully factored n (to the
37 * extent possible) and then built a/b starting from the largest factor first.
38 *
39 * This can't be fixed now without breaking existing users but if some
40 * incompatible change (or new flag, etc) is added in the future, consider
41 * fixing the factoring for those users.
42 */
43
44 const size_t n_low_zero = low_zero_bits(n);
45
46 a <<= (n_low_zero / 2);
47 b <<= n_low_zero - (n_low_zero / 2);
48 n >>= n_low_zero;
49
50 for(size_t i = 0; i != PRIME_TABLE_SIZE; ++i) {
51 while(n % PRIMES[i] == 0) {
52 a *= PRIMES[i];
53 if(a > b) {
54 std::swap(a, b);
55 }
57 }
58 }
59
60 if(a > b) {
61 std::swap(a, b);
62 }
63 a *= n;
64
65 if(a <= 1 || b <= 1) {
66 throw Internal_Error("Could not factor n for use in FPE");
67 }
68}
69
70} // namespace
71
72FPE_FE1::FPE_FE1(const BigInt& n, size_t rounds, bool compat_mode, std::string_view mac_algo) : m_rounds(rounds) {
73 if(m_rounds < 3) {
74 throw Invalid_Argument("FPE_FE1 rounds too small");
75 }
76
78
79 m_n_bytes = n.serialize();
80
81 if(m_n_bytes.size() > MAX_N_BYTES) {
82 throw Invalid_Argument("N is too large for FPE encryption");
83 }
84
85 factor(n, m_a, m_b);
86
87 if(compat_mode) {
88 if(m_a < m_b) {
89 std::swap(m_a, m_b);
90 }
91 } else {
92 if(m_a > m_b) {
93 std::swap(m_a, m_b);
94 }
95 }
96}
97
98FPE_FE1::FPE_FE1(FPE_FE1&& other) noexcept = default;
99
100FPE_FE1::~FPE_FE1() = default;
101
103 m_mac->clear();
104}
105
106std::string FPE_FE1::name() const {
107 return fmt("FPE_FE1({},{})", m_mac->name(), m_rounds);
108}
109
111 return m_mac->key_spec();
112}
113
115 return m_mac->has_keying_material();
116}
117
118void FPE_FE1::key_schedule(std::span<const uint8_t> key) {
119 m_mac->set_key(key);
120}
121
122BigInt FPE_FE1::F(const BigInt& R,
123 size_t round,
124 const secure_vector<uint8_t>& tweak_mac,
125 secure_vector<uint8_t>& tmp) const {
126 tmp = R.serialize<secure_vector<uint8_t>>();
127
128 m_mac->update(tweak_mac);
129 m_mac->update_be(static_cast<uint32_t>(round));
130
131 m_mac->update_be(static_cast<uint32_t>(tmp.size()));
132 m_mac->update(tmp.data(), tmp.size());
133
134 tmp = m_mac->final();
135 return BigInt::from_bytes(tmp);
136}
137
138secure_vector<uint8_t> FPE_FE1::compute_tweak_mac(const uint8_t tweak[], size_t tweak_len) const {
139 m_mac->update_be(static_cast<uint32_t>(m_n_bytes.size()));
140 m_mac->update(m_n_bytes.data(), m_n_bytes.size());
141
142 m_mac->update_be(static_cast<uint32_t>(tweak_len));
143 if(tweak_len > 0) {
144 m_mac->update(tweak, tweak_len);
145 }
146
147 return m_mac->final();
148}
149
150BigInt FPE_FE1::encrypt(const BigInt& input, const uint8_t tweak[], size_t tweak_len) const {
151 const secure_vector<uint8_t> tweak_mac = compute_tweak_mac(tweak, tweak_len);
152
153 BigInt X = input;
154
156
157 BigInt L;
158 BigInt R;
159 BigInt Fi;
160 for(size_t i = 0; i != m_rounds; ++i) {
161 ct_divide(X, m_b, L, R);
162 Fi = F(R, i, tweak_mac, tmp);
163 X = m_a * R + ct_modulo(L + Fi, m_a);
164 }
165
166 return X;
167}
168
169BigInt FPE_FE1::decrypt(const BigInt& input, const uint8_t tweak[], size_t tweak_len) const {
170 const secure_vector<uint8_t> tweak_mac = compute_tweak_mac(tweak, tweak_len);
171
172 BigInt X = input;
174
175 BigInt W;
176 BigInt R;
177 BigInt Fi;
178 for(size_t i = 0; i != m_rounds; ++i) {
179 ct_divide(X, m_a, R, W);
180
181 Fi = F(R, m_rounds - i - 1, tweak_mac, tmp);
182 X = m_b * ct_modulo(W - Fi, m_a) + R;
183 }
184
185 return X;
186}
187
188BigInt FPE_FE1::encrypt(const BigInt& x, uint64_t tweak) const {
189 uint8_t tweak8[8];
190 store_be(tweak, tweak8);
191 return encrypt(x, tweak8, sizeof(tweak8));
192}
193
194BigInt FPE_FE1::decrypt(const BigInt& x, uint64_t tweak) const {
195 uint8_t tweak8[8];
196 store_be(tweak, tweak8);
197 return decrypt(x, tweak8, sizeof(tweak8));
198}
199
200namespace FPE {
201
202BigInt fe1_encrypt(const BigInt& n, const BigInt& X, const SymmetricKey& key, const std::vector<uint8_t>& tweak) {
203 FPE_FE1 fpe(n, 3, true, "HMAC(SHA-256)");
204 fpe.set_key(key);
205 return fpe.encrypt(X, tweak.data(), tweak.size());
206}
207
208BigInt fe1_decrypt(const BigInt& n, const BigInt& X, const SymmetricKey& key, const std::vector<uint8_t>& tweak) {
209 FPE_FE1 fpe(n, 3, true, "HMAC(SHA-256)");
210 fpe.set_key(key);
211 return fpe.decrypt(X, tweak.data(), tweak.size());
212}
213
214} // namespace FPE
215
216} // namespace Botan
#define BOTAN_ARG_CHECK(expr, msg)
Definition assert.h:33
const word * data() const
Definition bigint.h:662
static BigInt one()
Definition bigint.h:55
static BigInt from_bytes(std::span< const uint8_t > bytes)
Definition bigint.cpp:83
static BigInt from_word(word n)
Definition bigint.cpp:35
T serialize(size_t len) const
Definition bigint.h:727
BigInt encrypt(const BigInt &x, const uint8_t tweak[], size_t tweak_len) const
Definition fpe_fe1.cpp:150
void clear() override
Definition fpe_fe1.cpp:102
std::string name() const override
Definition fpe_fe1.cpp:106
BOTAN_FUTURE_EXPLICIT FPE_FE1(const BigInt &n, size_t rounds=5, bool compat_mode=false, std::string_view mac_algo="HMAC(SHA-256)")
Definition fpe_fe1.cpp:72
bool has_keying_material() const override
Definition fpe_fe1.cpp:114
~FPE_FE1() override
Key_Length_Specification key_spec() const override
Definition fpe_fe1.cpp:110
BigInt decrypt(const BigInt &x, const uint8_t tweak[], size_t tweak_len) const
Definition fpe_fe1.cpp:169
static std::unique_ptr< MessageAuthenticationCode > create_or_throw(std::string_view algo_spec, std::string_view provider="")
Definition mac.cpp:147
void set_key(const OctetString &key)
Definition sym_algo.cpp:14
BigInt fe1_decrypt(const BigInt &n, const BigInt &X, const SymmetricKey &key, const std::vector< uint8_t > &tweak)
Definition fpe_fe1.cpp:208
BigInt fe1_encrypt(const BigInt &n, const BigInt &X, const SymmetricKey &key, const std::vector< uint8_t > &tweak)
Definition fpe_fe1.cpp:202
OctetString SymmetricKey
Definition symkey.h:140
std::string fmt(std::string_view format, const T &... args)
Definition fmt.h:53
const uint16_t PRIMES[]
Definition primes.cpp:12
size_t low_zero_bits(const BigInt &n)
Definition numthry.cpp:194
const size_t PRIME_TABLE_SIZE
Definition numthry.h:175
BigInt ct_modulo(const BigInt &x, const BigInt &y)
Definition divide.cpp:192
void ct_divide(const BigInt &x, const BigInt &y, BigInt &q_out, BigInt &r_out)
Definition divide.cpp:51
std::vector< T, secure_allocator< T > > secure_vector
Definition secmem.h:68
constexpr auto store_be(ParamTs &&... params)
Definition loadstor.h:745