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

Better handling of outstanding CHANNEL_REQUESTS on channel destruction.

Part the first: make sure that all structures describing channel
requests are freed when the SSH connection is freed.  This involves
adding a means to ask a response handler to free any memory it holds.

Part the second: in ssh_channel_try_eof(), call
ssh2_channel_check_close() rather than emitting an SSH_MSG_CHANNEL_EOF
directly.  This avoids the possibility of closing the channel while a
CHANNEL_REQUEST is outstanding.

Also add some assertions that helped with tracking down the latter
problem.

[originally from svn r9623]
This commit is contained in:
Ben Harris 2012-08-25 21:06:48 +00:00
parent 4e623f5b23
commit 3d466aec90

44
ssh.c
View File

@ -4244,6 +4244,7 @@ static void ssh_channel_try_eof(struct ssh_channel *c)
if (ssh->version == 2 && bufchain_size(&c->v.v2.outbuffer) > 0) if (ssh->version == 2 && bufchain_size(&c->v.v2.outbuffer) > 0)
return; /* can't send EOF: pending outgoing data */ return; /* can't send EOF: pending outgoing data */
c->pending_eof = FALSE; /* we're about to send it */
if (ssh->version == 1) { if (ssh->version == 1) {
send_packet(ssh, SSH1_MSG_CHANNEL_CLOSE, PKT_INT, c->remoteid, send_packet(ssh, SSH1_MSG_CHANNEL_CLOSE, PKT_INT, c->remoteid,
PKT_END); PKT_END);
@ -4254,17 +4255,8 @@ static void ssh_channel_try_eof(struct ssh_channel *c)
ssh2_pkt_adduint32(pktout, c->remoteid); ssh2_pkt_adduint32(pktout, c->remoteid);
ssh2_pkt_send(ssh, pktout); ssh2_pkt_send(ssh, pktout);
c->closes |= CLOSES_SENT_EOF; c->closes |= CLOSES_SENT_EOF;
if (!((CLOSES_SENT_EOF | CLOSES_RCVD_EOF) & ~c->closes)) { ssh2_channel_check_close(c);
/*
* Also send MSG_CLOSE.
*/
pktout = ssh2_pkt_init(SSH2_MSG_CHANNEL_CLOSE);
ssh2_pkt_adduint32(pktout, c->remoteid);
ssh2_pkt_send(ssh, pktout);
c->closes |= CLOSES_SENT_CLOSE;
} }
}
c->pending_eof = FALSE; /* we've sent it now */
} }
void sshfwd_write_eof(struct ssh_channel *c) void sshfwd_write_eof(struct ssh_channel *c)
@ -6599,6 +6591,7 @@ static void ssh2_queue_chanreq_handler(struct ssh_channel *c,
struct outstanding_channel_request *ocr = struct outstanding_channel_request *ocr =
snew(struct outstanding_channel_request); snew(struct outstanding_channel_request);
assert(!(c->closes & (CLOSES_SENT_CLOSE | CLOSES_RCVD_CLOSE)));
ocr->handler = handler; ocr->handler = handler;
ocr->ctx = ctx; ocr->ctx = ctx;
ocr->next = NULL; ocr->next = NULL;
@ -6615,12 +6608,18 @@ static void ssh2_queue_chanreq_handler(struct ssh_channel *c,
* when it arrives. The returned packet is ready to have any * when it arrives. The returned packet is ready to have any
* request-specific data added and be sent. Note that if a handler is * request-specific data added and be sent. Note that if a handler is
* provided, it's essential that the request actually be sent. * provided, it's essential that the request actually be sent.
*
* The handler will usually be passed the response packet in pktin.
* If pktin is NULL, this means that no reply will ever be forthcoming
* (e.g. because the entire connection is being destroyed) and the
* handler should free any storage it's holding.
*/ */
static struct Packet *ssh2_chanreq_init(struct ssh_channel *c, char *type, static struct Packet *ssh2_chanreq_init(struct ssh_channel *c, char *type,
cchandler_fn_t handler, void *ctx) cchandler_fn_t handler, void *ctx)
{ {
struct Packet *pktout; struct Packet *pktout;
assert(!(c->closes & (CLOSES_SENT_CLOSE | CLOSES_RCVD_CLOSE)));
pktout = ssh2_pkt_init(SSH2_MSG_CHANNEL_REQUEST); pktout = ssh2_pkt_init(SSH2_MSG_CHANNEL_REQUEST);
ssh2_pkt_adduint32(pktout, c->remoteid); ssh2_pkt_adduint32(pktout, c->remoteid);
ssh2_pkt_addstring(pktout, type); ssh2_pkt_addstring(pktout, type);
@ -6911,8 +6910,10 @@ static void ssh_channel_destroy(struct ssh_channel *c)
} }
del234(ssh->channels, c); del234(ssh->channels, c);
if (ssh->version == 2) if (ssh->version == 2) {
bufchain_clear(&c->v.v2.outbuffer); bufchain_clear(&c->v.v2.outbuffer);
assert(c->v.v2.chanreq_head == NULL);
}
sfree(c); sfree(c);
/* /*
@ -6958,6 +6959,7 @@ static void ssh2_channel_check_close(struct ssh_channel *c)
} }
if (!((CLOSES_SENT_CLOSE | CLOSES_RCVD_CLOSE) & ~c->closes)) { if (!((CLOSES_SENT_CLOSE | CLOSES_RCVD_CLOSE) & ~c->closes)) {
assert(c->v.v2.chanreq_head == NULL);
/* /*
* We have both sent and received CLOSE, which means we're * We have both sent and received CLOSE, which means we're
* completely done with the channel. * completely done with the channel.
@ -7465,7 +7467,6 @@ static void ssh2_send_ttymode(void *data, char *mode, char *val)
ssh2_pkt_adduint32(pktout, arg); ssh2_pkt_adduint32(pktout, arg);
} }
static void ssh2_msg_authconn(Ssh ssh, struct Packet *pktin);
static void ssh2_maybe_setup_x11(struct ssh_channel *c, struct Packet *pktin, static void ssh2_maybe_setup_x11(struct ssh_channel *c, struct Packet *pktin,
void *ctx) void *ctx)
{ {
@ -7504,12 +7505,14 @@ static void ssh2_maybe_setup_x11(struct ssh_channel *c, struct Packet *pktin,
crWaitUntilV(pktin); crWaitUntilV(pktin);
if (pktin) {
if (pktin->type == SSH2_MSG_CHANNEL_SUCCESS) { if (pktin->type == SSH2_MSG_CHANNEL_SUCCESS) {
logevent("X11 forwarding enabled"); logevent("X11 forwarding enabled");
ssh->X11_fwd_enabled = TRUE; ssh->X11_fwd_enabled = TRUE;
} else } else
logevent("X11 forwarding refused"); logevent("X11 forwarding refused");
} }
}
sfree(s); sfree(s);
crFinishV; crFinishV;
} }
@ -7534,12 +7537,14 @@ static void ssh2_maybe_setup_agent(struct ssh_channel *c, struct Packet *pktin,
crWaitUntilV(pktin); crWaitUntilV(pktin);
if (pktin) {
if (pktin->type == SSH2_MSG_CHANNEL_SUCCESS) { if (pktin->type == SSH2_MSG_CHANNEL_SUCCESS) {
logevent("Agent forwarding enabled"); logevent("Agent forwarding enabled");
ssh->agentfwd_enabled = TRUE; ssh->agentfwd_enabled = TRUE;
} else } else
logevent("Agent forwarding refused"); logevent("Agent forwarding refused");
} }
}
sfree(s); sfree(s);
crFinishV; crFinishV;
} }
@ -7581,6 +7586,7 @@ static void ssh2_maybe_setup_pty(struct ssh_channel *c, struct Packet *pktin,
crWaitUntilV(pktin); crWaitUntilV(pktin);
if (pktin) {
if (pktin->type == SSH2_MSG_CHANNEL_SUCCESS) { if (pktin->type == SSH2_MSG_CHANNEL_SUCCESS) {
logeventf(ssh, "Allocated pty (ospeed %dbps, ispeed %dbps)", logeventf(ssh, "Allocated pty (ospeed %dbps, ispeed %dbps)",
ssh->ospeed, ssh->ispeed); ssh->ospeed, ssh->ispeed);
@ -7589,6 +7595,7 @@ static void ssh2_maybe_setup_pty(struct ssh_channel *c, struct Packet *pktin,
c_write_str(ssh, "Server refused to allocate pty\r\n"); c_write_str(ssh, "Server refused to allocate pty\r\n");
ssh->editing = ssh->echoing = 1; ssh->editing = ssh->echoing = 1;
} }
}
} else { } else {
ssh->editing = ssh->echoing = 1; ssh->editing = ssh->echoing = 1;
} }
@ -7639,6 +7646,7 @@ static void ssh2_setup_env(struct ssh_channel *c, struct Packet *pktin,
while (s->env_left > 0) { while (s->env_left > 0) {
crWaitUntilV(pktin); crWaitUntilV(pktin);
if (!pktin) goto out;
if (pktin->type == SSH2_MSG_CHANNEL_SUCCESS) if (pktin->type == SSH2_MSG_CHANNEL_SUCCESS)
s->env_ok++; s->env_ok++;
s->env_left--; s->env_left--;
@ -7655,6 +7663,7 @@ static void ssh2_setup_env(struct ssh_channel *c, struct Packet *pktin,
c_write_str(ssh, "Server refused to set all environment variables\r\n"); c_write_str(ssh, "Server refused to set all environment variables\r\n");
} }
} }
out:
sfree(s); sfree(s);
crFinishV; crFinishV;
} }
@ -9725,6 +9734,17 @@ static void ssh_free(void *handle)
pfd_close(c->u.pfd.s); pfd_close(c->u.pfd.s);
break; break;
} }
if (ssh->version == 2) {
struct outstanding_channel_request *ocr, *nocr;
ocr = c->v.v2.chanreq_head;
while (ocr) {
ocr->handler(c, NULL, ocr->ctx);
nocr = ocr->next;
sfree(ocr);
ocr = nocr;
}
bufchain_clear(&c->v.v2.outbuffer);
}
sfree(c); sfree(c);
} }
freetree234(ssh->channels); freetree234(ssh->channels);