summaryrefslogtreecommitdiffstats
path: root/rdp/rdp_sec.c
diff options
context:
space:
mode:
Diffstat (limited to 'rdp/rdp_sec.c')
-rw-r--r--rdp/rdp_sec.c651
1 files changed, 651 insertions, 0 deletions
diff --git a/rdp/rdp_sec.c b/rdp/rdp_sec.c
new file mode 100644
index 00000000..0bd727ee
--- /dev/null
+++ b/rdp/rdp_sec.c
@@ -0,0 +1,651 @@
+/*
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License
+ along with this program; if not, write to the Free Software
+ Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+
+ xrdp: A Remote Desktop Protocol server.
+ Copyright (C) Jay Sorg 2005
+
+ librdp secure layer
+
+*/
+
+#include "rdp.h"
+
+static char g_pad_54[40] =
+{ 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54,
+ 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54, 54,
+ 54, 54, 54, 54, 54, 54, 54, 54 };
+
+static char g_pad_92[48] =
+{ 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92,
+ 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92,
+ 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92, 92 };
+
+/*****************************************************************************/
+struct rdp_sec* APP_CC
+rdp_sec_create(struct rdp_rdp* owner)
+{
+ struct rdp_sec* self;
+
+ self = (struct rdp_sec*)g_malloc(sizeof(struct rdp_sec), 1);
+ self->rdp_layer = owner;
+ make_stream(self->client_mcs_data);
+ init_stream(self->client_mcs_data, 8192);
+ make_stream(self->server_mcs_data);
+ init_stream(self->server_mcs_data, 8192);
+ self->mcs_layer = rdp_mcs_create(self, self->client_mcs_data,
+ self->server_mcs_data);
+ self->decrypt_rc4_info = g_rc4_info_create();
+ self->encrypt_rc4_info = g_rc4_info_create();
+ self->lic_layer = rdp_lic_create(self);
+ return self;
+}
+
+/*****************************************************************************/
+void APP_CC
+rdp_sec_delete(struct rdp_sec* self)
+{
+ if (self == 0)
+ {
+ return;
+ }
+ rdp_lic_delete(self->lic_layer);
+ rdp_mcs_delete(self->mcs_layer);
+ free_stream(self->client_mcs_data);
+ free_stream(self->server_mcs_data);
+ g_rc4_info_delete(self->decrypt_rc4_info);
+ g_rc4_info_delete(self->encrypt_rc4_info);
+ g_free(self);
+}
+
+/*****************************************************************************/
+/* Reduce key entropy from 64 to 40 bits */
+static void APP_CC
+rdp_sec_make_40bit(char* key)
+{
+ key[0] = 0xd1;
+ key[1] = 0x26;
+ key[2] = 0x9e;
+}
+
+/*****************************************************************************/
+/* returns error */
+/* update an encryption key */
+static int APP_CC
+rdp_sec_update(char* key, char* update_key, int key_len)
+{
+ char shasig[20];
+ void* sha1_info;
+ void* md5_info;
+ void* rc4_info;
+
+ sha1_info = g_sha1_info_create();
+ md5_info = g_md5_info_create();
+ rc4_info = g_rc4_info_create();
+ g_sha1_clear(sha1_info);
+ g_sha1_transform(sha1_info, update_key, key_len);
+ g_sha1_transform(sha1_info, g_pad_54, 40);
+ g_sha1_transform(sha1_info, key, key_len);
+ g_sha1_complete(sha1_info, shasig);
+ g_md5_clear(md5_info);
+ g_md5_transform(md5_info, update_key, key_len);
+ g_md5_transform(md5_info, g_pad_92, 48);
+ g_md5_transform(md5_info, shasig, 20);
+ g_md5_complete(md5_info, key);
+ g_rc4_set_key(rc4_info, key, key_len);
+ g_rc4_crypt(rc4_info, key, key_len);
+ if (key_len == 8)
+ {
+ rdp_sec_make_40bit(key);
+ }
+ g_sha1_info_delete(sha1_info);
+ g_md5_info_delete(md5_info);
+ g_rc4_info_delete(rc4_info);
+ return 0;
+}
+
+/*****************************************************************************/
+static void APP_CC
+rdp_sec_decrypt(struct rdp_sec* self, char* data, int len)
+{
+ if (self->decrypt_use_count == 4096)
+ {
+ rdp_sec_update(self->decrypt_key, self->decrypt_update_key,
+ self->rc4_key_len);
+ g_rc4_set_key(self->decrypt_rc4_info, self->decrypt_key,
+ self->rc4_key_len);
+ self->decrypt_use_count = 0;
+ }
+ g_rc4_crypt(self->decrypt_rc4_info, data, len);
+ self->decrypt_use_count++;
+}
+
+/*****************************************************************************/
+/* returns error */
+int APP_CC
+rdp_sec_recv(struct rdp_sec* self, struct stream* s, int* chan)
+{
+ int flags;
+
+ DEBUG((" in rdp_sec_recv\r\n"));
+ if (rdp_mcs_recv(self->mcs_layer, s, chan) != 0)
+ {
+ DEBUG((" error in rdp_sec_recv, rdp_mcs_recv failed\r\n"));
+ return 1;
+ }
+ in_uint32_le(s, flags);
+ DEBUG((" rdp_sec_recv flags %8.8x\r\n", flags));
+ if (flags & SEC_ENCRYPT) /* 0x08 */
+ {
+ in_uint8s(s, 8); /* signature */
+ rdp_sec_decrypt(self, s->p, s->end - s->p);
+ }
+ if (flags & SEC_LICENCE_NEG) /* 0x80 */
+ {
+ DEBUG((" in rdp_sec_recv, got SEC_LICENCE_NEG\r\n"));
+ rdp_lic_process(self->lic_layer, s);
+ *chan = 0;
+ }
+ DEBUG((" out rdp_sec_recv\r\n"));
+ return 0;
+}
+
+/*****************************************************************************/
+/* prepare client mcs data to send in mcs layer */
+static void APP_CC
+rdp_sec_out_mcs_data(struct rdp_sec* self)
+{
+ struct stream* s;
+ int hostlen;
+ int length;
+
+ s = self->client_mcs_data;
+ init_stream(s, 512);
+ self->rdp_layer->mod->hostname[15] = 0; /* limit length to 15 */
+ hostlen = 2 * g_strlen(self->rdp_layer->mod->hostname);
+ length = 158 + 76 + 12 + 4;
+ /* Generic Conference Control (T.124) ConferenceCreateRequest */
+ out_uint16_be(s, 5);
+ out_uint16_be(s, 0x14);
+ out_uint8(s, 0x7c);
+ out_uint16_be(s, 1);
+ out_uint16_be(s, (length | 0x8000)); /* remaining length */
+ out_uint16_be(s, 8); /* length? */
+ out_uint16_be(s, 16);
+ out_uint8(s, 0);
+ out_uint16_le(s, 0xc001);
+ out_uint8(s, 0);
+ out_uint32_le(s, 0x61637544); /* OEM ID: "Duca", as in Ducati. */
+ out_uint16_be(s, ((length - 14) | 0x8000)); /* remaining length */
+ /* Client information */
+ out_uint16_le(s, SEC_TAG_CLI_INFO);
+ out_uint16_le(s, 212); /* length */
+ out_uint16_le(s, 1); /* RDP version. 1 == RDP4, 4 == RDP5. */
+ out_uint16_le(s, 8);
+ out_uint16_le(s, self->rdp_layer->mod->width);
+ out_uint16_le(s, self->rdp_layer->mod->height);
+ out_uint16_le(s, 0xca01);
+ out_uint16_le(s, 0xaa03);
+ out_uint32_le(s, self->rdp_layer->mod->keylayout);
+ out_uint32_le(s, 2600); /* Client build */
+ /* Unicode name of client, padded to 32 bytes */
+ rdp_rdp_out_unistr(s, self->rdp_layer->mod->hostname);
+ out_uint8s(s, 30 - hostlen);
+ out_uint32_le(s, 4);
+ out_uint32_le(s, 0);
+ out_uint32_le(s, 12);
+ out_uint8s(s, 64); /* reserved? 4 + 12 doublewords */
+ out_uint16_le(s, 0xca01); /* color depth? */
+ out_uint16_le(s, 1);
+ out_uint32_le(s, 0);
+ out_uint8(s, self->rdp_layer->mod->rdp_bpp);
+ out_uint16_le(s, 0x0700);
+ out_uint8(s, 0);
+ out_uint32_le(s, 1);
+ out_uint8s(s, 64); /* End of client info */
+ out_uint16_le(s, SEC_TAG_CLI_4);
+ out_uint16_le(s, 12);
+ out_uint32_le(s, 9);
+ out_uint32_le(s, 0);
+ /* Client encryption settings */
+ out_uint16_le(s, SEC_TAG_CLI_CRYPT);
+ out_uint16_le(s, 12); /* length */
+ /* encryption supported, 128-bit supported */
+ out_uint32_le(s, 0x3);
+ out_uint32_le(s, 0); /* Unknown */
+ s_mark_end(s);
+}
+
+/*****************************************************************************/
+/* Parse a public key structure */
+/* returns boolean */
+static int APP_CC
+rdp_sec_parse_public_key(struct rdp_sec* self, struct stream* s,
+ char* modulus, char* exponent)
+{
+ int magic;
+ int modulus_len;
+
+ in_uint32_le(s, magic);
+ if (magic != SEC_RSA_MAGIC)
+ {
+ return 0;
+ }
+ in_uint32_le(s, modulus_len);
+ if (modulus_len != SEC_MODULUS_SIZE + SEC_PADDING_SIZE)
+ {
+ return 0;
+ }
+ in_uint8s(s, 8);
+ in_uint8a(s, exponent, SEC_EXPONENT_SIZE);
+ in_uint8a(s, modulus, SEC_MODULUS_SIZE);
+ in_uint8s(s, SEC_PADDING_SIZE);
+ return s_check(s);
+}
+
+/*****************************************************************************/
+/* Parse a crypto information structure */
+/* returns boolean */
+static int APP_CC
+rdp_sec_parse_crypt_info(struct rdp_sec* self, struct stream* s,
+ char* modulus, char* exponent)
+{
+ int random_len;
+ int rsa_info_len;
+ int flags;
+ int tag;
+ int length;
+ char* next_tag;
+ char* end;
+
+ in_uint32_le(s, self->rc4_key_size); /* 1 = 40-bit, 2 = 128-bit */
+ in_uint32_le(s, self->crypt_level); /* 1 = low, 2 = medium, 3 = high */
+ if (self->crypt_level == 0) /* no encryption */
+ {
+ return 0;
+ }
+ in_uint32_le(s, random_len);
+ in_uint32_le(s, rsa_info_len);
+ if (random_len != SEC_RANDOM_SIZE)
+ {
+ return 0;
+ }
+ in_uint8a(s, self->server_random, random_len);
+ /* RSA info */
+ end = s->p + rsa_info_len;
+ if (end > s->end)
+ {
+ return 0;
+ }
+ in_uint32_le(s, flags); /* 1 = RDP4-style, 0x80000002 = X.509 */
+ if (flags & 1)
+ {
+ in_uint8s(s, 8); /* unknown */
+ while (s->p < end)
+ {
+ in_uint16_le(s, tag);
+ in_uint16_le(s, length);
+ next_tag = s->p + length;
+ DEBUG((" rdp_sec_parse_crypt_info tag %d length %d\r\n", tag, length));
+ switch (tag)
+ {
+ case SEC_TAG_PUBKEY:
+ if (!rdp_sec_parse_public_key(self, s, modulus, exponent))
+ {
+ return 0;
+ }
+ break;
+ case SEC_TAG_KEYSIG:
+ break;
+ default:
+ break;
+ }
+ s->p = next_tag;
+ }
+ }
+ else
+ {
+ /* todo */
+ return 0;
+ }
+ return s_check_end(s);
+}
+
+/*****************************************************************************/
+static void APP_CC
+rdp_sec_rsa_op(char* out, char* in, char* mod, char* exp)
+{
+ g_mod_exp(out, SEC_MODULUS_SIZE, /* 64 */
+ in, SEC_RANDOM_SIZE, /* 32 */
+ mod, SEC_MODULUS_SIZE, /* 64 */
+ exp, SEC_EXPONENT_SIZE); /* 4 */
+ //g_hexdump(out, SEC_MODULUS_SIZE);
+ //g_hexdump(in, SEC_RANDOM_SIZE);
+ //g_hexdump(mod, SEC_MODULUS_SIZE);
+ //g_hexdump(exp, SEC_EXPONENT_SIZE);
+}
+
+/*****************************************************************************/
+void APP_CC
+rdp_sec_hash_48(char* out, char* in, char* salt1, char* salt2, int salt)
+{
+ int i;
+ void* sha1_info;
+ void* md5_info;
+ char pad[4];
+ char sha1_sig[20];
+ char md5_sig[16];
+
+ sha1_info = g_sha1_info_create();
+ md5_info = g_md5_info_create();
+ for (i = 0; i < 3; i++)
+ {
+ g_memset(pad, salt + i, 4);
+ g_sha1_clear(sha1_info);
+ g_sha1_transform(sha1_info, pad, i + 1);
+ g_sha1_transform(sha1_info, in, 48);
+ g_sha1_transform(sha1_info, salt1, 32);
+ g_sha1_transform(sha1_info, salt2, 32);
+ g_sha1_complete(sha1_info, sha1_sig);
+ g_md5_clear(md5_info);
+ g_md5_transform(md5_info, in, 48);
+ g_md5_transform(md5_info, sha1_sig, 20);
+ g_md5_complete(md5_info, md5_sig);
+ g_memcpy(out + i * 16, md5_sig, 16);
+ }
+ g_sha1_info_delete(sha1_info);
+ g_md5_info_delete(md5_info);
+}
+
+/*****************************************************************************/
+void APP_CC
+rdp_sec_hash_16(char* out, char* in, char* salt1, char* salt2)
+{
+ void* md5_info;
+
+ md5_info = g_md5_info_create();
+ g_md5_clear(md5_info);
+ g_md5_transform(md5_info, in, 16);
+ g_md5_transform(md5_info, salt1, 32);
+ g_md5_transform(md5_info, salt2, 32);
+ g_md5_complete(md5_info, out);
+ g_md5_info_delete(md5_info);
+}
+
+/*****************************************************************************/
+static int APP_CC
+rdp_sec_generate_keys(struct rdp_sec* self)
+{
+ char session_key[48];
+ char temp_hash[48];
+ char input[48];
+
+ g_memcpy(input, self->client_random, 24);
+ g_memcpy(input + 24, self->server_random, 24);
+ rdp_sec_hash_48(temp_hash, input, self->client_random,
+ self->server_random, 65);
+ rdp_sec_hash_48(session_key, temp_hash, self->client_random,
+ self->server_random, 88);
+ g_memcpy(self->sign_key, session_key, 16);
+ rdp_sec_hash_16(self->decrypt_key, session_key + 16, self->client_random,
+ self->server_random);
+ rdp_sec_hash_16(self->encrypt_key, session_key + 32, self->client_random,
+ self->server_random);
+ DEBUG((" rdp_sec_generate_keys, rc4_key_size is %d\r\n", self->rc4_key_size));
+ DEBUG((" rdp_sec_generate_keys, crypt_level is %d\r\n", self->crypt_level));
+ if (self->rc4_key_size == 1)
+ {
+ rdp_sec_make_40bit(self->sign_key);
+ rdp_sec_make_40bit(self->encrypt_key);
+ rdp_sec_make_40bit(self->decrypt_key);
+ self->rc4_key_len = 8;
+ }
+ else
+ {
+ self->rc4_key_len = 16;
+ }
+ g_memcpy(self->decrypt_update_key, self->decrypt_key, 16);
+ g_memcpy(self->encrypt_update_key, self->encrypt_key, 16);
+ g_rc4_set_key(self->decrypt_rc4_info, self->decrypt_key, self->rc4_key_len);
+ g_rc4_set_key(self->encrypt_rc4_info, self->encrypt_key, self->rc4_key_len);
+ return 0;
+}
+
+/*****************************************************************************/
+/* Process crypto information blob */
+static void APP_CC
+rdp_sec_process_crypt_info(struct rdp_sec* self, struct stream* s)
+{
+ char modulus[64];
+ char exponent[64];
+
+ g_memset(modulus, 0, sizeof(modulus));
+ g_memset(exponent, 0, sizeof(exponent));
+ if (!rdp_sec_parse_crypt_info(self, s, modulus, exponent))
+ {
+ DEBUG((" error in rdp_sec_process_crypt_info\r\n"));
+ return;
+ }
+ /* Generate a client random, and determine encryption keys */
+ g_random(self->client_random, 32);
+ rdp_sec_rsa_op(self->client_crypt_random, self->client_random,
+ modulus, exponent);
+ rdp_sec_generate_keys(self);
+}
+
+/*****************************************************************************/
+/* Process connect response data blob */
+static void APP_CC
+rdp_sec_process_mcs_data(struct rdp_sec* self)
+{
+ int tag;
+ int length;
+ int len;
+ char* next_tag;
+ struct stream* s;
+
+ s = self->server_mcs_data;
+ s->p = s->data;
+ in_uint8s(s, 21); /* header (T.124 ConferenceCreateResponse) */
+ in_uint8(s, len);
+ if (len & 0x80)
+ {
+ in_uint8(s, len);
+ }
+ while (s->p < s->end)
+ {
+ in_uint16_le(s, tag);
+ in_uint16_le(s, length);
+ DEBUG((" rdp_sec_process_mcs_data tag %d length %d\r\n", tag, length));
+ if (length <= 4)
+ {
+ return;
+ }
+ next_tag = (s->p + length) - 4;
+ switch (tag)
+ {
+ case SEC_TAG_SRV_INFO:
+ //rdp_sec_process_srv_info(self, s);
+ break;
+ case SEC_TAG_SRV_CRYPT:
+ rdp_sec_process_crypt_info(self, s);
+ break;
+ case SEC_TAG_SRV_CHANNELS:
+ break;
+ default:
+ break;
+ }
+ s->p = next_tag;
+ }
+}
+
+/*****************************************************************************/
+/* Transfer the client random to the server */
+/* returns error */
+static int APP_CC
+rdp_sec_establish_key(struct rdp_sec* self)
+{
+ int length;
+ int flags;
+ struct stream* s;
+
+ DEBUG((" sending client random\r\n"));
+ make_stream(s);
+ init_stream(s, 8192);
+ length = SEC_MODULUS_SIZE + SEC_PADDING_SIZE;
+ flags = SEC_CLIENT_RANDOM;
+ if (rdp_sec_init(self, s, flags) != 0)
+ {
+ free_stream(s);
+ return 1;
+ }
+ out_uint32_le(s, length);
+ out_uint8p(s, self->client_crypt_random, SEC_MODULUS_SIZE);
+ out_uint8s(s, SEC_PADDING_SIZE);
+ s_mark_end(s);
+ if (rdp_sec_send(self, s, flags) != 0)
+ {
+ free_stream(s);
+ return 1;
+ }
+ free_stream(s);
+ return 0;
+}
+
+/*****************************************************************************/
+/* Establish a secure connection */
+int APP_CC
+rdp_sec_connect(struct rdp_sec* self, char* ip, char* port)
+{
+ DEBUG((" in rdp_sec_connect\r\n"));
+ rdp_sec_out_mcs_data(self);
+ if (rdp_mcs_connect(self->mcs_layer, ip, port) != 0)
+ {
+ DEBUG((" out rdp_sec_connect error rdp_mcs_connect failed\r\n"));
+ return 1;
+ }
+ rdp_sec_process_mcs_data(self);
+ if (rdp_sec_establish_key(self) != 0)
+ {
+ DEBUG((" out rdp_sec_connect error rdp_sec_establish_key failed\r\n"));
+ return 1;
+ }
+ DEBUG((" out rdp_sec_connect\r\n"));
+ return 0;
+}
+
+/*****************************************************************************/
+/* returns error */
+int APP_CC
+rdp_sec_init(struct rdp_sec* self, struct stream* s, int flags)
+{
+ if (rdp_mcs_init(self->mcs_layer, s) != 0)
+ {
+ return 1;
+ }
+ if (flags & SEC_ENCRYPT)
+ {
+ s_push_layer(s, sec_hdr, 12);
+ }
+ else
+ {
+ s_push_layer(s, sec_hdr, 4);
+ }
+ return 0;
+}
+
+/*****************************************************************************/
+/* Output a uint32 into a buffer (little-endian) */
+void APP_CC
+rdp_sec_buf_out_uint32(char* buffer, int value)
+{
+ buffer[0] = value & 0xff;
+ buffer[1] = (value >> 8) & 0xff;
+ buffer[2] = (value >> 16) & 0xff;
+ buffer[3] = (value >> 24) & 0xff;
+}
+
+/*****************************************************************************/
+/* Generate a MAC hash (5.2.3.1), using a combination of SHA1 and MD5 */
+void APP_CC
+rdp_sec_sign(char* signature, int siglen, char* session_key, int keylen,
+ char* data, int datalen)
+{
+ char shasig[20];
+ char md5sig[16];
+ char lenhdr[4];
+ void* sha1_context;
+ void* md5_context;
+
+ rdp_sec_buf_out_uint32(lenhdr, datalen);
+ sha1_context = g_sha1_info_create();
+ g_sha1_clear(sha1_context);
+ g_sha1_transform(sha1_context, session_key, keylen);
+ g_sha1_transform(sha1_context, g_pad_54, 40);
+ g_sha1_transform(sha1_context, lenhdr, 4);
+ g_sha1_transform(sha1_context, data, datalen);
+ g_sha1_complete(sha1_context, shasig);
+ g_sha1_info_delete(sha1_context);
+ md5_context = g_md5_info_create();
+ g_md5_clear(md5_context);
+ g_md5_transform(md5_context, session_key, keylen);
+ g_md5_transform(md5_context, g_pad_92, 48);
+ g_md5_transform(md5_context, shasig, 20);
+ g_md5_complete(md5_context, md5sig);
+ g_md5_info_delete(md5_context);
+ g_memcpy(signature, md5sig, siglen);
+}
+
+/*****************************************************************************/
+/* Encrypt data using RC4 */
+static void APP_CC
+rdp_sec_encrypt(struct rdp_sec* self, char* data, int length)
+{
+ if (self->encrypt_use_count == 4096)
+ {
+ rdp_sec_update(self->encrypt_key, self->encrypt_update_key,
+ self->rc4_key_len);
+ g_rc4_set_key(self->encrypt_rc4_info, self->encrypt_key,
+ self->rc4_key_len);
+ self->encrypt_use_count = 0;
+ }
+ g_rc4_crypt(self->encrypt_rc4_info, data, length);
+ self->encrypt_use_count++;
+}
+
+/*****************************************************************************/
+/* returns error */
+int APP_CC
+rdp_sec_send(struct rdp_sec* self, struct stream* s, int flags)
+{
+ int datalen;
+
+ DEBUG((" in rdp_sec_send flags %8.8x\r\n", flags));
+ s_pop_layer(s, sec_hdr);
+ out_uint32_le(s, flags);
+ if (flags & SEC_ENCRYPT)
+ {
+ datalen = (s->end - s->p) - 8;
+ rdp_sec_sign(s->p, 8, self->sign_key, self->rc4_key_len, s->p + 8,
+ datalen);
+ rdp_sec_encrypt(self, s->p + 8, datalen);
+ }
+ if (rdp_mcs_send(self->mcs_layer, s) != 0)
+ {
+ DEBUG((" out rdp_sec_send, rdp_mcs_send failed\r\n"));
+ return 1;
+ }
+ DEBUG((" out rdp_sec_send\r\n"));
+ return 0;
+}