mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-10 01:48:00 +00:00
Fix crash if key exchange fails.
In the new modular SSH architecture, ssh2transport.c delegates the actual KEX packet exchange to ssh2kex_coroutine, which has different implementations for client and server. The KEX code actually in ssh2transport.c consists of looping on the coroutine until it zeroes out its state field in the ssh2transport state. But if something goes wrong enough during KEX that we call ssh_proto_error or any other fatal connection-terminating function, then when we return to ssh2transport.c, the ssh2transport state won't even exist for it to check that flag. Address Sanitiser pointed that out to me recently, so here's a fix in which we set an 'aborted' flag to tell the caller that its state has already been freed.
This commit is contained in:
parent
85fbb4216e
commit
1270d445e8
@ -12,7 +12,7 @@
|
|||||||
#include "storage.h"
|
#include "storage.h"
|
||||||
#include "ssh2transport.h"
|
#include "ssh2transport.h"
|
||||||
|
|
||||||
void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted)
|
||||||
{
|
{
|
||||||
PacketProtocolLayer *ppl = &s->ppl; /* for ppl_logevent */
|
PacketProtocolLayer *ppl = &s->ppl; /* for ppl_logevent */
|
||||||
PktIn *pktin;
|
PktIn *pktin;
|
||||||
@ -75,6 +75,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
s->p = get_mp_ssh2(pktin);
|
s->p = get_mp_ssh2(pktin);
|
||||||
@ -82,6 +83,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (get_err(pktin)) {
|
if (get_err(pktin)) {
|
||||||
ssh_proto_error(s->ppl.ssh,
|
ssh_proto_error(s->ppl.ssh,
|
||||||
"Unable to parse Diffie-Hellman group packet");
|
"Unable to parse Diffie-Hellman group packet");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
s->dh_ctx = dh_setup_gex(s->p, s->g);
|
s->dh_ctx = dh_setup_gex(s->p, s->g);
|
||||||
@ -123,6 +125,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
seat_set_busy_status(s->ppl.seat, BUSY_CPU);
|
seat_set_busy_status(s->ppl.seat, BUSY_CPU);
|
||||||
@ -133,6 +136,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (get_err(pktin)) {
|
if (get_err(pktin)) {
|
||||||
ssh_proto_error(s->ppl.ssh,
|
ssh_proto_error(s->ppl.ssh,
|
||||||
"Unable to parse Diffie-Hellman reply packet");
|
"Unable to parse Diffie-Hellman reply packet");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -141,6 +145,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (err) {
|
if (err) {
|
||||||
ssh_proto_error(s->ppl.ssh, "Diffie-Hellman reply failed "
|
ssh_proto_error(s->ppl.ssh, "Diffie-Hellman reply failed "
|
||||||
"validation: %s", err);
|
"validation: %s", err);
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -180,6 +185,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
s->ecdh_key = ssh_ecdhkex_newkey(s->kex_alg);
|
s->ecdh_key = ssh_ecdhkex_newkey(s->kex_alg);
|
||||||
if (!s->ecdh_key) {
|
if (!s->ecdh_key) {
|
||||||
ssh_sw_abort(s->ppl.ssh, "Unable to generate key for ECDH");
|
ssh_sw_abort(s->ppl.ssh, "Unable to generate key for ECDH");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -199,6 +205,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -220,6 +227,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (!get_err(pktin) && !s->K) {
|
if (!get_err(pktin) && !s->K) {
|
||||||
ssh_proto_error(s->ppl.ssh, "Received invalid elliptic curve "
|
ssh_proto_error(s->ppl.ssh, "Received invalid elliptic curve "
|
||||||
"point in ECDH reply");
|
"point in ECDH reply");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -227,6 +235,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
s->sigdata = get_string(pktin);
|
s->sigdata = get_string(pktin);
|
||||||
if (get_err(pktin)) {
|
if (get_err(pktin)) {
|
||||||
ssh_proto_error(s->ppl.ssh, "Unable to parse ECDH reply packet");
|
ssh_proto_error(s->ppl.ssh, "Unable to parse ECDH reply packet");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -285,6 +294,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
s->p = get_mp_ssh2(pktin);
|
s->p = get_mp_ssh2(pktin);
|
||||||
@ -292,6 +302,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (get_err(pktin)) {
|
if (get_err(pktin)) {
|
||||||
ssh_proto_error(s->ppl.ssh,
|
ssh_proto_error(s->ppl.ssh,
|
||||||
"Unable to parse Diffie-Hellman group packet");
|
"Unable to parse Diffie-Hellman group packet");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
s->dh_ctx = dh_setup_gex(s->p, s->g);
|
s->dh_ctx = dh_setup_gex(s->p, s->g);
|
||||||
@ -319,6 +330,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (s->gss_stat != SSH_GSS_OK) {
|
if (s->gss_stat != SSH_GSS_OK) {
|
||||||
ssh_sw_abort(s->ppl.ssh,
|
ssh_sw_abort(s->ppl.ssh,
|
||||||
"GSSAPI key exchange failed to initialise");
|
"GSSAPI key exchange failed to initialise");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -349,6 +361,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
"GSSAPI key exchange failed to initialise "
|
"GSSAPI key exchange failed to initialise "
|
||||||
"context: %s", err);
|
"context: %s", err);
|
||||||
sfree(err);
|
sfree(err);
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -362,6 +375,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (s->gss_sndtok.length == 0) {
|
if (s->gss_sndtok.length == 0) {
|
||||||
ssh_sw_abort(s->ppl.ssh, "GSSAPI key exchange failed: "
|
ssh_sw_abort(s->ppl.ssh, "GSSAPI key exchange failed: "
|
||||||
"no initial context token");
|
"no initial context token");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
put_string(pktout,
|
put_string(pktout,
|
||||||
@ -453,6 +467,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} while (s->gss_rcvtok.length ||
|
} while (s->gss_rcvtok.length ||
|
||||||
@ -511,6 +526,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -524,6 +540,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (!s->rsa_kex_key) {
|
if (!s->rsa_kex_key) {
|
||||||
ssh_proto_error(s->ppl.ssh,
|
ssh_proto_error(s->ppl.ssh,
|
||||||
"Unable to parse RSA public key packet");
|
"Unable to parse RSA public key packet");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -590,12 +607,14 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
s->sigdata = get_string(pktin);
|
s->sigdata = get_string(pktin);
|
||||||
if (get_err(pktin)) {
|
if (get_err(pktin)) {
|
||||||
ssh_proto_error(s->ppl.ssh, "Unable to parse RSA kex signature");
|
ssh_proto_error(s->ppl.ssh, "Unable to parse RSA kex signature");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -622,6 +641,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh_sw_abort(s->ppl.ssh, "GSSAPI key exchange MIC was "
|
ssh_sw_abort(s->ppl.ssh, "GSSAPI key exchange MIC was "
|
||||||
"not valid");
|
"not valid");
|
||||||
}
|
}
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -653,6 +673,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (s->kex_alg->main_type != KEXTYPE_GSS) {
|
if (s->kex_alg->main_type != KEXTYPE_GSS) {
|
||||||
if (!s->hkey) {
|
if (!s->hkey) {
|
||||||
ssh_proto_error(s->ppl.ssh, "Server's host key is invalid");
|
ssh_proto_error(s->ppl.ssh, "Server's host key is invalid");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -662,6 +683,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
#ifndef FUZZING
|
#ifndef FUZZING
|
||||||
ssh_proto_error(s->ppl.ssh, "Signature from server's host key "
|
ssh_proto_error(s->ppl.ssh, "Signature from server's host key "
|
||||||
"is invalid");
|
"is invalid");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
@ -752,6 +774,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ppl_logevent("%s", s->fingerprint);
|
ppl_logevent("%s", s->fingerprint);
|
||||||
ssh_sw_abort(s->ppl.ssh, "Server's host key did not match any "
|
ssh_sw_abort(s->ppl.ssh, "Server's host key did not match any "
|
||||||
"used in previous GSS kex");
|
"used in previous GSS kex");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -811,6 +834,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (s->dlgret == 0) { /* did not match */
|
if (s->dlgret == 0) { /* did not match */
|
||||||
ssh_sw_abort(s->ppl.ssh, "Host key did not appear in manually "
|
ssh_sw_abort(s->ppl.ssh, "Host key did not appear in manually "
|
||||||
"configured list");
|
"configured list");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
} else if (s->dlgret < 0) { /* none configured; use standard handling */
|
} else if (s->dlgret < 0) { /* none configured; use standard handling */
|
||||||
s->dlgret = seat_verify_ssh_host_key(
|
s->dlgret = seat_verify_ssh_host_key(
|
||||||
@ -824,6 +848,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (s->dlgret == 0) {
|
if (s->dlgret == 0) {
|
||||||
ssh_user_close(s->ppl.ssh,
|
ssh_user_close(s->ppl.ssh,
|
||||||
"User aborted at host key verification");
|
"User aborted at host key verification");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -864,6 +889,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
#ifndef FUZZING
|
#ifndef FUZZING
|
||||||
ssh_sw_abort(s->ppl.ssh,
|
ssh_sw_abort(s->ppl.ssh,
|
||||||
"Host key was different in repeat key exchange");
|
"Host key was different in repeat key exchange");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
@ -36,7 +36,7 @@ static void no_progress(void *param, int action, int phase, int iprogress)
|
|||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted)
|
||||||
{
|
{
|
||||||
PacketProtocolLayer *ppl = &s->ppl; /* for ppl_logevent */
|
PacketProtocolLayer *ppl = &s->ppl; /* for ppl_logevent */
|
||||||
PktIn *pktin;
|
PktIn *pktin;
|
||||||
@ -78,6 +78,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -134,12 +135,14 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
s->f = get_mp_ssh2(pktin);
|
s->f = get_mp_ssh2(pktin);
|
||||||
if (get_err(pktin)) {
|
if (get_err(pktin)) {
|
||||||
ssh_proto_error(s->ppl.ssh,
|
ssh_proto_error(s->ppl.ssh,
|
||||||
"Unable to parse Diffie-Hellman initial packet");
|
"Unable to parse Diffie-Hellman initial packet");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -148,6 +151,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (err) {
|
if (err) {
|
||||||
ssh_proto_error(s->ppl.ssh, "Diffie-Hellman initial packet "
|
ssh_proto_error(s->ppl.ssh, "Diffie-Hellman initial packet "
|
||||||
"failed validation: %s", err);
|
"failed validation: %s", err);
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -187,6 +191,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
s->ecdh_key = ssh_ecdhkex_newkey(s->kex_alg);
|
s->ecdh_key = ssh_ecdhkex_newkey(s->kex_alg);
|
||||||
if (!s->ecdh_key) {
|
if (!s->ecdh_key) {
|
||||||
ssh_sw_abort(s->ppl.ssh, "Unable to generate key for ECDH");
|
ssh_sw_abort(s->ppl.ssh, "Unable to generate key for ECDH");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -198,6 +203,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -209,6 +215,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
if (!get_err(pktin) && !s->K) {
|
if (!get_err(pktin) && !s->K) {
|
||||||
ssh_proto_error(s->ppl.ssh, "Received invalid elliptic curve "
|
ssh_proto_error(s->ppl.ssh, "Received invalid elliptic curve "
|
||||||
"point in ECDH initial packet");
|
"point in ECDH initial packet");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -262,6 +269,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
ssh2_pkt_type(s->ppl.bpp->pls->kctx,
|
||||||
s->ppl.bpp->pls->actx,
|
s->ppl.bpp->pls->actx,
|
||||||
pktin->type));
|
pktin->type));
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -274,6 +282,7 @@ void ssh2kex_coroutine(struct ssh2_transport_state *s)
|
|||||||
|
|
||||||
if (!s->K) {
|
if (!s->K) {
|
||||||
ssh_proto_error(s->ppl.ssh, "Unable to decrypt RSA kex secret");
|
ssh_proto_error(s->ppl.ssh, "Unable to decrypt RSA kex secret");
|
||||||
|
*aborted = true;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1236,9 +1236,12 @@ static void ssh2_transport_process_queue(PacketProtocolLayer *ppl)
|
|||||||
put_string(s->exhash, s->server_kexinit->u, s->server_kexinit->len);
|
put_string(s->exhash, s->server_kexinit->u, s->server_kexinit->len);
|
||||||
s->crStateKex = 0;
|
s->crStateKex = 0;
|
||||||
while (1) {
|
while (1) {
|
||||||
ssh2kex_coroutine(s);
|
bool aborted = false;
|
||||||
|
ssh2kex_coroutine(s, &aborted);
|
||||||
|
if (aborted)
|
||||||
|
return; /* disaster: our entire state has been freed */
|
||||||
if (!s->crStateKex)
|
if (!s->crStateKex)
|
||||||
break;
|
break; /* kex phase has terminated normally */
|
||||||
crReturnV;
|
crReturnV;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -224,7 +224,10 @@ void ssh2_transport_dialog_callback(void *, int);
|
|||||||
/* Provided by transport for use in kex */
|
/* Provided by transport for use in kex */
|
||||||
void ssh2transport_finalise_exhash(struct ssh2_transport_state *s);
|
void ssh2transport_finalise_exhash(struct ssh2_transport_state *s);
|
||||||
|
|
||||||
/* Provided by kex for use in transport */
|
/* Provided by kex for use in transport. Must set the 'aborted' flag
|
||||||
void ssh2kex_coroutine(struct ssh2_transport_state *s);
|
* if it throws a connection-terminating error, so that the caller
|
||||||
|
* won't have to check that by looking inside its state parameter
|
||||||
|
* which might already have been freed. */
|
||||||
|
void ssh2kex_coroutine(struct ssh2_transport_state *s, bool *aborted);
|
||||||
|
|
||||||
#endif /* PUTTY_SSH2TRANSPORT_H */
|
#endif /* PUTTY_SSH2TRANSPORT_H */
|
||||||
|
Loading…
Reference in New Issue
Block a user