mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 01:02:24 +00:00
281 lines
8.2 KiB
C
281 lines
8.2 KiB
C
|
/*
|
||
|
* Binary packet protocol for SSH-1.
|
||
|
*/
|
||
|
|
||
|
#include <assert.h>
|
||
|
|
||
|
#include "putty.h"
|
||
|
#include "ssh.h"
|
||
|
#include "sshbpp.h"
|
||
|
#include "sshcr.h"
|
||
|
|
||
|
struct ssh1_bpp_state {
|
||
|
int crState;
|
||
|
long len, pad, biglen, length, maxlen;
|
||
|
unsigned char *data;
|
||
|
unsigned long realcrc, gotcrc;
|
||
|
int chunk;
|
||
|
PktIn *pktin;
|
||
|
|
||
|
const struct ssh_cipher *cipher;
|
||
|
void *cipher_ctx;
|
||
|
|
||
|
void *crcda_ctx;
|
||
|
|
||
|
void *compctx, *decompctx;
|
||
|
|
||
|
BinaryPacketProtocol bpp;
|
||
|
};
|
||
|
|
||
|
static void ssh1_bpp_free(BinaryPacketProtocol *bpp);
|
||
|
static void ssh1_bpp_handle_input(BinaryPacketProtocol *bpp);
|
||
|
static PktOut *ssh1_bpp_new_pktout(int type);
|
||
|
static void ssh1_bpp_format_packet(BinaryPacketProtocol *bpp, PktOut *pkt);
|
||
|
|
||
|
const struct BinaryPacketProtocolVtable ssh1_bpp_vtable = {
|
||
|
ssh1_bpp_free,
|
||
|
ssh1_bpp_handle_input,
|
||
|
ssh1_bpp_new_pktout,
|
||
|
ssh1_bpp_format_packet,
|
||
|
};
|
||
|
|
||
|
BinaryPacketProtocol *ssh1_bpp_new(void)
|
||
|
{
|
||
|
struct ssh1_bpp_state *s = snew(struct ssh1_bpp_state);
|
||
|
memset(s, 0, sizeof(*s));
|
||
|
s->bpp.vt = &ssh1_bpp_vtable;
|
||
|
return &s->bpp;
|
||
|
}
|
||
|
|
||
|
static void ssh1_bpp_free(BinaryPacketProtocol *bpp)
|
||
|
{
|
||
|
struct ssh1_bpp_state *s = FROMFIELD(bpp, struct ssh1_bpp_state, bpp);
|
||
|
if (s->cipher)
|
||
|
s->cipher->free_context(s->cipher_ctx);
|
||
|
if (s->compctx)
|
||
|
zlib_compress_cleanup(s->compctx);
|
||
|
if (s->decompctx)
|
||
|
zlib_decompress_cleanup(s->decompctx);
|
||
|
if (s->crcda_ctx)
|
||
|
crcda_free_context(s->crcda_ctx);
|
||
|
if (s->pktin)
|
||
|
ssh_unref_packet(s->pktin);
|
||
|
sfree(s);
|
||
|
}
|
||
|
|
||
|
void ssh1_bpp_new_cipher(BinaryPacketProtocol *bpp,
|
||
|
const struct ssh_cipher *cipher,
|
||
|
const void *session_key)
|
||
|
{
|
||
|
struct ssh1_bpp_state *s;
|
||
|
assert(bpp->vt == &ssh1_bpp_vtable);
|
||
|
s = FROMFIELD(bpp, struct ssh1_bpp_state, bpp);
|
||
|
|
||
|
assert(!s->cipher);
|
||
|
|
||
|
s->cipher = cipher;
|
||
|
if (s->cipher) {
|
||
|
s->cipher_ctx = cipher->make_context();
|
||
|
cipher->sesskey(s->cipher_ctx, session_key);
|
||
|
|
||
|
assert(!s->crcda_ctx);
|
||
|
s->crcda_ctx = crcda_make_context();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void ssh1_bpp_start_compression(BinaryPacketProtocol *bpp)
|
||
|
{
|
||
|
struct ssh1_bpp_state *s;
|
||
|
assert(bpp->vt == &ssh1_bpp_vtable);
|
||
|
s = FROMFIELD(bpp, struct ssh1_bpp_state, bpp);
|
||
|
|
||
|
assert(!s->compctx);
|
||
|
assert(!s->decompctx);
|
||
|
|
||
|
s->compctx = zlib_compress_init();
|
||
|
s->decompctx = zlib_decompress_init();
|
||
|
}
|
||
|
|
||
|
static void ssh1_bpp_handle_input(BinaryPacketProtocol *bpp)
|
||
|
{
|
||
|
struct ssh1_bpp_state *s = FROMFIELD(bpp, struct ssh1_bpp_state, bpp);
|
||
|
|
||
|
crBegin(s->crState);
|
||
|
|
||
|
while (1) {
|
||
|
s->maxlen = 0;
|
||
|
s->length = 0;
|
||
|
|
||
|
{
|
||
|
unsigned char lenbuf[4];
|
||
|
crMaybeWaitUntilV(bufchain_try_fetch_consume(
|
||
|
bpp->in_raw, lenbuf, 4));
|
||
|
s->len = toint(GET_32BIT_MSB_FIRST(lenbuf));
|
||
|
}
|
||
|
|
||
|
if (s->len < 0 || s->len > 262144) { /* SSH1.5-mandated max size */
|
||
|
s->bpp.error = dupprintf(
|
||
|
"Extremely large packet length from server suggests"
|
||
|
" data stream corruption");
|
||
|
crStopV;
|
||
|
}
|
||
|
|
||
|
s->pad = 8 - (s->len % 8);
|
||
|
s->biglen = s->len + s->pad;
|
||
|
s->length = s->len - 5;
|
||
|
|
||
|
/*
|
||
|
* Allocate the packet to return, now we know its length.
|
||
|
*/
|
||
|
s->pktin = snew_plus(PktIn, s->biglen);
|
||
|
s->pktin->qnode.prev = s->pktin->qnode.next = NULL;
|
||
|
s->pktin->refcount = 1;
|
||
|
s->pktin->type = 0;
|
||
|
|
||
|
s->maxlen = s->biglen;
|
||
|
s->data = snew_plus_get_aux(s->pktin);
|
||
|
|
||
|
crMaybeWaitUntilV(bufchain_try_fetch_consume(
|
||
|
bpp->in_raw, s->data, s->biglen));
|
||
|
|
||
|
if (s->cipher && detect_attack(s->crcda_ctx,
|
||
|
s->data, s->biglen, NULL)) {
|
||
|
s->bpp.error = dupprintf(
|
||
|
"Network attack (CRC compensation) detected!");
|
||
|
crStopV;
|
||
|
}
|
||
|
|
||
|
if (s->cipher)
|
||
|
s->cipher->decrypt(s->cipher_ctx, s->data, s->biglen);
|
||
|
|
||
|
s->realcrc = crc32_compute(s->data, s->biglen - 4);
|
||
|
s->gotcrc = GET_32BIT(s->data + s->biglen - 4);
|
||
|
if (s->gotcrc != s->realcrc) {
|
||
|
s->bpp.error = dupprintf(
|
||
|
"Incorrect CRC received on packet");
|
||
|
crStopV;
|
||
|
}
|
||
|
|
||
|
if (s->decompctx) {
|
||
|
unsigned char *decompblk;
|
||
|
int decomplen;
|
||
|
if (!zlib_decompress_block(s->decompctx,
|
||
|
s->data + s->pad, s->length + 1,
|
||
|
&decompblk, &decomplen)) {
|
||
|
s->bpp.error = dupprintf(
|
||
|
"Zlib decompression encountered invalid data");
|
||
|
crStopV;
|
||
|
}
|
||
|
|
||
|
if (s->maxlen < s->pad + decomplen) {
|
||
|
PktIn *old_pktin = s->pktin;
|
||
|
|
||
|
s->maxlen = s->pad + decomplen;
|
||
|
s->pktin = snew_plus(PktIn, s->maxlen);
|
||
|
*s->pktin = *old_pktin; /* structure copy */
|
||
|
s->data = snew_plus_get_aux(s->pktin);
|
||
|
|
||
|
smemclr(old_pktin, s->biglen);
|
||
|
sfree(old_pktin);
|
||
|
}
|
||
|
|
||
|
memcpy(s->data + s->pad, decompblk, decomplen);
|
||
|
sfree(decompblk);
|
||
|
s->length = decomplen - 1;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
* Now we can find the bounds of the semantic content of the
|
||
|
* packet, and the initial type byte.
|
||
|
*/
|
||
|
s->data += s->pad;
|
||
|
s->pktin->type = *s->data++;
|
||
|
BinarySource_INIT(s->pktin, s->data, s->length);
|
||
|
|
||
|
if (s->bpp.logctx) {
|
||
|
logblank_t blanks[MAX_BLANKS];
|
||
|
int nblanks = ssh1_censor_packet(
|
||
|
s->bpp.pls, s->pktin->type, FALSE,
|
||
|
make_ptrlen(s->data, s->length), blanks);
|
||
|
log_packet(s->bpp.logctx, PKT_INCOMING, s->pktin->type,
|
||
|
ssh1_pkt_type(s->pktin->type),
|
||
|
get_ptr(s->pktin), get_avail(s->pktin), nblanks, blanks,
|
||
|
NULL, 0, NULL);
|
||
|
}
|
||
|
|
||
|
pq_push(s->bpp.in_pq, s->pktin);
|
||
|
|
||
|
{
|
||
|
int type = s->pktin->type;
|
||
|
s->pktin = NULL;
|
||
|
|
||
|
if (type == SSH1_MSG_DISCONNECT)
|
||
|
s->bpp.seen_disconnect = TRUE;
|
||
|
}
|
||
|
}
|
||
|
crFinishV;
|
||
|
}
|
||
|
|
||
|
static PktOut *ssh1_bpp_new_pktout(int pkt_type)
|
||
|
{
|
||
|
PktOut *pkt = ssh_new_packet();
|
||
|
pkt->length = 4 + 8; /* space for length + max padding */
|
||
|
put_byte(pkt, pkt_type);
|
||
|
pkt->prefix = pkt->length;
|
||
|
pkt->type = pkt_type;
|
||
|
pkt->downstream_id = 0;
|
||
|
pkt->additional_log_text = NULL;
|
||
|
return pkt;
|
||
|
}
|
||
|
|
||
|
static void ssh1_bpp_format_packet(BinaryPacketProtocol *bpp, PktOut *pkt)
|
||
|
{
|
||
|
struct ssh1_bpp_state *s = FROMFIELD(bpp, struct ssh1_bpp_state, bpp);
|
||
|
int pad, biglen, i, pktoffs;
|
||
|
unsigned long crc;
|
||
|
int len;
|
||
|
|
||
|
if (s->bpp.logctx) {
|
||
|
ptrlen pktdata = make_ptrlen(pkt->data + pkt->prefix,
|
||
|
pkt->length - pkt->prefix);
|
||
|
logblank_t blanks[MAX_BLANKS];
|
||
|
int nblanks = ssh1_censor_packet(
|
||
|
s->bpp.pls, pkt->type, TRUE, pktdata, blanks);
|
||
|
log_packet(s->bpp.logctx, PKT_OUTGOING, pkt->type,
|
||
|
ssh1_pkt_type(pkt->type),
|
||
|
pktdata.ptr, pktdata.len, nblanks, blanks,
|
||
|
NULL, 0, NULL);
|
||
|
}
|
||
|
|
||
|
if (s->compctx) {
|
||
|
unsigned char *compblk;
|
||
|
int complen;
|
||
|
zlib_compress_block(s->compctx, pkt->data + 12, pkt->length - 12,
|
||
|
&compblk, &complen);
|
||
|
/* Replace the uncompressed packet data with the compressed
|
||
|
* version. */
|
||
|
pkt->length = 12;
|
||
|
put_data(pkt, compblk, complen);
|
||
|
sfree(compblk);
|
||
|
}
|
||
|
|
||
|
put_uint32(pkt, 0); /* space for CRC */
|
||
|
len = pkt->length - 4 - 8; /* len(type+data+CRC) */
|
||
|
pad = 8 - (len % 8);
|
||
|
pktoffs = 8 - pad;
|
||
|
biglen = len + pad; /* len(padding+type+data+CRC) */
|
||
|
|
||
|
for (i = pktoffs; i < 4+8; i++)
|
||
|
pkt->data[i] = random_byte();
|
||
|
crc = crc32_compute(pkt->data + pktoffs + 4,
|
||
|
biglen - 4); /* all ex len */
|
||
|
PUT_32BIT(pkt->data + pktoffs + 4 + biglen - 4, crc);
|
||
|
PUT_32BIT(pkt->data + pktoffs, len);
|
||
|
|
||
|
if (s->cipher)
|
||
|
s->cipher->encrypt(s->cipher_ctx, pkt->data + pktoffs + 4, biglen);
|
||
|
|
||
|
bufchain_add(s->bpp.out_raw, pkt->data + pktoffs,
|
||
|
biglen + 4); /* len(length+padding+type+data+CRC) */
|
||
|
}
|