-
Notifications
You must be signed in to change notification settings - Fork 408
/
Copy pathcryptoalgs.hpp
403 lines (350 loc) · 10.2 KB
/
cryptoalgs.hpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
// OpenVPN -- An application to securely tunnel IP networks
// over a single port, with support for SSL/TLS-based
// session authentication and key exchange,
// packet encryption, packet authentication, and
// packet compression.
//
// Copyright (C) 2012- OpenVPN Inc.
//
// SPDX-License-Identifier: MPL-2.0 OR AGPL-3.0-only WITH openvpn3-openssl-exception
//
// Crypto algorithms
#ifndef OPENVPN_CRYPTO_CRYPTOALGS_H
#define OPENVPN_CRYPTO_CRYPTOALGS_H
#include <functional>
#include <string>
#include <list>
#include <openvpn/common/size.hpp>
#include <openvpn/common/exception.hpp>
#include <openvpn/common/string.hpp>
#include <openvpn/common/likely.hpp>
#include <openvpn/common/arraysize.hpp>
#include <openvpn/crypto/definitions.hpp>
namespace openvpn::CryptoAlgs {
OPENVPN_EXCEPTION(crypto_alg);
OPENVPN_SIMPLE_EXCEPTION(crypto_alg_index);
enum class KeyDerivation
{
UNDEFINED,
OPENVPN_PRF,
TLS_EKM
};
inline const char *name(const KeyDerivation kd)
{
switch (kd)
{
case KeyDerivation::UNDEFINED:
return "[PRF undefined]";
case KeyDerivation::OPENVPN_PRF:
return "OpenVPN PRF";
case KeyDerivation::TLS_EKM:
return "TLS Keying Material Exporter [RFC5705]";
default:
return "Unknown";
}
}
enum Type
{
NONE = 0,
// CBC ciphers
AES_128_CBC,
AES_192_CBC,
AES_256_CBC,
DES_CBC,
DES_EDE3_CBC,
BF_CBC,
// CTR ciphers
AES_256_CTR,
// AEAD ciphers
AES_128_GCM,
AES_192_GCM,
AES_256_GCM,
CHACHA20_POLY1305,
// digests
MD4,
MD5,
SHA1,
SHA224,
SHA256,
SHA384,
SHA512,
SIZE,
};
enum Mode
{
MODE_UNDEF = 0,
CBC_HMAC,
AEAD,
MODE_MASK = 0x03,
};
enum AlgFlags
{ // bits below must start after Mode bits
F_CIPHER = (1 << 2), // alg is a cipher
F_DIGEST = (1 << 3), // alg is a digest
F_ALLOW_DC = (1 << 4) // alg may be used in OpenVPN data channel
};
// size in bytes of AEAD "nonce tail" normally taken from
// HMAC key material
enum
{
AEAD_NONCE_TAIL_SIZE = 8
};
class Alg
{
public:
constexpr Alg(const char *name,
const unsigned int flags,
const unsigned int size,
const unsigned int iv_length,
const unsigned int block_size,
uint64_t aead_usage_limit)
: name_(name),
flags_(flags),
size_(size),
iv_length_(iv_length),
block_size_(block_size),
aead_usage_limit_(aead_usage_limit)
{
}
const char *name() const
{
return name_;
}
unsigned int flags() const
{
return flags_;
} // contains Mode and AlgFlags
Mode mode() const
{
return Mode(flags_ & MODE_MASK);
}
size_t size() const
{
return size_;
} // digest size
size_t key_length() const
{
return size_;
} // cipher key length
size_t iv_length() const
{
return iv_length_;
} // cipher only
size_t block_size() const
{
return block_size_;
} // cipher only
bool dc_cipher() const
{
return (flags_ & F_CIPHER) && (flags_ & F_ALLOW_DC);
}
bool dc_digest() const
{
return (flags_ & F_DIGEST) && (flags_ & F_ALLOW_DC);
}
void allow_dc(bool allow)
{
if (allow)
flags_ |= F_ALLOW_DC;
else
flags_ &= ~F_ALLOW_DC;
}
/** Returns the number q + s of total invocations + plain text blocks that should not be
* exceeded */
uint64_t aead_usage_limit() const
{
return aead_usage_limit_;
}
private:
const char *name_;
unsigned int flags_;
unsigned int size_;
unsigned int iv_length_;
unsigned int block_size_;
uint64_t aead_usage_limit_;
};
/** The limit for AES-GCM ciphers according to https://datatracker.ietf.org/doc/draft-irtf-cfrg-aead-limits/ */
static constexpr uint64_t gcm_limit = (1ull << 36) - 1;
inline std::array<Alg, Type::SIZE> algs = {
// clang-format off
Alg{"none", F_CIPHER|F_DIGEST|CBC_HMAC, 0, 0, 0, 0 },
Alg{"AES-128-CBC", F_CIPHER|CBC_HMAC, 16, 16, 16, 0 },
Alg{"AES-192-CBC", F_CIPHER|CBC_HMAC, 24, 16, 16, 0 },
Alg{"AES-256-CBC", F_CIPHER|CBC_HMAC, 32, 16, 16, 0 },
Alg{"DES-CBC", F_CIPHER|CBC_HMAC, 8, 8, 8, 0 },
Alg{"DES-EDE3-CBC", F_CIPHER|CBC_HMAC, 24, 8, 8, 0 },
Alg{"BF-CBC", F_CIPHER|CBC_HMAC, 16, 8, 8, 0 },
Alg{"AES-256-CTR", F_CIPHER, 32, 16, 16, 0 },
Alg{"AES-128-GCM", F_CIPHER|AEAD, 16, 12, 16, gcm_limit },
Alg{"AES-192-GCM", F_CIPHER|AEAD, 24, 12, 16, gcm_limit },
Alg{"AES-256-GCM", F_CIPHER|AEAD, 32, 12, 16, gcm_limit },
Alg{"CHACHA20-POLY1305", F_CIPHER|AEAD, 32, 12, 16, 0 },
Alg{"MD4", F_DIGEST, 16, 0, 0, 0 },
Alg{"MD5", F_DIGEST, 16, 0, 0, 0 },
Alg{"SHA1", F_DIGEST, 20, 0, 0, 0 },
Alg{"SHA224", F_DIGEST, 28, 0, 0, 0 },
Alg{"SHA256", F_DIGEST, 32, 0, 0, 0 },
Alg{"SHA384", F_DIGEST, 48, 0, 0, 0 },
Alg{"SHA512", F_DIGEST, 64, 0, 0, 0 }
// clang-format on
};
inline bool defined(const Type type)
{
return type != NONE;
}
inline const Alg &get_index(const size_t i)
{
if (unlikely(i >= algs.size()))
throw crypto_alg_index();
return algs[i];
}
inline const Alg *get_ptr(const Type type)
{
const Alg &alg_ref = get_index(static_cast<size_t>(type));
return &alg_ref;
}
inline const Alg &get(const Type type)
{
return get_index(static_cast<size_t>(type));
}
inline std::size_t for_each(std::function<bool(Type, const Alg &)> fn)
{
std::size_t count = 0;
for (std::size_t i = 0; i < algs.size(); ++i)
if (fn(static_cast<Type>(i), algs[i]))
count++;
return count;
}
inline Type lookup(const std::string &name)
{
for (size_t i = 0; i < algs.size(); ++i)
{
if (string::strcasecmp(name, algs[i].name()) == 0)
return static_cast<Type>(i);
}
OPENVPN_THROW(crypto_alg, name << ": not found");
}
inline const char *name(const Type type, const char *default_name = nullptr)
{
if (type == NONE && default_name)
return default_name;
else
return get(type).name();
}
inline size_t size(const Type type)
{
const Alg &alg = get(type);
return alg.size();
}
inline size_t key_length(const Type type)
{
const Alg &alg = get(type);
return alg.key_length();
}
inline size_t iv_length(const Type type)
{
const Alg &alg = get(type);
return alg.iv_length();
}
inline size_t block_size(const Type type)
{
const Alg &alg = get(type);
return alg.block_size();
}
inline Mode mode(const Type type)
{
const Alg &alg = get(type);
return alg.mode();
}
inline uint64_t aead_usage_limit(const Type type)
{
const Alg &alg = get(type);
return alg.aead_usage_limit();
}
inline Type legal_dc_cipher(const Type type)
{
const Alg &alg = get(type);
if (!alg.dc_cipher())
OPENVPN_THROW(crypto_alg, alg.name() << ": bad cipher for data channel use");
return type;
}
inline Type legal_dc_digest(const Type type)
{
const Alg &alg = get(type);
if (!alg.dc_digest())
OPENVPN_THROW(crypto_alg, alg.name() << ": bad digest for data channel use");
return type;
}
inline Type dc_cbc_cipher(const Type type)
{
const Alg &alg = get(type);
if (!(alg.flags() & CBC_HMAC))
OPENVPN_THROW(crypto_alg, alg.name() << ": bad cipher for data channel use");
return type;
}
inline Type dc_cbc_hash(const Type type)
{
const Alg &alg = get(type);
if (!(alg.flags() & F_DIGEST))
OPENVPN_THROW(crypto_alg, alg.name() << ": bad digest for data channel use");
return type;
}
inline void allow_dc_algs(const std::list<Type> types)
{
for (auto &alg : algs)
alg.allow_dc(false);
for (auto &type : types)
algs.at(type).allow_dc(true);
}
/**
* Allows the default algorithms but only those which are available with
* the library context.
* @param libctx Library context to use
* @param preferred Allow only the preferred algorithms, also disabling
* legacy (only AEAD)
* @param legacy Allow also legacy algorithm that are vulnerable to SWEET32
* no effect if preferred is true
*/
template <typename CRYPTO_API>
inline void allow_default_dc_algs(SSLLib::Ctx libctx, bool preferred = false, bool legacy = false)
{
/* Disable all and reenable the ones actually allowed later */
for (auto &alg : algs)
alg.allow_dc(false);
CryptoAlgs::for_each([preferred, libctx, legacy](CryptoAlgs::Type type, const CryptoAlgs::Alg &alg) -> bool
{
/* Defined in the algorithm but not actually related to data channel */
if (type == MD4 || type == AES_256_CTR)
return false;
if (preferred && alg.mode() != AEAD)
return false;
if (alg.mode() == AEAD && !CRYPTO_API::CipherContextAEAD::is_supported(libctx, type))
return false;
/* 64 bit block ciphers vulnerable to SWEET32 */
if (alg.flags() & F_CIPHER && !legacy && alg.block_size() <= 8)
return false;
/* This excludes MD4 */
if (alg.flags() & F_DIGEST && !legacy && alg.size() < 20)
return false;
if ((alg.flags() & F_CIPHER && alg.mode() != AEAD && type != NONE)
&& !CRYPTO_API::CipherContext::is_supported(libctx, type))
return false;
/* This algorithm has passed all checks, enable it for DC */
algs.at(type).allow_dc(true);
return true; });
}
/**
* Check if a specific algorithm depends on an additional digest or not
*
* @param type CryptoAlgs::Type to check
*
* @return Returns true if the queried algorithm depends on a digest,
* otherwise false.
*/
inline bool use_cipher_digest(const Type type)
{
const Alg &alg = get(type);
return alg.mode() != AEAD;
}
} // namespace openvpn::CryptoAlgs
#endif