diff options
Diffstat (limited to 'libxrdp/xrdp_rdp.c')
-rw-r--r-- | libxrdp/xrdp_rdp.c | 847 |
1 files changed, 847 insertions, 0 deletions
diff --git a/libxrdp/xrdp_rdp.c b/libxrdp/xrdp_rdp.c new file mode 100644 index 00000000..155e109c --- /dev/null +++ b/libxrdp/xrdp_rdp.c @@ -0,0 +1,847 @@ +/* + 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 2004-2005 + + rdp layer + +*/ + +#include "libxrdp.h" + +static char unknown1[172] = +{ 0xff, 0x02, 0xb6, 0x00, 0x28, 0x00, 0x00, 0x00, + 0x27, 0x00, 0x27, 0x00, 0x03, 0x00, 0x04, 0x00, + 0x00, 0x00, 0x26, 0x00, 0x01, 0x00, 0x1e, 0x00, + 0x02, 0x00, 0x1f, 0x00, 0x03, 0x00, 0x1d, 0x00, + 0x04, 0x00, 0x27, 0x00, 0x05, 0x00, 0x0b, 0x00, + 0x06, 0x00, 0x28, 0x00, 0x08, 0x00, 0x21, 0x00, + 0x09, 0x00, 0x20, 0x00, 0x0a, 0x00, 0x22, 0x00, + 0x0b, 0x00, 0x25, 0x00, 0x0c, 0x00, 0x24, 0x00, + 0x0d, 0x00, 0x23, 0x00, 0x0e, 0x00, 0x19, 0x00, + 0x0f, 0x00, 0x16, 0x00, 0x10, 0x00, 0x15, 0x00, + 0x11, 0x00, 0x1c, 0x00, 0x12, 0x00, 0x1b, 0x00, + 0x13, 0x00, 0x1a, 0x00, 0x14, 0x00, 0x17, 0x00, + 0x15, 0x00, 0x18, 0x00, 0x16, 0x00, 0x0e, 0x00, + 0x18, 0x00, 0x0c, 0x00, 0x19, 0x00, 0x0d, 0x00, + 0x1a, 0x00, 0x12, 0x00, 0x1b, 0x00, 0x14, 0x00, + 0x1f, 0x00, 0x13, 0x00, 0x20, 0x00, 0x00, 0x00, + 0x21, 0x00, 0x0a, 0x00, 0x22, 0x00, 0x06, 0x00, + 0x23, 0x00, 0x07, 0x00, 0x24, 0x00, 0x08, 0x00, + 0x25, 0x00, 0x09, 0x00, 0x26, 0x00, 0x04, 0x00, + 0x27, 0x00, 0x03, 0x00, 0x28, 0x00, 0x02, 0x00, + 0x29, 0x00, 0x01, 0x00, 0x2a, 0x00, 0x05, 0x00, + 0x2b, 0x00, 0x2a, 0x00 }; + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_read_config(struct xrdp_client_info* client_info) +{ + int fd; + int index; + struct list* items; + struct list* values; + char* item; + char* value; + + fd = g_file_open("xrdp.ini"); + if (fd > 0) + { + items = list_create(); + items->auto_free = 1; + values = list_create(); + values->auto_free = 1; + file_read_section(fd, "globals", items, values); + for (index = 0; index < items->count; index++) + { + item = (char*)list_get_item(items, index); + value = (char*)list_get_item(values, index); + if (g_strcmp(item, "bitmap_cache") == 0) + { + if (g_strcmp(value, "yes") == 0) + { + client_info->use_bitmap_cache = 1; + } + } + else if (g_strcmp(item, "bitmap_compression") == 0) + { + if (g_strcmp(value, "yes") == 0) + { + client_info->use_bitmap_comp = 1; + } + } + } + list_delete(items); + list_delete(values); + g_file_close(fd); + } + return 0; +} + +/*****************************************************************************/ +struct xrdp_rdp* APP_CC +xrdp_rdp_create(struct xrdp_session* session, int sck) +{ + struct xrdp_rdp* self; + + self = (struct xrdp_rdp*)g_malloc(sizeof(struct xrdp_rdp), 1); + self->session = session; + self->share_id = 66538; + self->sec_layer = xrdp_sec_create(self, sck); + /* read ini settings */ + xrdp_rdp_read_config(&self->client_info); + return self; +} + +/*****************************************************************************/ +void APP_CC +xrdp_rdp_delete(struct xrdp_rdp* self) +{ + if (self == 0) + { + return; + } + xrdp_sec_delete(self->sec_layer); + g_free(self); +} + +/*****************************************************************************/ +int APP_CC +xrdp_rdp_init(struct xrdp_rdp* self, struct stream* s) +{ + if (xrdp_sec_init(self->sec_layer, s) != 0) + { + return 1; + } + s_push_layer(s, rdp_hdr, 6); + return 0; +} + +/*****************************************************************************/ +int APP_CC +xrdp_rdp_init_data(struct xrdp_rdp* self, struct stream* s) +{ + if (xrdp_sec_init(self->sec_layer, s) != 0) + { + return 1; + } + s_push_layer(s, rdp_hdr, 18); + return 0; +} + +/*****************************************************************************/ +/* returns erros */ +int APP_CC +xrdp_rdp_recv(struct xrdp_rdp* self, struct stream* s, int* code) +{ + int error; + int len; + int pdu_code; + int chan; + + DEBUG(("in xrdp_rdp_recv\n\r")); + if (s->next_packet == 0 || s->next_packet >= s->end) + { + chan = 0; + error = xrdp_sec_recv(self->sec_layer, s, &chan); + if (error == -1) /* special code for send demand active */ + { + s->next_packet = 0; + *code = -1; + DEBUG(("out xrdp_rdp_recv\n\r")); + return 0; + } + if (error != 0) + { + DEBUG(("out xrdp_rdp_recv error\n\r")); + return 1; + } + if (chan != MCS_GLOBAL_CHANNEL && chan > 0) + { + s->next_packet = 0; + *code = 0; + DEBUG(("out xrdp_rdp_recv\n\r")); + return 0; + } + s->next_packet = s->p; + } + else + { + s->p = s->next_packet; + } + in_uint16_le(s, len); + if (len == 0x8000) + { + s->next_packet += 8; + *code = 0; + DEBUG(("out xrdp_rdp_recv\n\r")); + return 0; + } + in_uint16_le(s, pdu_code); + *code = pdu_code & 0xf; + in_uint8s(s, 2); /* mcs user id */ + s->next_packet += len; + DEBUG(("out xrdp_rdp_recv\n\r")); + return 0; +} + +/*****************************************************************************/ +int APP_CC +xrdp_rdp_send(struct xrdp_rdp* self, struct stream* s, int pdu_type) +{ + int len; + + DEBUG(("in xrdp_rdp_send\n\r")); + s_pop_layer(s, rdp_hdr); + len = s->end - s->p; + out_uint16_le(s, len); + out_uint16_le(s, 0x10 | pdu_type); + out_uint16_le(s, self->mcs_channel); + if (xrdp_sec_send(self->sec_layer, s, 0) != 0) + { + DEBUG(("out xrdp_rdp_send error\n\r")); + return 1; + } + DEBUG(("out xrdp_rdp_send\n\r")); + return 0; +} + +/*****************************************************************************/ +int APP_CC +xrdp_rdp_send_data(struct xrdp_rdp* self, struct stream* s, + int data_pdu_type) +{ + int len; + + DEBUG(("in xrdp_rdp_send_data\n\r")); + s_pop_layer(s, rdp_hdr); + len = s->end - s->p; + out_uint16_le(s, len); + out_uint16_le(s, 0x10 | RDP_PDU_DATA); + out_uint16_le(s, self->mcs_channel); + out_uint32_le(s, self->share_id); + out_uint8(s, 0); + out_uint8(s, 1); + out_uint16_le(s, len - 14); + out_uint8(s, data_pdu_type); + out_uint8(s, 0); + out_uint16_le(s, 0); + if (xrdp_sec_send(self->sec_layer, s, 0) != 0) + { + DEBUG(("out xrdp_rdp_send_data error\n\r")); + return 1; + } + DEBUG(("out xrdp_rdp_send_data\n\r")); + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_parse_client_mcs_data(struct xrdp_rdp* self) +{ + struct stream* p; + int i; + + p = &self->sec_layer->client_mcs_data; + p->p = p->data; + in_uint8s(p, 31); + in_uint16_le(p, self->client_info.width); + in_uint16_le(p, self->client_info.height); + in_uint8s(p, 120); + self->client_info.bpp = 8; + in_uint16_le(p, i); + switch (i) + { + case 0xca01: + in_uint8s(p, 6); + in_uint8(p, i); + if (i > 8) + { + self->client_info.bpp = i; + } + break; + case 0xca02: + self->client_info.bpp = 15; + break; + case 0xca03: + self->client_info.bpp = 16; + break; + case 0xca04: + self->client_info.bpp = 24; + break; + } + p->p = p->data; + DEBUG(("client width %d, client height %d bpp %d\n\r", + self->client_info.width, self->client_info.height, + self->client_info.bpp)); + return 0; +} + +/*****************************************************************************/ +int APP_CC +xrdp_rdp_incoming(struct xrdp_rdp* self) +{ + DEBUG(("in xrdp_rdp_incoming\n\r")); + if (xrdp_sec_incoming(self->sec_layer) != 0) + { + return 1; + } + self->mcs_channel = self->sec_layer->mcs_layer->userid + + MCS_USERCHANNEL_BASE; + xrdp_rdp_parse_client_mcs_data(self); + DEBUG(("out xrdp_rdp_incoming mcs channel %d\n\r", self->mcs_channel)); + return 0; +} + +/*****************************************************************************/ +int APP_CC +xrdp_rdp_send_demand_active(struct xrdp_rdp* self) +{ + struct stream* s; + + make_stream(s); + init_stream(s, 8192); + if (xrdp_rdp_init(self, s) != 0) + { + free_stream(s); + return 1; + } + + out_uint32_le(s, self->share_id); + out_uint32_be(s, 0x04000401); + out_uint32_be(s, 0x524e5300); + out_uint32_be(s, 0x08000000); + out_uint32_be(s, 0x09000800); + out_uint16_le(s, self->mcs_channel); + out_uint16_be(s, 0xb5e2); + + /* Output general capability set */ + out_uint16_le(s, RDP_CAPSET_GENERAL); /* 1 */ + out_uint16_le(s, RDP_CAPLEN_GENERAL); /* 24(0x18) */ + out_uint16_le(s, 1); /* OS major type */ + out_uint16_le(s, 3); /* OS minor type */ + out_uint16_le(s, 0x200); /* Protocol version */ + out_uint16_le(s, 0); /* pad */ + out_uint16_le(s, 0); /* Compression types */ + out_uint16_le(s, 0); /* pad */ + out_uint16_le(s, 0); /* Update capability */ + out_uint16_le(s, 0); /* Remote unshare capability */ + out_uint16_le(s, 0); /* Compression level */ + out_uint16_le(s, 0); /* Pad */ + + /* Output bitmap capability set */ + out_uint16_le(s, RDP_CAPSET_BITMAP); /* 2 */ + out_uint16_le(s, RDP_CAPLEN_BITMAP); /* 28(0x1c) */ + out_uint16_le(s, self->client_info.bpp); /* Preferred BPP */ + out_uint16_le(s, 1); /* Receive 1 BPP */ + out_uint16_le(s, 1); /* Receive 4 BPP */ + out_uint16_le(s, 1); /* Receive 8 BPP */ + out_uint16_le(s, self->client_info.width); /* width */ + out_uint16_le(s, self->client_info.height); /* height */ + out_uint16_le(s, 0); /* Pad */ + out_uint16_le(s, 1); /* Allow resize */ + out_uint16_le(s, 1); /* bitmap compression */ + out_uint16_le(s, 0); /* unknown */ + out_uint16_le(s, 0); /* unknown */ + out_uint16_le(s, 0); /* pad */ + + /* Output ? */ + out_uint16_le(s, 14); + out_uint16_le(s, 4); + + /* Output order capability set */ + out_uint16_le(s, RDP_CAPSET_ORDER); /* 3 */ + out_uint16_le(s, RDP_CAPLEN_ORDER); /* 88(0x58) */ + out_uint8s(s, 16); + out_uint32_be(s, 0x40420f00); + out_uint16_le(s, 1); /* Cache X granularity */ + out_uint16_le(s, 20); /* Cache Y granularity */ + out_uint16_le(s, 0); /* Pad */ + out_uint16_le(s, 1); /* Max order level */ + out_uint16_le(s, 0x2f); /* Number of fonts */ + out_uint16_le(s, 0x22); /* Capability flags */ + /* caps */ + out_uint8(s, 1); /* dest blt */ + out_uint8(s, 1); /* pat blt */ + out_uint8(s, 1); /* screen blt */ + out_uint8(s, 1); /* memblt */ + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 0); + out_uint8(s, 0); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 1); + out_uint8(s, 0); + out_uint8(s, 0); + out_uint8(s, 0); + out_uint16_le(s, 0x6a1); + out_uint8s(s, 6); /* ? */ + out_uint32_le(s, 0x0f4240); /* desk save */ + out_uint32_le(s, 0); /* ? */ + out_uint32_le(s, 0); /* ? */ + + /* Output color cache capability set */ + out_uint16_le(s, RDP_CAPSET_COLCACHE); + out_uint16_le(s, RDP_CAPLEN_COLCACHE); + out_uint16_le(s, 6); /* cache size */ + out_uint16_le(s, 0); /* pad */ + + /* Output pointer capability set */ + out_uint16_le(s, RDP_CAPSET_POINTER); + out_uint16_le(s, RDP_CAPLEN_POINTER); + out_uint16_le(s, 1); /* Colour pointer */ + out_uint16_le(s, 0x19); /* Cache size */ + + /* Output ? */ + out_uint16_le(s, 0xd); + out_uint16_le(s, 0x58); /* 88 */ + out_uint8(s, 1); + out_uint8s(s, 83); + + s_mark_end(s); + + if (xrdp_rdp_send(self, s, RDP_PDU_DEMAND_ACTIVE) != 0) + { + free_stream(s); + return 1; + } + + free_stream(s); + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_process_capset_general(struct xrdp_rdp* self, struct stream* s, + int len) +{ + int i; + + in_uint8s(s, 10); + in_uint16_le(s, i); + self->client_info.use_compact_packets = (i != 0); + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_process_capset_order(struct xrdp_rdp* self, struct stream* s, + int len) +{ + int i; + + in_uint8s(s, 72); + in_uint32_le(s, i); /* desktop cache size, usually 0x38400 */ + self->client_info.desktop_cache = i; + return 0; +} + +/*****************************************************************************/ +/* get the bitmap cache size */ +static int APP_CC +xrdp_process_capset_bmpcache(struct xrdp_rdp* self, struct stream* s, + int len) +{ + in_uint8s(s, 24); + in_uint16_le(s, self->client_info.cache1_entries); + in_uint16_le(s, self->client_info.cache1_size); + in_uint16_le(s, self->client_info.cache2_entries); + in_uint16_le(s, self->client_info.cache2_size); + in_uint16_le(s, self->client_info.cache3_entries); + in_uint16_le(s, self->client_info.cache3_size); + return 0; +} + +/*****************************************************************************/ +/* get the number of client cursor cache */ +static int APP_CC +xrdp_process_capset_pointercache(struct xrdp_rdp* self, struct stream* s, + int len) +{ + int i; + + in_uint8s(s, 2); /* color pointer */ + in_uint16_le(s, i); + i = MIN(i, 32); + self->client_info.pointer_cache_entries = i; + return 0; +} + +/*****************************************************************************/ +int APP_CC +xrdp_rdp_process_confirm_active(struct xrdp_rdp* self, struct stream* s) +{ + int cap_len; + int source_len; + int num_caps; + int index; + int type; + int len; + char* p; + + in_uint8s(s, 4); /* rdp_shareid */ + in_uint8s(s, 2); /* userid */ + in_uint16_le(s, source_len); /* sizeof RDP_SOURCE */ + in_uint16_le(s, cap_len); + in_uint8s(s, source_len); + in_uint16_le(s, num_caps); + in_uint8s(s, 2); /* pad */ + for (index = 0; index < num_caps; index++) + { + p = s->p; + in_uint16_le(s, type); + in_uint16_le(s, len); + switch (type) + { + case RDP_CAPSET_GENERAL: /* 1 */ + xrdp_process_capset_general(self, s, len); + break; + case RDP_CAPSET_ORDER: /* 3 */ + xrdp_process_capset_order(self, s, len); + break; + case RDP_CAPSET_BMPCACHE: /* 4 */ + xrdp_process_capset_bmpcache(self, s, len); + break; + case RDP_CAPSET_POINTER: /* 8 */ + xrdp_process_capset_pointercache(self, s, len); + break; + } + s->p = p + len; + } + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_process_data_pointer(struct xrdp_rdp* self, struct stream* s) +{ + return 0; +} + +/*****************************************************************************/ +/* RDP_DATA_PDU_INPUT */ +static int APP_CC +xrdp_rdp_process_data_input(struct xrdp_rdp* self, struct stream* s) +{ + int num_events; + int index; + int msg_type; + int device_flags; + int param1; + int param2; + int time; + + in_uint16_le(s, num_events); + in_uint8s(s, 2); /* pad */ + DEBUG(("xrdp_rdp_process_data_input %d events\n\r", num_events)); + for (index = 0; index < num_events; index++) + { + in_uint32_le(s, time); + in_uint16_le(s, msg_type); + in_uint16_le(s, device_flags); + in_sint16_le(s, param1); + in_sint16_le(s, param2); + DEBUG(("xrdp_rdp_process_data_input event %4.4x flags %4.4x param1 %d \ +param2 %d time %d\n\r", msg_type, device_flags, param1, param2, time)); + if (self->session->callback != 0) + { + /* msg_type can be + RDP_INPUT_SYNCHRONIZE - 0 + RDP_INPUT_SCANCODE - 4 + RDP_INPUT_MOUSE - 0x8001 */ + self->session->callback(self->session->id, msg_type, param1, param2, + device_flags, time); + } + } + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_send_synchronise(struct xrdp_rdp* self) +{ + struct stream* s; + + make_stream(s); + init_stream(s, 8192); + if (xrdp_rdp_init_data(self, s) != 0) + { + free_stream(s); + return 1; + } + out_uint16_le(s, 1); + out_uint16_le(s, 1002); + s_mark_end(s); + if (xrdp_rdp_send_data(self, s, RDP_DATA_PDU_SYNCHRONISE) != 0) + { + free_stream(s); + return 1; + } + free_stream(s); + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_send_control(struct xrdp_rdp* self, int action) +{ + struct stream* s; + + make_stream(s); + init_stream(s, 8192); + if (xrdp_rdp_init_data(self, s) != 0) + { + free_stream(s); + return 1; + } + out_uint16_le(s, action); + out_uint16_le(s, 0); /* userid */ + out_uint32_le(s, 1002); /* control id */ + s_mark_end(s); + if (xrdp_rdp_send_data(self, s, RDP_DATA_PDU_CONTROL) != 0) + { + free_stream(s); + return 1; + } + free_stream(s); + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_process_data_control(struct xrdp_rdp* self, struct stream* s) +{ + int action; + + in_uint16_le(s, action); + in_uint8s(s, 2); /* user id */ + in_uint8s(s, 4); /* control id */ + if (action == RDP_CTL_REQUEST_CONTROL) + { + xrdp_rdp_send_synchronise(self); + xrdp_rdp_send_control(self, RDP_CTL_COOPERATE); + xrdp_rdp_send_control(self, RDP_CTL_GRANT_CONTROL); + } + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_process_data_sync(struct xrdp_rdp* self) +{ + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_process_screen_update(struct xrdp_rdp* self, struct stream* s) +{ + int op; + int left; + int top; + int right; + int bottom; + int cx; + int cy; + + in_uint32_le(s, op); + in_uint16_le(s, left); + in_uint16_le(s, top); + in_uint16_le(s, right); + in_uint16_le(s, bottom); + cx = (right - left) + 1; + cy = (bottom - top) + 1; + if (self->session->callback != 0) + { + self->session->callback(self->session->id, 0x4444, left, top, cx, cy); + } + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_send_unknown1(struct xrdp_rdp* self) +{ + struct stream* s; + + make_stream(s); + init_stream(s, 8192); + if (xrdp_rdp_init_data(self, s) != 0) + { + free_stream(s); + return 1; + } + out_uint8a(s, unknown1, 172); + s_mark_end(s); + if (xrdp_rdp_send_data(self, s, 0x28) != 0) + { + free_stream(s); + return 1; + } + free_stream(s); + return 0; +} + +/*****************************************************************************/ +static int APP_CC +xrdp_rdp_process_data_font(struct xrdp_rdp* self, struct stream* s) +{ + int seq; + + in_uint8s(s, 2); /* num of fonts */ + in_uint8s(s, 2); /* unknown */ + in_uint16_le(s, seq); + /* 419 client sends Seq 1, then 2 */ + /* 2600 clients sends only Seq 3 */ + if (seq == 2 || seq == 3) /* after second font message, we are up and */ + { /* running */ + xrdp_rdp_send_unknown1(self); + self->session->up_and_running = 1; + } + return 0; +} + +/*****************************************************************************/ +/* sent 37 pdu */ +static int APP_CC +xrdp_rdp_send_disconnect_query_response(struct xrdp_rdp* self) +{ + struct stream* s; + + make_stream(s); + init_stream(s, 8192); + if (xrdp_rdp_init_data(self, s) != 0) + { + free_stream(s); + return 1; + } + s_mark_end(s); + if (xrdp_rdp_send_data(self, s, 37) != 0) + { + free_stream(s); + return 1; + } + free_stream(s); + return 0; +} + +#if 0 /* not used */ +/*****************************************************************************/ +/* sent RDP_DATA_PDU_DISCONNECT 47 pdu */ +static int APP_CC +xrdp_rdp_send_disconnect_reason(struct xrdp_rdp* self, int reason) +{ + struct stream* s; + + make_stream(s); + init_stream(s, 8192); + if (xrdp_rdp_init_data(self, s) != 0) + { + free_stream(s); + return 1; + } + out_uint32_le(s, reason); + s_mark_end(s); + if (xrdp_rdp_send_data(self, s, RDP_DATA_PDU_DISCONNECT) != 0) + { + free_stream(s); + return 1; + } + free_stream(s); + return 0; +} +#endif + +/*****************************************************************************/ +/* RDP_PDU_DATA */ +int APP_CC +xrdp_rdp_process_data(struct xrdp_rdp* self, struct stream* s) +{ + int len; + int data_type; + int ctype; + int clen; + + in_uint8s(s, 6); + in_uint16_le(s, len); + in_uint8(s, data_type); + in_uint8(s, ctype); + in_uint16_le(s, clen); + DEBUG(("xrdp_rdp_process_data code %d\n\r", data_type)); + switch (data_type) + { + case RDP_DATA_PDU_POINTER: /* 27 */ + xrdp_rdp_process_data_pointer(self, s); + break; + case RDP_DATA_PDU_INPUT: /* 28 */ + xrdp_rdp_process_data_input(self, s); + break; + case RDP_DATA_PDU_CONTROL: /* 20 */ + xrdp_rdp_process_data_control(self, s); + break; + case RDP_DATA_PDU_SYNCHRONISE: /* 31 */ + xrdp_rdp_process_data_sync(self); + break; + case 33: /* 33 ?? Invalidate an area I think */ + xrdp_rdp_process_screen_update(self, s); + break; + case 35: + /* 35 ?? this comes when minimuzing a full screen mstsc.exe 2600 */ + /* I think this is saying the client no longer wants screen */ + /* updates and it will issue a 33 above to catch up */ + /* so minimized apps don't take bandwidth */ + break; + case 36: /* 36 ?? disconnect query? */ + /* when this message comes, send a 37 back so the client */ + /* is sure the connection is alive and it can ask if user */ + /* really wants to disconnect */ + xrdp_rdp_send_disconnect_query_response(self); /* send a 37 back */ + break; + case RDP_DATA_PDU_FONT2: /* 39 */ + xrdp_rdp_process_data_font(self, s); + break; + default: + g_printf("unknown in xrdp_rdp_process_data %d\n\r", data_type); + break; + } + return 0; +} + +/*****************************************************************************/ +int APP_CC +xrdp_rdp_disconnect(struct xrdp_rdp* self) +{ + return xrdp_sec_disconnect(self->sec_layer); +} |