1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-08 08:58:00 +00:00

Uppity: send SSH2_MSG_EXT_INFO.

This adds the framework to be able to send it in both client _and_
server (in the post-NEWKEYS slot); it's just that currently only the
server has anything it wants to put in it.

Uppity now announces its public key type list, which is enough by
itself to allow it to accept RFC 8332 rsa-sha2-* signatures during
userauth. (Because the key verification code receives an ssh-rsa host
key and validates it against the SHA2-based key algorithm structure
derived from the id string that was sent separately.)
This commit is contained in:
Simon Tatham 2020-11-22 08:41:49 +00:00
parent 24444eb396
commit fa134affeb
2 changed files with 52 additions and 2 deletions

View File

@ -858,7 +858,8 @@ static bool ssh2_scan_kexinits(
transport_direction *cs, transport_direction *sc,
bool *warn_kex, bool *warn_hk, bool *warn_cscipher, bool *warn_sccipher,
Ssh *ssh, bool *ignore_guess_cs_packet, bool *ignore_guess_sc_packet,
int *n_server_hostkeys, int server_hostkeys[MAXKEXLIST], unsigned *hkflags)
int *n_server_hostkeys, int server_hostkeys[MAXKEXLIST], unsigned *hkflags,
bool *can_send_ext_info)
{
BinarySource client[1], server[1];
int i;
@ -1058,6 +1059,20 @@ static bool ssh2_scan_kexinits(
}
}
/*
* Check whether the other side advertised support for EXT_INFO.
*/
{
ptrlen extinfo_advert =
(server_hostkeys ? PTRLEN_LITERAL("ext-info-c") :
PTRLEN_LITERAL("ext-info-s"));
ptrlen list = (server_hostkeys ? clists[KEXLIST_KEX] :
slists[KEXLIST_KEX]);
for (ptrlen word; get_commasep_word(&list, &word) ;)
if (ptrlen_eq_ptrlen(word, extinfo_advert))
*can_send_ext_info = true;
}
if (server_hostkeys) {
/*
* Finally, make an auxiliary pass over the server's host key
@ -1229,7 +1244,7 @@ static void ssh2_transport_process_queue(PacketProtocolLayer *ppl)
s->kexlists, &s->kex_alg, &s->hostkey_alg, s->cstrans,
s->sctrans, &s->warn_kex, &s->warn_hk, &s->warn_cscipher,
&s->warn_sccipher, s->ppl.ssh, NULL, &s->ignorepkt, &nhk, hks,
&s->hkflags))
&s->hkflags, &s->can_send_ext_info))
return; /* false means a fatal error function was called */
/*
@ -1428,6 +1443,40 @@ static void ssh2_transport_process_queue(PacketProtocolLayer *ppl)
strbuf_free(mac_key);
}
/*
* If that was our first key exchange, this is the moment to send
* our EXT_INFO, if we're sending one.
*/
if (!s->post_newkeys_ext_info) {
s->post_newkeys_ext_info = true; /* never do this again */
if (s->can_send_ext_info) {
strbuf *extinfo = strbuf_new();
uint32_t n_exts = 0;
if (s->ssc) {
/* Server->client EXT_INFO lists our supported user
* key algorithms. */
n_exts++;
put_stringz(extinfo, "server-sig-algs");
strbuf *list = strbuf_new();
for (size_t i = 0; i < n_keyalgs; i++)
add_to_commasep(list, all_keyalgs[i]->ssh_id);
put_stringsb(extinfo, list);
} else {
/* Client->server EXT_INFO is currently not sent, but here's
* where we should put things in it if we ever want to. */
}
/* Only send EXT_INFO if it's non-empty */
if (n_exts) {
pktout = ssh_bpp_new_pktout(s->ppl.bpp, SSH2_MSG_EXT_INFO);
put_uint32(pktout, n_exts);
put_datapl(pktout, ptrlen_from_strbuf(extinfo));
pq_push(s->ppl.out_pq, pktout);
}
}
}
/*
* Now our end of the key exchange is complete, we can send all
* our queued higher-layer packets. Transfer the whole of the next

View File

@ -185,6 +185,7 @@ struct ssh2_transport_state {
bool need_gss_transient_hostkey;
bool warned_about_no_gss_transient_hostkey;
bool got_session_id;
bool can_send_ext_info, post_newkeys_ext_info;
int dlgret;
bool guessok;
bool ignorepkt;