1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-09 17:38:00 +00:00

Refactoring to prepare for changes in the PPK format.

'bool old_fmt' in ppk_load_s has now given way to a numeric version
field, which will allow it to be set to 3 in future, instead of just 1
or 2. The ad-hoc integer variable 'cipher' is replaced with a pointer
to a small struct that mentions individual details like key lengths,
to aid parametrisation.

The old ssh2_ppk_derivekey is now a larger function that derives all
three of the key components used in the private-blob protection: not
just the cipher key, but the cipher IV and the MAC key as well. The
main part of it is a switch on the file-format version, which
currently has only one case (PPK v1 and v2 don't differ in the key
derivation), but gives me a handy place to drop in a new case in a
future commit, changing the derivation of all those things.

All the key material is stored end-to-end in a single strbuf, with
ptrlens pointing into it. That makes it easy to free all in one go
later.
This commit is contained in:
Simon Tatham 2021-02-18 20:26:10 +00:00
parent c61158aa34
commit 6f025c0b84

158
sshpubk.c
View File

@ -585,17 +585,54 @@ const ssh_keyalg *find_pubkey_alg(const char *name)
return find_pubkey_alg_len(ptrlen_from_asciz(name)); return find_pubkey_alg_len(ptrlen_from_asciz(name));
} }
static void ssh2_ppk_derivekey(ptrlen passphrase, uint8_t *key) struct ppk_cipher {
const char *name;
size_t blocklen, keylen, ivlen;
};
static const struct ppk_cipher ppk_cipher_none = { "none", 1, 0, 0 };
static const struct ppk_cipher ppk_cipher_aes256_cbc = { "aes256-cbc", 16, 32, 16 };
static void ssh2_ppk_derive_keys(
unsigned fmt_version, const struct ppk_cipher *ciphertype,
ptrlen passphrase, strbuf *storage, ptrlen *cipherkey, ptrlen *cipheriv,
ptrlen *mackey)
{ {
ssh_hash *h; size_t mac_keylen;
h = ssh_hash_new(&ssh_sha1);
put_uint32(h, 0); switch (fmt_version) {
put_datapl(h, passphrase); case 2:
ssh_hash_digest(h, key + 0); case 1: {
ssh_hash_reset(h); /* Counter-mode iteration to generate cipher key data. */
put_uint32(h, 1); for (unsigned ctr = 0; ctr * 20 < ciphertype->keylen; ctr++) {
put_datapl(h, passphrase); ssh_hash *h = ssh_hash_new(&ssh_sha1);
ssh_hash_final(h, key + 20); put_uint32(h, ctr);
put_datapl(h, passphrase);
ssh_hash_final(h, strbuf_append(storage, 20));
}
strbuf_shrink_to(storage, ciphertype->keylen);
/* In this version of the format, the CBC IV was always all 0. */
put_padding(storage, ciphertype->ivlen, 0);
/* Completely separate hash for the MAC key. */
ssh_hash *h = ssh_hash_new(&ssh_sha1);
mac_keylen = ssh_hash_alg(h)->hlen;
put_datapl(h, PTRLEN_LITERAL("putty-private-key-file-mac-key"));
put_datapl(h, passphrase);
ssh_hash_final(h, strbuf_append(storage, mac_keylen));
break;
}
default:
unreachable("bad format version in ssh2_ppk_derive_keys");
}
BinarySource src[1];
BinarySource_BARE_INIT_PL(src, ptrlen_from_strbuf(storage));
*cipherkey = get_data(src, ciphertype->keylen);
*cipheriv = get_data(src, ciphertype->ivlen);
*mackey = get_data(src, mac_keylen);
} }
static int userkey_parse_line_counter(const char *text) static int userkey_parse_line_counter(const char *text)
@ -608,24 +645,23 @@ static int userkey_parse_line_counter(const char *text)
return -1; return -1;
} }
static const unsigned char zero_iv[16] = {0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0};
ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase, ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase,
const char **errorstr) const char **errorstr)
{ {
char header[40], *b, *encryption, *comment, *mac; char header[40], *b, *encryption, *comment, *mac;
const ssh_keyalg *alg; const ssh_keyalg *alg;
ssh2_userkey *ret; ssh2_userkey *ret;
int cipher, cipherblk; strbuf *public_blob, *private_blob, *cipher_mac_keys_blob;
strbuf *public_blob, *private_blob; ptrlen cipherkey, cipheriv, mackey;
const struct ppk_cipher *ciphertype;
int i; int i;
bool is_mac, old_fmt; bool is_mac;
int passlen = passphrase ? strlen(passphrase) : 0; unsigned fmt_version;
const char *error = NULL; const char *error = NULL;
ret = NULL; /* return NULL for most errors */ ret = NULL; /* return NULL for most errors */
encryption = comment = mac = NULL; encryption = comment = mac = NULL;
public_blob = private_blob = NULL; public_blob = private_blob = cipher_mac_keys_blob = NULL;
/* Read the first header line which contains the key type. */ /* Read the first header line which contains the key type. */
if (!read_header(src, header)) { if (!read_header(src, header)) {
@ -633,11 +669,11 @@ ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase,
goto error; goto error;
} }
if (0 == strcmp(header, "PuTTY-User-Key-File-2")) { if (0 == strcmp(header, "PuTTY-User-Key-File-2")) {
old_fmt = false; fmt_version = 2;
} else if (0 == strcmp(header, "PuTTY-User-Key-File-1")) { } else if (0 == strcmp(header, "PuTTY-User-Key-File-1")) {
/* this is an old key file; warn and then continue */ /* this is an old key file; warn and then continue */
old_keyfile_warning(); old_keyfile_warning();
old_fmt = true; fmt_version = 1;
} else if (0 == strncmp(header, "PuTTY-User-Key-File-", 20)) { } else if (0 == strncmp(header, "PuTTY-User-Key-File-", 20)) {
/* this is a key file FROM THE FUTURE; refuse it, but with a /* this is a key file FROM THE FUTURE; refuse it, but with a
* more specific error message than the generic one below */ * more specific error message than the generic one below */
@ -664,11 +700,9 @@ ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase,
if ((encryption = read_body(src)) == NULL) if ((encryption = read_body(src)) == NULL)
goto error; goto error;
if (!strcmp(encryption, "aes256-cbc")) { if (!strcmp(encryption, "aes256-cbc")) {
cipher = 1; ciphertype = &ppk_cipher_aes256_cbc;
cipherblk = 16;
} else if (!strcmp(encryption, "none")) { } else if (!strcmp(encryption, "none")) {
cipher = 0; ciphertype = &ppk_cipher_none;
cipherblk = 1;
} else { } else {
goto error; goto error;
} }
@ -712,26 +746,25 @@ ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase,
if ((mac = read_body(src)) == NULL) if ((mac = read_body(src)) == NULL)
goto error; goto error;
is_mac = true; is_mac = true;
} else if (0 == strcmp(header, "Private-Hash") && old_fmt) { } else if (0 == strcmp(header, "Private-Hash") && fmt_version == 1) {
if ((mac = read_body(src)) == NULL) if ((mac = read_body(src)) == NULL)
goto error; goto error;
is_mac = false; is_mac = false;
} else } else
goto error; goto error;
cipher_mac_keys_blob = strbuf_new();
ssh2_ppk_derive_keys(fmt_version, ciphertype,
ptrlen_from_asciz(passphrase ? passphrase : ""),
cipher_mac_keys_blob, &cipherkey, &cipheriv, &mackey);
/* /*
* Decrypt the private blob. * Decrypt the private blob.
*/ */
if (cipher) { if (private_blob->len % ciphertype->blocklen)
unsigned char key[40]; goto error;
if (ciphertype == &ppk_cipher_aes256_cbc) {
if (!passphrase) aes256_decrypt_pubkey(cipherkey.ptr, cipheriv.ptr,
goto error;
if (private_blob->len % cipherblk)
goto error;
ssh2_ppk_derivekey(ptrlen_from_asciz(passphrase), key);
aes256_decrypt_pubkey(key, zero_iv,
private_blob->u, private_blob->len); private_blob->u, private_blob->len);
} }
@ -739,12 +772,14 @@ ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase,
* Verify the MAC. * Verify the MAC.
*/ */
{ {
char realmac[41];
unsigned char binary[20]; unsigned char binary[20];
char realmac[sizeof(binary) * 2 + 1];
strbuf *macdata; strbuf *macdata;
bool free_macdata; bool free_macdata;
if (old_fmt) { const ssh2_macalg *mac_alg = &ssh_hmac_sha1;
if (fmt_version == 1) {
/* MAC (or hash) only covers the private blob. */ /* MAC (or hash) only covers the private blob. */
macdata = private_blob; macdata = private_blob;
free_macdata = false; free_macdata = false;
@ -761,25 +796,14 @@ ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase,
} }
if (is_mac) { if (is_mac) {
ssh_hash *hash;
ssh2_mac *mac; ssh2_mac *mac;
unsigned char mackey[20];
char header[] = "putty-private-key-file-mac-key";
hash = ssh_hash_new(&ssh_sha1); mac = ssh2_mac_new(mac_alg, NULL);
put_data(hash, header, sizeof(header)-1); ssh2_mac_setkey(mac, mackey);
if (cipher && passphrase)
put_data(hash, passphrase, passlen);
ssh_hash_final(hash, mackey);
mac = ssh2_mac_new(&ssh_hmac_sha1, NULL);
ssh2_mac_setkey(mac, make_ptrlen(mackey, 20));
ssh2_mac_start(mac); ssh2_mac_start(mac);
put_data(mac, macdata->s, macdata->len); put_data(mac, macdata->s, macdata->len);
ssh2_mac_genresult(mac, binary); ssh2_mac_genresult(mac, binary);
ssh2_mac_free(mac); ssh2_mac_free(mac);
smemclr(mackey, sizeof(mackey));
} else { } else {
hash_simple(&ssh_sha1, ptrlen_from_strbuf(macdata), binary); hash_simple(&ssh_sha1, ptrlen_from_strbuf(macdata), binary);
} }
@ -787,13 +811,13 @@ ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase,
if (free_macdata) if (free_macdata)
strbuf_free(macdata); strbuf_free(macdata);
for (i = 0; i < 20; i++) for (i = 0; i < mac_alg->len; i++)
sprintf(realmac + 2 * i, "%02x", binary[i]); sprintf(realmac + 2 * i, "%02x", binary[i]);
if (strcmp(mac, realmac)) { if (strcmp(mac, realmac)) {
/* An incorrect MAC is an unconditional Error if the key is /* An incorrect MAC is an unconditional Error if the key is
* unencrypted. Otherwise, it means Wrong Passphrase. */ * unencrypted. Otherwise, it means Wrong Passphrase. */
if (cipher) { if (ciphertype->keylen != 0) {
error = "wrong passphrase"; error = "wrong passphrase";
ret = SSH2_WRONG_PASSPHRASE; ret = SSH2_WRONG_PASSPHRASE;
} else { } else {
@ -835,6 +859,8 @@ ssh2_userkey *ppk_load_s(BinarySource *src, const char *passphrase,
strbuf_free(public_blob); strbuf_free(public_blob);
if (private_blob) if (private_blob)
strbuf_free(private_blob); strbuf_free(private_blob);
if (cipher_mac_keys_blob)
strbuf_free(cipher_mac_keys_blob);
if (errorstr) if (errorstr)
*errorstr = error; *errorstr = error;
return ret; return ret;
@ -1260,12 +1286,14 @@ void base64_encode(FILE *fp, const unsigned char *data, int datalen, int cpl)
strbuf *ppk_save_sb(ssh2_userkey *key, const char *passphrase) strbuf *ppk_save_sb(ssh2_userkey *key, const char *passphrase)
{ {
strbuf *pub_blob, *priv_blob; strbuf *pub_blob, *priv_blob, *cipher_mac_keys_blob;
unsigned char *priv_blob_encrypted; unsigned char *priv_blob_encrypted;
int priv_encrypted_len; int priv_encrypted_len;
int cipherblk; int cipherblk;
int i; int i;
const char *cipherstr; const char *cipherstr;
ptrlen cipherkey, cipheriv, mackey;
const struct ppk_cipher *ciphertype;
unsigned char priv_mac[20]; unsigned char priv_mac[20];
/* /*
@ -1282,9 +1310,11 @@ strbuf *ppk_save_sb(ssh2_userkey *key, const char *passphrase)
if (passphrase) { if (passphrase) {
cipherstr = "aes256-cbc"; cipherstr = "aes256-cbc";
cipherblk = 16; cipherblk = 16;
ciphertype = &ppk_cipher_aes256_cbc;
} else { } else {
cipherstr = "none"; cipherstr = "none";
cipherblk = 1; cipherblk = 1;
ciphertype = &ppk_cipher_none;
} }
priv_encrypted_len = priv_blob->len + cipherblk - 1; priv_encrypted_len = priv_blob->len + cipherblk - 1;
priv_encrypted_len -= priv_encrypted_len % cipherblk; priv_encrypted_len -= priv_encrypted_len % cipherblk;
@ -1298,11 +1328,14 @@ strbuf *ppk_save_sb(ssh2_userkey *key, const char *passphrase)
memcpy(priv_blob_encrypted + priv_blob->len, priv_mac, memcpy(priv_blob_encrypted + priv_blob->len, priv_mac,
priv_encrypted_len - priv_blob->len); priv_encrypted_len - priv_blob->len);
cipher_mac_keys_blob = strbuf_new();
ssh2_ppk_derive_keys(2, ciphertype,
ptrlen_from_asciz(passphrase ? passphrase : ""),
cipher_mac_keys_blob, &cipherkey, &cipheriv, &mackey);
/* Now create the MAC. */ /* Now create the MAC. */
{ {
strbuf *macdata; strbuf *macdata;
unsigned char mackey[20];
char header[] = "putty-private-key-file-mac-key";
macdata = strbuf_new_nm(); macdata = strbuf_new_nm();
put_stringz(macdata, ssh_key_ssh_id(key->key)); put_stringz(macdata, ssh_key_ssh_id(key->key));
@ -1311,25 +1344,15 @@ strbuf *ppk_save_sb(ssh2_userkey *key, const char *passphrase)
put_string(macdata, pub_blob->s, pub_blob->len); put_string(macdata, pub_blob->s, pub_blob->len);
put_string(macdata, priv_blob_encrypted, priv_encrypted_len); put_string(macdata, priv_blob_encrypted, priv_encrypted_len);
ssh_hash *h = ssh_hash_new(&ssh_sha1); mac_simple(&ssh_hmac_sha1, mackey,
put_data(h, header, sizeof(header)-1);
if (passphrase)
put_data(h, passphrase, strlen(passphrase));
ssh_hash_final(h, mackey);
mac_simple(&ssh_hmac_sha1, make_ptrlen(mackey, 20),
ptrlen_from_strbuf(macdata), priv_mac); ptrlen_from_strbuf(macdata), priv_mac);
strbuf_free(macdata); strbuf_free(macdata);
smemclr(mackey, sizeof(mackey));
} }
if (passphrase) { if (passphrase) {
unsigned char key[40]; assert(cipherkey.len == 32);
aes256_encrypt_pubkey(cipherkey.ptr, cipheriv.ptr,
ssh2_ppk_derivekey(ptrlen_from_asciz(passphrase), key);
aes256_encrypt_pubkey(key, zero_iv,
priv_blob_encrypted, priv_encrypted_len); priv_blob_encrypted, priv_encrypted_len);
smemclr(key, sizeof(key));
} }
strbuf *out = strbuf_new_nm(); strbuf *out = strbuf_new_nm();
@ -1346,6 +1369,7 @@ strbuf *ppk_save_sb(ssh2_userkey *key, const char *passphrase)
strbuf_catf(out, "%02x", priv_mac[i]); strbuf_catf(out, "%02x", priv_mac[i]);
strbuf_catf(out, "\n"); strbuf_catf(out, "\n");
strbuf_free(cipher_mac_keys_blob);
strbuf_free(pub_blob); strbuf_free(pub_blob);
strbuf_free(priv_blob); strbuf_free(priv_blob);
smemclr(priv_blob_encrypted, priv_encrypted_len); smemclr(priv_blob_encrypted, priv_encrypted_len);