mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-10 01:48:00 +00:00
90 lines
3.1 KiB
C
90 lines
3.1 KiB
C
|
/*
|
||
|
* Top-level vtables to select an AES implementation.
|
||
|
*/
|
||
|
|
||
|
#include <assert.h>
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
#include "putty.h"
|
||
|
#include "ssh.h"
|
||
|
#include "aes.h"
|
||
|
|
||
|
static ssh_cipher *aes_select(const ssh_cipheralg *alg)
|
||
|
{
|
||
|
const ssh_cipheralg *const *real_algs = (const ssh_cipheralg **)alg->extra;
|
||
|
|
||
|
for (size_t i = 0; real_algs[i]; i++) {
|
||
|
const ssh_cipheralg *alg = real_algs[i];
|
||
|
const struct aes_extra *alg_extra =
|
||
|
(const struct aes_extra *)alg->extra;
|
||
|
if (check_availability(alg_extra))
|
||
|
return ssh_cipher_new(alg);
|
||
|
}
|
||
|
|
||
|
/* We should never reach the NULL at the end of the list, because
|
||
|
* the last non-NULL entry should be software-only AES, which is
|
||
|
* always available. */
|
||
|
unreachable("aes_select ran off the end of its list");
|
||
|
}
|
||
|
|
||
|
#if HAVE_AES_NI
|
||
|
#define IF_NI(...) __VA_ARGS__
|
||
|
#else
|
||
|
#define IF_NI(...)
|
||
|
#endif
|
||
|
|
||
|
#if HAVE_NEON_CRYPTO
|
||
|
#define IF_NEON(...) __VA_ARGS__
|
||
|
#else
|
||
|
#define IF_NEON(...)
|
||
|
#endif
|
||
|
|
||
|
#define AES_SELECTOR_VTABLE(mode_c, mode_protocol, mode_display, bits) \
|
||
|
static const ssh_cipheralg * \
|
||
|
ssh_aes ## bits ## _ ## mode_c ## _impls[] = { \
|
||
|
IF_NI(&ssh_aes ## bits ## _ ## mode_c ## _ni,) \
|
||
|
IF_NEON(&ssh_aes ## bits ## _ ## mode_c ## _neon,) \
|
||
|
&ssh_aes ## bits ## _ ## mode_c ## _sw, \
|
||
|
NULL, \
|
||
|
}; \
|
||
|
const ssh_cipheralg ssh_aes ## bits ## _ ## mode_c = { \
|
||
|
.new = aes_select, \
|
||
|
.ssh2_id = "aes" #bits "-" mode_protocol, \
|
||
|
.blksize = 16, \
|
||
|
.real_keybits = bits, \
|
||
|
.padded_keybytes = bits/8, \
|
||
|
.text_name = "AES-" #bits " " mode_display \
|
||
|
" (dummy selector vtable)", \
|
||
|
.extra = ssh_aes ## bits ## _ ## mode_c ## _impls, \
|
||
|
}
|
||
|
|
||
|
AES_SELECTOR_VTABLE(cbc, "cbc", "CBC", 128);
|
||
|
AES_SELECTOR_VTABLE(cbc, "cbc", "CBC", 192);
|
||
|
AES_SELECTOR_VTABLE(cbc, "cbc", "CBC", 256);
|
||
|
AES_SELECTOR_VTABLE(sdctr, "ctr", "SDCTR", 128);
|
||
|
AES_SELECTOR_VTABLE(sdctr, "ctr", "SDCTR", 192);
|
||
|
AES_SELECTOR_VTABLE(sdctr, "ctr", "SDCTR", 256);
|
||
|
|
||
|
static const ssh_cipheralg ssh_rijndael_lysator = {
|
||
|
/* Same as aes256_cbc, but with a different protocol ID */
|
||
|
.new = aes_select,
|
||
|
.ssh2_id = "rijndael-cbc@lysator.liu.se",
|
||
|
.blksize = 16,
|
||
|
.real_keybits = 256,
|
||
|
.padded_keybytes = 256/8,
|
||
|
.text_name = "AES-256 CBC (dummy selector vtable)",
|
||
|
.extra = ssh_aes256_cbc_impls,
|
||
|
};
|
||
|
|
||
|
static const ssh_cipheralg *const aes_list[] = {
|
||
|
&ssh_aes256_sdctr,
|
||
|
&ssh_aes256_cbc,
|
||
|
&ssh_rijndael_lysator,
|
||
|
&ssh_aes192_sdctr,
|
||
|
&ssh_aes192_cbc,
|
||
|
&ssh_aes128_sdctr,
|
||
|
&ssh_aes128_cbc,
|
||
|
};
|
||
|
|
||
|
const ssh2_ciphers ssh2_aes = { lenof(aes_list), aes_list };
|