/** * FreeRDP: A Remote Desktop Protocol Client * Update Data PDUs * * Copyright 2011 Marc-Andre Moreau * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "update.h" #include "bitmap.h" uint8 UPDATE_TYPE_STRINGS[][32] = { "Orders", "Bitmap", "Palette", "Synchronize" }; void rdp_recv_orders_update(rdpRdp* rdp, STREAM* s) { uint16 numberOrders; stream_seek_uint16(s); /* pad2OctetsA (2 bytes) */ stream_read_uint16(s, numberOrders); /* numberOrders (2 bytes) */ stream_seek_uint16(s); /* pad2OctetsB (2 bytes) */ while (numberOrders > 0) { rdp_recv_order(rdp, s); numberOrders--; } } void rdp_read_bitmap_data(STREAM* s, BITMAP_DATA* bitmap_data) { uint8* srcData; uint16 dstSize; boolean status; uint16 bytesPerPixel; stream_read_uint16(s, bitmap_data->left); stream_read_uint16(s, bitmap_data->top); stream_read_uint16(s, bitmap_data->right); stream_read_uint16(s, bitmap_data->bottom); stream_read_uint16(s, bitmap_data->width); stream_read_uint16(s, bitmap_data->height); stream_read_uint16(s, bitmap_data->bpp); stream_read_uint16(s, bitmap_data->flags); stream_read_uint16(s, bitmap_data->length); bytesPerPixel = (bitmap_data->bpp + 7) / 8; if (bitmap_data->flags & BITMAP_COMPRESSION) { uint16 cbCompMainBodySize; uint16 cbUncompressedSize; stream_seek_uint16(s); /* cbCompFirstRowSize (2 bytes) */ stream_read_uint16(s, cbCompMainBodySize); /* cbCompMainBodySize (2 bytes) */ stream_seek_uint16(s); /* cbScanWidth (2 bytes) */ stream_read_uint16(s, cbUncompressedSize); /* cbUncompressedSize (2 bytes) */ dstSize = cbUncompressedSize; bitmap_data->length = cbCompMainBodySize; } else { dstSize = bitmap_data->width * bitmap_data->height * bytesPerPixel; } stream_get_mark(s, srcData); stream_seek(s, bitmap_data->length); bitmap_data->data = (uint8*) xmalloc(dstSize); //printf("bytesPerPixel:%d, width:%d, height:%d dstSize:%d flags:0x%04X\n", // bytesPerPixel, bitmap_data->width, bitmap_data->height, dstSize, bitmap_data->flags); status = bitmap_decompress(srcData, bitmap_data->data, bitmap_data->width, bitmap_data->height, bitmap_data->length, bitmap_data->bpp, bitmap_data->bpp); if (status != True) printf("bitmap decompression failed\n"); } void rdp_read_bitmap_update(rdpRdp* rdp, STREAM* s, BITMAP_UPDATE* bitmap_update) { int i; stream_read_uint16(s, bitmap_update->number); /* numberRectangles (2 bytes) */ bitmap_update->bitmaps = (BITMAP_DATA*) xmalloc(sizeof(BITMAP_DATA) * bitmap_update->number); /* rectangles */ for (i = 0; i < bitmap_update->number; i++) { rdp_read_bitmap_data(s, &bitmap_update->bitmaps[i]); } } void rdp_read_palette_update(rdpRdp* rdp, STREAM* s, PALETTE_UPDATE* palette_update) { int i; uint8 byte; uint32 color; stream_seek_uint16(s); /* pad2Octets (2 bytes) */ stream_seek_uint32(palette_update->number); /* numberColors (4 bytes), must be set to 256 */ if (palette_update->number > 256) palette_update->number = 256; /* paletteEntries */ for (i = 0; i < palette_update->number; i++) { stream_read_uint8(s, byte); color = byte; stream_read_uint8(s, byte); color |= (byte << 8); stream_read_uint8(s, byte); color |= (byte << 16); palette_update->entries[i] = color; } } void rdp_read_synchronize_update(rdpRdp* rdp, STREAM* s) { stream_seek_uint16(s); /* pad2Octets (2 bytes) */ /** * The Synchronize Update is an artifact from the * T.128 protocol and should be ignored. */ } void rdp_recv_update_data_pdu(rdpRdp* rdp, STREAM* s) { uint16 updateType; stream_read_uint16(s, updateType); /* updateType (2 bytes) */ if (updateType != UPDATE_TYPE_ORDERS) printf("%s Update Data PDU\n", UPDATE_TYPE_STRINGS[updateType]); switch (updateType) { case UPDATE_TYPE_ORDERS: rdp_recv_orders_update(rdp, s); break; case UPDATE_TYPE_BITMAP: rdp_read_bitmap_update(rdp, s, &rdp->update->bitmap_update); break; case UPDATE_TYPE_PALETTE: rdp_read_palette_update(rdp, s, &rdp->update->palette_update); break; case UPDATE_TYPE_SYNCHRONIZE: rdp_read_synchronize_update(rdp, s); break; } } rdpUpdate* update_new() { rdpUpdate* update; update = (rdpUpdate*) xzalloc(sizeof(rdpUpdate)); if (update != NULL) { } return update; } void update_free(rdpUpdate* update) { if (update != NULL) { xfree(update); } }