📄 rdp.c
字号:
rdp_send_confirm_active(); rdp_send_synchronise(); rdp_send_control(RDP_CTL_COOPERATE); rdp_send_control(RDP_CTL_REQUEST_CONTROL); rdp_recv(&type); /* RDP_PDU_SYNCHRONIZE */ rdp_recv(&type); /* RDP_CTL_COOPERATE */ rdp_recv(&type); /* RDP_CTL_GRANT_CONTROL */ rdp_send_input(0, RDP_INPUT_SYNCHRONIZE, 0, g_numlock_sync ? ui_get_numlock_state(read_keyboard_state()) : 0, 0); if (g_use_rdp5) { rdp_enum_bmpcache2(); rdp_send_fonts(3); } else { rdp_send_fonts(1); rdp_send_fonts(2); } rdp_recv(&type); /* RDP_PDU_UNKNOWN 0x28 (Fonts?) */ reset_order_state();}/* Process a colour pointer PDU */voidprocess_colour_pointer_pdu(STREAM s){ uint16 x, y, width, height, cache_idx, masklen, datalen; uint8 *mask, *data; HCURSOR cursor; in_uint16_le(s, cache_idx); in_uint16_le(s, x); in_uint16_le(s, y); in_uint16_le(s, width); in_uint16_le(s, height); in_uint16_le(s, masklen); in_uint16_le(s, datalen); in_uint8p(s, data, datalen); in_uint8p(s, mask, masklen); cursor = ui_create_cursor(x, y, width, height, mask, data); ui_set_cursor(cursor); cache_put_cursor(cache_idx, cursor);}/* Process a cached pointer PDU */voidprocess_cached_pointer_pdu(STREAM s){ uint16 cache_idx; in_uint16_le(s, cache_idx); ui_set_cursor(cache_get_cursor(cache_idx));}/* Process a system pointer PDU */voidprocess_system_pointer_pdu(STREAM s){ uint16 system_pointer_type; in_uint16(s, system_pointer_type); switch (system_pointer_type) { case RDP_NULL_POINTER: ui_set_null_cursor(); break; default: unimpl("System pointer message 0x%x\n", system_pointer_type); }}/* Process a pointer PDU */static voidprocess_pointer_pdu(STREAM s){ uint16 message_type; uint16 x, y; in_uint16_le(s, message_type); in_uint8s(s, 2); /* pad */ switch (message_type) { case RDP_POINTER_MOVE: in_uint16_le(s, x); in_uint16_le(s, y); if (s_check(s)) ui_move_pointer(x, y); break; case RDP_POINTER_COLOR: process_colour_pointer_pdu(s); break; case RDP_POINTER_CACHED: process_cached_pointer_pdu(s); break; case RDP_POINTER_SYSTEM: process_system_pointer_pdu(s); break; default: unimpl("Pointer message 0x%x\n", message_type); }}/* Process bitmap updates */voidprocess_bitmap_updates(STREAM s){ uint16 num_updates; uint16 left, top, right, bottom, width, height; uint16 cx, cy, bpp, Bpp, compress, bufsize, size; uint8 *data, *bmpdata; int i; in_uint16_le(s, num_updates); for (i = 0; i < num_updates; i++) { in_uint16_le(s, left); in_uint16_le(s, top); in_uint16_le(s, right); in_uint16_le(s, bottom); in_uint16_le(s, width); in_uint16_le(s, height); in_uint16_le(s, bpp); Bpp = (bpp + 7) / 8; in_uint16_le(s, compress); in_uint16_le(s, bufsize); cx = right - left + 1; cy = bottom - top + 1; DEBUG(("BITMAP_UPDATE(l=%d,t=%d,r=%d,b=%d,w=%d,h=%d,Bpp=%d,cmp=%d)\n", left, top, right, bottom, width, height, Bpp, compress)); if (!compress) { int y; bmpdata = (uint8 *) xmalloc(width * height * Bpp); for (y = 0; y < height; y++) { in_uint8a(s, &bmpdata[(height - y - 1) * (width * Bpp)], width * Bpp); } ui_paint_bitmap(left, top, cx, cy, width, height, bmpdata); xfree(bmpdata); continue; } if (compress & 0x400) { size = bufsize; } else { in_uint8s(s, 2); /* pad */ in_uint16_le(s, size); in_uint8s(s, 4); /* line_size, final_size */ } in_uint8p(s, data, size); bmpdata = (uint8 *) xmalloc(width * height * Bpp); if (bitmap_decompress(bmpdata, width, height, data, size, Bpp)) { ui_paint_bitmap(left, top, cx, cy, width, height, bmpdata); } else { DEBUG_RDP5(("Failed to decompress data\n")); } xfree(bmpdata); }}/* Process a palette update */voidprocess_palette(STREAM s){ COLOURENTRY *entry; COLOURMAP map; HCOLOURMAP hmap; int i; in_uint8s(s, 2); /* pad */ in_uint16_le(s, map.ncolours); in_uint8s(s, 2); /* pad */ map.colours = (COLOURENTRY *) xmalloc(sizeof(COLOURENTRY) * map.ncolours); DEBUG(("PALETTE(c=%d)\n", map.ncolours)); for (i = 0; i < map.ncolours; i++) { entry = &map.colours[i]; in_uint8(s, entry->red); in_uint8(s, entry->green); in_uint8(s, entry->blue); } hmap = ui_create_colourmap(&map); ui_set_colourmap(hmap); xfree(map.colours);}/* Process an update PDU */static voidprocess_update_pdu(STREAM s){ uint16 update_type, count; in_uint16_le(s, update_type); ui_begin_update(); switch (update_type) { case RDP_UPDATE_ORDERS: in_uint8s(s, 2); /* pad */ in_uint16_le(s, count); in_uint8s(s, 2); /* pad */ process_orders(s, count); break; case RDP_UPDATE_BITMAP: process_bitmap_updates(s); break; case RDP_UPDATE_PALETTE: process_palette(s); break; case RDP_UPDATE_SYNCHRONIZE: break; default: unimpl("update %d\n", update_type); } ui_end_update();}/* Process a disconnect PDU */voidprocess_disconnect_pdu(STREAM s, uint32 * ext_disc_reason){ in_uint32_le(s, *ext_disc_reason); DEBUG(("Received disconnect PDU\n"));}/* Process data PDU */static BOOLprocess_data_pdu(STREAM s, uint32 * ext_disc_reason){ uint8 data_pdu_type; uint8 ctype; uint16 clen; uint32 len; uint32 roff, rlen; struct stream *ns = &(g_mppc_dict.ns); in_uint8s(s, 6); /* shareid, pad, streamid */ in_uint16(s, len); in_uint8(s, data_pdu_type); in_uint8(s, ctype); in_uint16(s, clen); clen -= 18; if (ctype & RDP_MPPC_COMPRESSED) { if (len > RDP_MPPC_DICT_SIZE) error("error decompressed packet size exceeds max\n"); if (mppc_expand(s->p, clen, ctype, &roff, &rlen) == -1) error("error while decompressing packet\n"); /* len -= 18; */ /* allocate memory and copy the uncompressed data into the temporary stream */ ns->data = (uint8 *) xrealloc(ns->data, rlen); memcpy((ns->data), (unsigned char *) (g_mppc_dict.hist + roff), rlen); ns->size = rlen; ns->end = (ns->data + ns->size); ns->p = ns->data; ns->rdp_hdr = ns->p; s = ns; } switch (data_pdu_type) { case RDP_DATA_PDU_UPDATE: process_update_pdu(s); break; case RDP_DATA_PDU_CONTROL: DEBUG(("Received Control PDU\n")); break; case RDP_DATA_PDU_SYNCHRONISE: DEBUG(("Received Sync PDU\n")); break; case RDP_DATA_PDU_POINTER: process_pointer_pdu(s); break; case RDP_DATA_PDU_BELL: ui_bell(); break; case RDP_DATA_PDU_LOGON: DEBUG(("Received Logon PDU\n")); /* User logged on */ break; case RDP_DATA_PDU_DISCONNECT: process_disconnect_pdu(s, ext_disc_reason); /* We used to return true and disconnect immediately here, but * Windows Vista sends a disconnect PDU with reason 0 when * reconnecting to a disconnected session, and MSTSC doesn't * drop the connection. I think we should just save the status. */ break; default: unimpl("data PDU %d\n", data_pdu_type); } return False;}/* Process redirect PDU from Session Directory */static BOOLprocess_redirect_pdu(STREAM s /*, uint32 * ext_disc_reason */ ){ uint32 len; /* these 2 bytes are unknown, seem to be zeros */ in_uint8s(s, 2); /* read connection flags */ in_uint32_le(s, g_redirect_flags); /* read length of ip string */ in_uint32_le(s, len); /* read ip string */ rdp_in_unistr(s, g_redirect_server, len); /* read length of cookie string */ in_uint32_le(s, len); /* read cookie string (plain ASCII) */ in_uint8a(s, g_redirect_cookie, len); g_redirect_cookie[len] = 0; /* read length of username string */ in_uint32_le(s, len); /* read username string */ rdp_in_unistr(s, g_redirect_username, len); /* read length of domain string */ in_uint32_le(s, len); /* read domain string */ rdp_in_unistr(s, g_redirect_domain, len); /* read length of password string */ in_uint32_le(s, len); /* read password string */ rdp_in_unistr(s, g_redirect_password, len); g_redirect = True; return True;}/* Process incoming packets *//* nevers gets out of here till app is done */voidrdp_main_loop(BOOL * deactivated, uint32 * ext_disc_reason){ while (rdp_loop(deactivated, ext_disc_reason)) ;}/* used in uiports and rdp_main_loop, processes the rdp packets waiting */BOOLrdp_loop(BOOL * deactivated, uint32 * ext_disc_reason){ uint8 type; BOOL disc = False; /* True when a disconnect PDU was received */ BOOL cont = True; STREAM s; while (cont) { s = rdp_recv(&type); if (s == NULL) return False; switch (type) { case RDP_PDU_DEMAND_ACTIVE: process_demand_active(s); *deactivated = False; break; case RDP_PDU_DEACTIVATE: DEBUG(("RDP_PDU_DEACTIVATE\n")); *deactivated = True; break; case RDP_PDU_REDIRECT: return process_redirect_pdu(s); break; case RDP_PDU_DATA: disc = process_data_pdu(s, ext_disc_reason); break; case 0: break; default: unimpl("PDU %d\n", type); } if (disc) return False; cont = g_next_packet < s->end; } return True;}/* Establish a connection up to the RDP layer */BOOLrdp_connect(char *server, uint32 flags, char *domain, char *password, char *command, char *directory){ if (!sec_connect(server, g_username)) return False; rdp_send_logon_info(flags, domain, g_username, password, command, directory); return True;}/* Establish a reconnection up to the RDP layer */BOOLrdp_reconnect(char *server, uint32 flags, char *domain, char *password, char *command, char *directory, char *cookie){ if (!sec_reconnect(server)) return False; rdp_send_logon_info(flags, domain, g_username, password, command, directory); return True;}/* Called during redirection to reset the state to support redirection */voidrdp_reset_state(void){ g_next_packet = NULL; /* reset the packet information */ g_rdp_shareid = 0; sec_reset_state();}/* Disconnect from the RDP layer */voidrdp_disconnect(void){ sec_disconnect();}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -