22#include <freerdp/config.h>
25#include <winpr/print.h>
26#include <winpr/synch.h>
27#include <winpr/thread.h>
28#include <winpr/collections.h>
29#include <winpr/assert.h>
30#include <winpr/cast.h>
39#include <freerdp/log.h>
40#include <freerdp/peer.h>
41#include <freerdp/codec/bitmap.h>
43#include "../cache/pointer.h"
44#include "../cache/palette.h"
45#include "../cache/bitmap.h"
47#define TAG FREERDP_TAG("core.update")
49#define FORCE_ASYNC_UPDATE_OFF
51static const char*
const UPDATE_TYPE_STRINGS[] = {
"Orders",
"Bitmap",
"Palette",
"Synchronize" };
53static const char* update_type_to_string(UINT16 updateType)
55 if (updateType >= ARRAYSIZE(UPDATE_TYPE_STRINGS))
58 return UPDATE_TYPE_STRINGS[updateType];
61static BOOL update_recv_orders(rdpUpdate* update,
wStream* s)
63 UINT16 numberOrders = 0;
67 if (!Stream_CheckAndLogRequiredLength(TAG, s, 6))
70 Stream_Seek_UINT16(s);
71 Stream_Read_UINT16(s, numberOrders);
72 Stream_Seek_UINT16(s);
74 while (numberOrders > 0)
76 if (!update_recv_order(update, s))
78 WLog_ERR(TAG,
"update_recv_order() failed");
88static BOOL update_read_bitmap_data(rdpUpdate* update,
wStream* s,
BITMAP_DATA* bitmapData)
91 WINPR_ASSERT(bitmapData);
93 if (!Stream_CheckAndLogRequiredLength(TAG, s, 18))
96 Stream_Read_UINT16(s, bitmapData->destLeft);
97 Stream_Read_UINT16(s, bitmapData->destTop);
98 Stream_Read_UINT16(s, bitmapData->destRight);
99 Stream_Read_UINT16(s, bitmapData->destBottom);
100 Stream_Read_UINT16(s, bitmapData->width);
101 Stream_Read_UINT16(s, bitmapData->height);
102 Stream_Read_UINT16(s, bitmapData->bitsPerPixel);
103 Stream_Read_UINT16(s, bitmapData->flags);
104 Stream_Read_UINT16(s, bitmapData->bitmapLength);
106 if ((bitmapData->width == 0) || (bitmapData->height == 0))
108 WLog_ERR(TAG,
"Invalid BITMAP_DATA: width=%" PRIu16
", height=%" PRIu16, bitmapData->width,
113 if (bitmapData->flags & BITMAP_COMPRESSION)
115 if (!(bitmapData->flags & NO_BITMAP_COMPRESSION_HDR))
117 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
120 Stream_Read_UINT16(s,
121 bitmapData->cbCompFirstRowSize);
122 Stream_Read_UINT16(s,
123 bitmapData->cbCompMainBodySize);
124 Stream_Read_UINT16(s, bitmapData->cbScanWidth);
125 Stream_Read_UINT16(s,
126 bitmapData->cbUncompressedSize);
127 bitmapData->bitmapLength = bitmapData->cbCompMainBodySize;
130 bitmapData->compressed = TRUE;
133 bitmapData->compressed = FALSE;
135 if (!Stream_CheckAndLogRequiredLength(TAG, s, bitmapData->bitmapLength))
138 if (bitmapData->bitmapLength > 0)
140 bitmapData->bitmapDataStream = malloc(bitmapData->bitmapLength);
142 if (!bitmapData->bitmapDataStream)
145 memcpy(bitmapData->bitmapDataStream, Stream_ConstPointer(s), bitmapData->bitmapLength);
146 Stream_Seek(s, bitmapData->bitmapLength);
154 WINPR_ASSERT(bitmapData);
155 if (!Stream_EnsureRemainingCapacity(s, 18))
157 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->destLeft));
158 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->destTop));
159 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->destRight));
160 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->destBottom));
161 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->width));
162 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->height));
163 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->bitsPerPixel));
164 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->flags));
165 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->bitmapLength));
169static BOOL update_write_bitmap_data_no_comp_header(
const BITMAP_DATA* bitmapData,
wStream* s)
171 WINPR_ASSERT(bitmapData);
172 if (!Stream_EnsureRemainingCapacity(s, 8))
176 s, WINPR_ASSERTING_INT_CAST(
177 uint16_t, bitmapData->cbCompFirstRowSize));
179 s, WINPR_ASSERTING_INT_CAST(
180 uint16_t, bitmapData->cbCompMainBodySize));
182 s, WINPR_ASSERTING_INT_CAST(uint16_t, bitmapData->cbScanWidth));
184 s, WINPR_ASSERTING_INT_CAST(
185 uint16_t, bitmapData->cbUncompressedSize));
189static BOOL update_write_bitmap_data(rdpUpdate* update_pub,
wStream* s,
BITMAP_DATA* bitmapData)
193 WINPR_ASSERT(bitmapData);
195 if (!Stream_EnsureRemainingCapacity(s, 64 + bitmapData->bitmapLength))
198 if (update->common.autoCalculateBitmapData)
200 bitmapData->flags = 0;
201 bitmapData->cbCompFirstRowSize = 0;
203 if (bitmapData->compressed)
204 bitmapData->flags |= BITMAP_COMPRESSION;
206 if (update->common.context->settings->NoBitmapCompressionHeader)
208 bitmapData->flags |= NO_BITMAP_COMPRESSION_HDR;
209 bitmapData->cbCompMainBodySize = bitmapData->bitmapLength;
213 if (!update_write_bitmap_data_header(bitmapData, s))
216 if (bitmapData->flags & BITMAP_COMPRESSION)
218 if ((bitmapData->flags & NO_BITMAP_COMPRESSION_HDR) == 0)
220 if (!update_write_bitmap_data_no_comp_header(bitmapData, s))
225 if (!Stream_EnsureRemainingCapacity(s, bitmapData->bitmapLength))
227 Stream_Write(s, bitmapData->bitmapDataStream, bitmapData->bitmapLength);
240 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
243 Stream_Read_UINT16(s, bitmapUpdate->number);
244 WLog_Print(up->log, WLOG_TRACE,
"BitmapUpdate: %" PRIu32
"", bitmapUpdate->number);
248 if (!bitmapUpdate->rectangles)
252 for (UINT32 i = 0; i < bitmapUpdate->number; i++)
254 if (!update_read_bitmap_data(update, s, &bitmapUpdate->rectangles[i]))
260 WINPR_PRAGMA_DIAG_PUSH
261 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
262 free_bitmap_update(update->context, bitmapUpdate);
263 WINPR_PRAGMA_DIAG_POP
267static BOOL update_write_bitmap_update(rdpUpdate* update,
wStream* s,
270 WINPR_ASSERT(update);
271 WINPR_ASSERT(bitmapUpdate);
273 if (!Stream_EnsureRemainingCapacity(s, 32))
276 Stream_Write_UINT16(s, UPDATE_TYPE_BITMAP);
277 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(
278 uint16_t, bitmapUpdate->number));
281 for (UINT32 i = 0; i < bitmapUpdate->number; i++)
283 if (!update_write_bitmap_data(update, s, &bitmapUpdate->rectangles[i]))
297 if (!Stream_CheckAndLogRequiredLength(TAG, s, 6))
300 Stream_Seek_UINT16(s);
301 Stream_Read_UINT32(s, palette_update->number);
303 if (palette_update->number > 256)
304 palette_update->number = 256;
306 if (!Stream_CheckAndLogRequiredLengthOfSize(TAG, s, palette_update->number, 3ull))
310 for (UINT32 i = 0; i < palette_update->number; i++)
313 Stream_Read_UINT8(s, entry->red);
314 Stream_Read_UINT8(s, entry->green);
315 Stream_Read_UINT8(s, entry->blue);
318 return palette_update;
320 WINPR_PRAGMA_DIAG_PUSH
321 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
322 free_palette_update(update->context, palette_update);
323 WINPR_PRAGMA_DIAG_POP
327static BOOL update_read_synchronize(rdpUpdate* update,
wStream* s)
329 WINPR_UNUSED(update);
330 return Stream_SafeSeek(s, 2);
339 WINPR_ASSERT(play_sound);
341 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
344 Stream_Read_UINT32(s, play_sound->duration);
345 Stream_Read_UINT32(s, play_sound->frequency);
349BOOL update_recv_play_sound(rdpUpdate* update,
wStream* s)
353 WINPR_ASSERT(update);
355 if (!update_read_play_sound(s, &play_sound))
358 return IFCALLRESULT(FALSE, update->PlaySound, update->context, &play_sound);
365 WINPR_ASSERT(update);
367 if (!pointer_position)
370 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
373 Stream_Read_UINT16(s, pointer_position->xPos);
374 Stream_Read_UINT16(s, pointer_position->yPos);
375 return pointer_position;
377 WINPR_PRAGMA_DIAG_PUSH
378 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
379 free_pointer_position_update(update->context, pointer_position);
380 WINPR_PRAGMA_DIAG_POP
388 WINPR_ASSERT(update);
393 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
396 Stream_Read_UINT32(s, pointer_system->type);
397 return pointer_system;
399 WINPR_PRAGMA_DIAG_PUSH
400 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
401 free_pointer_system_update(update->context, pointer_system);
402 WINPR_PRAGMA_DIAG_POP
407 BYTE xorBpp, UINT32 flags)
409 BYTE* newMask =
nullptr;
410 UINT32 scanlineSize = 0;
413 WINPR_ASSERT(pointer_color);
415 if (flags & LARGE_POINTER_FLAG_96x96)
421 if (!Stream_CheckAndLogRequiredLength(TAG, s, 14))
424 Stream_Read_UINT16(s, pointer_color->cacheIndex);
425 Stream_Read_UINT16(s, pointer_color->hotSpotX);
426 Stream_Read_UINT16(s, pointer_color->hotSpotY);
436 Stream_Read_UINT16(s, pointer_color->width);
437 Stream_Read_UINT16(s, pointer_color->height);
439 if ((pointer_color->width > max) || (pointer_color->height > max))
442 Stream_Read_UINT16(s, pointer_color->lengthAndMask);
443 Stream_Read_UINT16(s, pointer_color->lengthXorMask);
451 if (pointer_color->hotSpotX >= pointer_color->width)
452 pointer_color->hotSpotX = 0;
454 if (pointer_color->hotSpotY >= pointer_color->height)
455 pointer_color->hotSpotY = 0;
457 if (pointer_color->lengthXorMask > 0)
470 if (!Stream_CheckAndLogRequiredLength(TAG, s, pointer_color->lengthXorMask))
473 scanlineSize = (7 + xorBpp * pointer_color->width) / 8;
474 scanlineSize = ((scanlineSize + 1) / 2) * 2;
476 if (scanlineSize * pointer_color->height != pointer_color->lengthXorMask)
479 "invalid lengthXorMask: width=%" PRIu32
" height=%" PRIu32
", %" PRIu32
480 " instead of %" PRIu32
"",
481 pointer_color->width, pointer_color->height, pointer_color->lengthXorMask,
482 scanlineSize * pointer_color->height);
486 newMask = realloc(pointer_color->xorMaskData, pointer_color->lengthXorMask);
491 pointer_color->xorMaskData = newMask;
492 Stream_Read(s, pointer_color->xorMaskData, pointer_color->lengthXorMask);
495 if (pointer_color->lengthAndMask > 0)
504 if (!Stream_CheckAndLogRequiredLength(TAG, s, pointer_color->lengthAndMask))
507 scanlineSize = ((7 + pointer_color->width) / 8);
508 scanlineSize = ((1 + scanlineSize) / 2) * 2;
510 if (scanlineSize * pointer_color->height != pointer_color->lengthAndMask)
512 WLog_ERR(TAG,
"invalid lengthAndMask: %" PRIu32
" instead of %" PRIu32
"",
513 pointer_color->lengthAndMask, scanlineSize * pointer_color->height);
517 newMask = realloc(pointer_color->andMaskData, pointer_color->lengthAndMask);
522 pointer_color->andMaskData = newMask;
523 Stream_Read(s, pointer_color->andMaskData, pointer_color->lengthAndMask);
526 if (Stream_GetRemainingLength(s) > 0)
527 Stream_Seek_UINT8(s);
538 WINPR_ASSERT(update);
543 if (!s_update_read_pointer_color(s, pointer_color, xorBpp,
544 update->context->settings->LargePointerFlag))
547 return pointer_color;
549 WINPR_PRAGMA_DIAG_PUSH
550 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
551 free_pointer_color_update(update->context, pointer_color);
552 WINPR_PRAGMA_DIAG_POP
558 BYTE* newMask =
nullptr;
559 UINT32 scanlineSize = 0;
564 if (!Stream_CheckAndLogRequiredLength(TAG, s, 20))
567 Stream_Read_UINT16(s, pointer->xorBpp);
568 Stream_Read_UINT16(s, pointer->cacheIndex);
569 Stream_Read_UINT16(s, pointer->hotSpotX);
570 Stream_Read_UINT16(s, pointer->hotSpotY);
572 Stream_Read_UINT16(s, pointer->width);
573 Stream_Read_UINT16(s, pointer->height);
575 if ((pointer->width > 384) || (pointer->height > 384))
578 Stream_Read_UINT32(s, pointer->lengthAndMask);
579 Stream_Read_UINT32(s, pointer->lengthXorMask);
581 if (pointer->hotSpotX >= pointer->width)
582 pointer->hotSpotX = 0;
584 if (pointer->hotSpotY >= pointer->height)
585 pointer->hotSpotY = 0;
587 if (pointer->lengthXorMask > 0)
600 if (!Stream_CheckAndLogRequiredLength(TAG, s, pointer->lengthXorMask))
603 scanlineSize = (7 + pointer->xorBpp * pointer->width) / 8;
604 scanlineSize = ((scanlineSize + 1) / 2) * 2;
606 if (scanlineSize * pointer->height != pointer->lengthXorMask)
609 "invalid lengthXorMask: width=%" PRIu32
" height=%" PRIu32
", %" PRIu32
610 " instead of %" PRIu32
"",
611 pointer->width, pointer->height, pointer->lengthXorMask,
612 scanlineSize * pointer->height);
616 newMask = realloc(pointer->xorMaskData, pointer->lengthXorMask);
621 pointer->xorMaskData = newMask;
622 Stream_Read(s, pointer->xorMaskData, pointer->lengthXorMask);
625 if (pointer->lengthAndMask > 0)
634 if (!Stream_CheckAndLogRequiredLength(TAG, s, pointer->lengthAndMask))
637 scanlineSize = ((7 + pointer->width) / 8);
638 scanlineSize = ((1 + scanlineSize) / 2) * 2;
640 if (scanlineSize * pointer->height != pointer->lengthAndMask)
642 WLog_ERR(TAG,
"invalid lengthAndMask: %" PRIu32
" instead of %" PRIu32
"",
643 pointer->lengthAndMask, scanlineSize * pointer->height);
647 newMask = realloc(pointer->andMaskData, pointer->lengthAndMask);
652 pointer->andMaskData = newMask;
653 Stream_Read(s, pointer->andMaskData, pointer->lengthAndMask);
656 if (Stream_GetRemainingLength(s) > 0)
657 Stream_Seek_UINT8(s);
668 WINPR_ASSERT(update);
673 if (!s_update_read_pointer_large(s, pointer))
678 WINPR_PRAGMA_DIAG_PUSH
679 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
680 free_pointer_large_update(update->context, pointer);
681 WINPR_PRAGMA_DIAG_POP
689 WINPR_ASSERT(update);
694 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
697 Stream_Read_UINT16(s, pointer_new->xorBpp);
699 if ((pointer_new->xorBpp < 1) || (pointer_new->xorBpp > 32))
701 WLog_ERR(TAG,
"invalid xorBpp %" PRIu32
"", pointer_new->xorBpp);
705 WINPR_ASSERT(pointer_new->xorBpp <= UINT8_MAX);
706 if (!s_update_read_pointer_color(
707 s, &pointer_new->colorPtrAttr, (UINT8)pointer_new->xorBpp,
708 update->context->settings->LargePointerFlag))
713 WINPR_PRAGMA_DIAG_PUSH
714 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
715 free_pointer_new_update(update->context, pointer_new);
716 WINPR_PRAGMA_DIAG_POP
724 WINPR_ASSERT(update);
729 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
732 Stream_Read_UINT16(s, pointer->cacheIndex);
735 WINPR_PRAGMA_DIAG_PUSH
736 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
737 free_pointer_cached_update(update->context, pointer);
738 WINPR_PRAGMA_DIAG_POP
742BOOL update_recv_pointer(rdpUpdate* update,
wStream* s)
745 UINT16 messageType = 0;
747 WINPR_ASSERT(update);
749 rdpContext* context = update->context;
750 rdpPointerUpdate* pointer = update->pointer;
752 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2 + 2))
755 Stream_Read_UINT16(s, messageType);
756 Stream_Seek_UINT16(s);
760 case PTR_MSG_TYPE_POSITION:
764 if (pointer_position)
766 rc = IFCALLRESULT(FALSE, pointer->PointerPosition, context, pointer_position);
767 free_pointer_position_update(context, pointer_position);
772 case PTR_MSG_TYPE_SYSTEM:
778 rc = IFCALLRESULT(FALSE, pointer->PointerSystem, context, pointer_system);
779 free_pointer_system_update(context, pointer_system);
784 case PTR_MSG_TYPE_COLOR:
790 rc = IFCALLRESULT(FALSE, pointer->PointerColor, context, pointer_color);
791 free_pointer_color_update(context, pointer_color);
796 case PTR_MSG_TYPE_POINTER_LARGE:
802 rc = IFCALLRESULT(FALSE, pointer->PointerLarge, context, pointer_large);
803 free_pointer_large_update(context, pointer_large);
808 case PTR_MSG_TYPE_POINTER:
814 rc = IFCALLRESULT(FALSE, pointer->PointerNew, context, pointer_new);
815 free_pointer_new_update(context, pointer_new);
820 case PTR_MSG_TYPE_CACHED:
826 rc = IFCALLRESULT(FALSE, pointer->PointerCached, context, pointer_cached);
827 free_pointer_cached_update(context, pointer_cached);
839BOOL update_recv(rdpUpdate* update,
wStream* s)
842 UINT16 updateType = 0;
844 rdpContext* context = update->context;
846 WINPR_ASSERT(context);
848 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
851 Stream_Read_UINT16(s, updateType);
852 WLog_Print(up->log, WLOG_TRACE,
"%s Update Data PDU", update_type_to_string(updateType));
854 if (!update_begin_paint(update))
859 case UPDATE_TYPE_ORDERS:
860 rc = update_recv_orders(update, s);
863 case UPDATE_TYPE_BITMAP:
865 BITMAP_UPDATE* bitmap_update = update_read_bitmap_update(update, s);
869 WLog_ERR(TAG,
"UPDATE_TYPE_BITMAP - update_read_bitmap_update() failed");
873 rc = IFCALLRESULT(FALSE, update->BitmapUpdate, context, bitmap_update);
874 free_bitmap_update(context, bitmap_update);
878 case UPDATE_TYPE_PALETTE:
884 WLog_ERR(TAG,
"UPDATE_TYPE_PALETTE - update_read_palette() failed");
888 rc = IFCALLRESULT(FALSE, update->Palette, context, palette_update);
889 free_palette_update(context, palette_update);
893 case UPDATE_TYPE_SYNCHRONIZE:
894 if (!update_read_synchronize(update, s))
896 rc = IFCALLRESULT(TRUE, update->Synchronize, context);
905 if (!update_end_paint(update))
910 WLog_ERR(TAG,
"UPDATE_TYPE %s [%" PRIu16
"] failed", update_type_to_string(updateType),
918void update_reset_state(rdpUpdate* update)
923 WINPR_ASSERT(primary);
925 ZeroMemory(&primary->order_info,
sizeof(
ORDER_INFO));
938 free(primary->polyline.points);
947 free(primary->fast_glyph.glyphData.aj);
950 free(primary->polygon_sc.points);
953 free(primary->polygon_cb.points);
958 primary->order_info.orderType = ORDER_TYPE_PATBLT;
960 if (!up->initialState)
963 WINPR_ASSERT(altsec);
965 altsec->switch_surface.bitmapId = SCREEN_BITMAP_SURFACE;
966 if (altsec->common.SwitchSurface)
968 if (!altsec->common.SwitchSurface(update->context, &(altsec->switch_surface)))
969 WLog_Print(up->log, WLOG_WARN,
"altsec->common.SwitchSurface failed");
974BOOL update_post_connect(rdpUpdate* update)
979 WINPR_ASSERT(update->context);
980 WINPR_ASSERT(update->context->settings);
981 up->asynchronous = update->context->settings->AsyncUpdate;
983 if (up->asynchronous)
985#if defined(FORCE_ASYNC_UPDATE_OFF)
986 WLog_WARN(TAG,
"AsyncUpdate requested, but forced deactivated");
987 WLog_WARN(TAG,
"see https://github.com/FreeRDP/FreeRDP/issues/10153 for details");
989 if (!(up->proxy = update_message_proxy_new(update)))
994 altsec->switch_surface.bitmapId = SCREEN_BITMAP_SURFACE;
995 const BOOL rc = IFCALLRESULT(TRUE, update->altsec->SwitchSurface, update->context,
996 &(altsec->switch_surface));
997 up->initialState = FALSE;
1001void update_post_disconnect(rdpUpdate* update)
1005 WINPR_ASSERT(update->context);
1006 WINPR_ASSERT(update->context->settings);
1008 up->asynchronous = update->context->settings->AsyncUpdate;
1010 if (up->asynchronous)
1012#if !defined(FORCE_ASYNC_UPDATE_OFF)
1013 update_message_proxy_free(up->proxy);
1017 up->initialState = TRUE;
1020static BOOL s_update_begin_paint(rdpContext* context)
1023 WINPR_ASSERT(context);
1028 if (!update_end_paint(&update->common))
1032 WINPR_ASSERT(context->rdp);
1033 s = fastpath_update_pdu_init_new(context->rdp->fastpath);
1038 Stream_SealLength(s);
1039 Stream_GetLength(s, update->offsetOrders);
1041 update->combineUpdates = TRUE;
1042 update->numberOrders = 0;
1047static BOOL s_update_end_paint(rdpContext* context)
1051 WINPR_ASSERT(context);
1058 update->us =
nullptr;
1060 Stream_SealLength(s);
1061 Stream_SetPosition(s, update->offsetOrders);
1062 Stream_Write_UINT16(s, update->numberOrders);
1063 Stream_SetPosition(s, Stream_Length(s));
1065 if (update->numberOrders > 0)
1067 WLog_DBG(TAG,
"sending %" PRIu16
" orders", update->numberOrders);
1068 if (!fastpath_send_update_pdu(context->rdp->fastpath, FASTPATH_UPDATETYPE_ORDERS, s, FALSE))
1072 update->combineUpdates = FALSE;
1073 update->numberOrders = 0;
1074 update->offsetOrders = 0;
1078 Stream_Free(s, TRUE);
1082static BOOL update_flush(rdpContext* context)
1086 WINPR_ASSERT(context);
1087 update = update_cast(context->update);
1089 if (update->numberOrders > 0)
1091 if (!update_end_paint(&update->common))
1094 if (!update_begin_paint(&update->common))
1100static BOOL update_force_flush(rdpContext* context)
1102 return update_flush(context);
1105static BOOL update_check_flush(rdpContext* context,
size_t size)
1107 WINPR_ASSERT(context);
1114 if (!update_begin_paint(&update->common))
1119 if (Stream_GetPosition(s) + size + 64 >= FASTPATH_MAX_PACKET_SIZE)
1122 if (!update_flush(context))
1129static BOOL update_set_bounds(rdpContext* context,
const rdpBounds* bounds)
1133 WINPR_ASSERT(context);
1135 update = update_cast(context->update);
1137 CopyMemory(&update->previousBounds, &update->currentBounds,
sizeof(rdpBounds));
1140 ZeroMemory(&update->currentBounds,
sizeof(rdpBounds));
1142 CopyMemory(&update->currentBounds, bounds,
sizeof(rdpBounds));
1147static BOOL update_bounds_is_null(rdpBounds* bounds)
1149 WINPR_ASSERT(bounds);
1150 return ((bounds->left == 0) && (bounds->top == 0) && (bounds->right == 0) &&
1151 (bounds->bottom == 0));
1154static BOOL update_bounds_equals(rdpBounds* bounds1, rdpBounds* bounds2)
1156 WINPR_ASSERT(bounds1);
1157 WINPR_ASSERT(bounds2);
1159 return ((bounds1->left == bounds2->left) && (bounds1->top == bounds2->top) &&
1160 (bounds1->right == bounds2->right) && (bounds1->bottom == bounds2->bottom));
1163static size_t update_prepare_bounds(rdpContext* context,
ORDER_INFO* orderInfo)
1168 WINPR_ASSERT(context);
1169 WINPR_ASSERT(orderInfo);
1171 update = update_cast(context->update);
1173 orderInfo->boundsFlags = 0;
1175 if (update_bounds_is_null(&update->currentBounds))
1178 orderInfo->controlFlags |= ORDER_BOUNDS;
1180 if (update_bounds_equals(&update->previousBounds, &update->currentBounds))
1182 orderInfo->controlFlags |= ORDER_ZERO_BOUNDS_DELTAS;
1189 if (update->previousBounds.left != update->currentBounds.left)
1191 orderInfo->bounds.left = update->currentBounds.left;
1192 orderInfo->boundsFlags |= BOUND_LEFT;
1196 if (update->previousBounds.top != update->currentBounds.top)
1198 orderInfo->bounds.top = update->currentBounds.top;
1199 orderInfo->boundsFlags |= BOUND_TOP;
1203 if (update->previousBounds.right != update->currentBounds.right)
1205 orderInfo->bounds.right = update->currentBounds.right;
1206 orderInfo->boundsFlags |= BOUND_RIGHT;
1210 if (update->previousBounds.bottom != update->currentBounds.bottom)
1212 orderInfo->bounds.bottom = update->currentBounds.bottom;
1213 orderInfo->boundsFlags |= BOUND_BOTTOM;
1221static size_t update_prepare_order_info(rdpContext* context,
ORDER_INFO* orderInfo,
1224 WINPR_ASSERT(context);
1225 WINPR_ASSERT(orderInfo);
1227 orderInfo->fieldFlags = 0;
1228 orderInfo->orderType = orderType;
1229 orderInfo->controlFlags = ORDER_STANDARD;
1230 orderInfo->controlFlags |= ORDER_TYPE_CHANGE;
1232 length += get_primary_drawing_order_field_bytes(orderInfo->orderType,
nullptr);
1233 length += update_prepare_bounds(context, orderInfo);
1238static int update_write_order_info(rdpContext* context,
wStream* s,
const ORDER_INFO* orderInfo,
1241 WINPR_UNUSED(context);
1242 WINPR_ASSERT(orderInfo);
1243 WINPR_ASSERT(orderInfo->controlFlags <= UINT8_MAX);
1245 const size_t position = Stream_GetPosition(s);
1246 const UINT8 controlFlags = (UINT8)orderInfo->controlFlags;
1248 Stream_SetPosition(s, offset);
1249 Stream_Write_UINT8(s, controlFlags);
1251 if (orderInfo->controlFlags & ORDER_TYPE_CHANGE)
1253 s, WINPR_ASSERTING_INT_CAST(uint8_t, orderInfo->orderType));
1255 if (!update_write_field_flags(
1256 s, orderInfo->fieldFlags, controlFlags,
1257 get_primary_drawing_order_field_bytes(orderInfo->orderType,
nullptr)))
1259 if (!update_write_bounds(s, orderInfo))
1261 Stream_SetPosition(s, position);
1268 WINPR_ASSERT(areas || (count == 0));
1270 Stream_Write_UINT8(s, count);
1273 for (BYTE i = 0; i < count; i++)
1275 Stream_Write_UINT16(s, areas[i].left);
1276 Stream_Write_UINT16(s, areas[i].top);
1277 Stream_Write_UINT16(s, areas[i].right);
1278 Stream_Write_UINT16(s, areas[i].bottom);
1282static BOOL update_send_refresh_rect(rdpContext* context, BYTE count,
const RECTANGLE_16* areas)
1284 WINPR_ASSERT(context);
1285 rdpRdp* rdp = context->rdp;
1287 WINPR_ASSERT(rdp->settings);
1288 if (rdp->settings->RefreshRect)
1290 UINT16 sec_flags = 0;
1291 wStream* s = rdp_data_pdu_init(rdp, &sec_flags);
1296 update_write_refresh_rect(s, count, areas);
1297 return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_REFRESH_RECT, rdp->mcs->userId, sec_flags);
1307 Stream_Write_UINT8(s, allow);
1314 Stream_Write_UINT16(s, area->left);
1315 Stream_Write_UINT16(s, area->top);
1316 Stream_Write_UINT16(s, area->right);
1317 Stream_Write_UINT16(s, area->bottom);
1321static BOOL update_send_suppress_output(rdpContext* context, BYTE allow,
const RECTANGLE_16* area)
1323 WINPR_ASSERT(context);
1324 rdpRdp* rdp = context->rdp;
1327 WINPR_ASSERT(rdp->settings);
1328 if (rdp->settings->SuppressOutput)
1330 UINT16 sec_flags = 0;
1331 wStream* s = rdp_data_pdu_init(rdp, &sec_flags);
1336 update_write_suppress_output(s, allow, area);
1337 WINPR_ASSERT(rdp->mcs);
1338 return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_SUPPRESS_OUTPUT, rdp->mcs->userId,
1345static BOOL update_send_surface_command(rdpContext* context,
wStream* s)
1348 WINPR_ASSERT(context);
1349 rdpRdp* rdp = context->rdp;
1353 update = fastpath_update_pdu_init(rdp->fastpath);
1358 if (!Stream_EnsureRemainingCapacity(update, Stream_GetPosition(s)))
1364 Stream_Write(update, Stream_Buffer(s), Stream_GetPosition(s));
1365 ret = fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_SURFCMDS, update, FALSE);
1367 Stream_Release(update);
1371static BOOL update_send_surface_bits(rdpContext* context,
1375 WINPR_ASSERT(context);
1376 rdpRdp* rdp = context->rdp;
1379 WINPR_ASSERT(surfaceBitsCommand);
1382 if (!update_force_flush(context))
1384 s = fastpath_update_pdu_init(rdp->fastpath);
1389 if (!update_write_surfcmd_surface_bits(s, surfaceBitsCommand))
1392 if (!fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_SURFCMDS, s,
1393 surfaceBitsCommand->skipCompression))
1396 ret = update_force_flush(context);
1402static BOOL update_send_surface_frame_marker(rdpContext* context,
1406 WINPR_ASSERT(context);
1407 rdpRdp* rdp = context->rdp;
1409 if (!update_force_flush(context))
1413 s = fastpath_update_pdu_init(rdp->fastpath);
1418 WINPR_ASSERT(surfaceFrameMarker->frameAction <= UINT16_MAX);
1419 if (!update_write_surfcmd_frame_marker(s, (UINT16)surfaceFrameMarker->frameAction,
1420 surfaceFrameMarker->frameId) ||
1421 !fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_SURFCMDS, s, FALSE))
1424 ret = update_force_flush(context);
1430static BOOL update_send_surface_frame_bits(rdpContext* context,
const SURFACE_BITS_COMMAND* cmd,
1431 BOOL first, BOOL last, UINT32 frameId)
1435 WINPR_ASSERT(context);
1436 rdpRdp* rdp = context->rdp;
1439 if (!update_force_flush(context))
1443 s = fastpath_update_pdu_init(rdp->fastpath);
1450 if (!update_write_surfcmd_frame_marker(s, SURFACECMD_FRAMEACTION_BEGIN, frameId))
1454 if (!update_write_surfcmd_surface_bits(s, cmd))
1459 if (!update_write_surfcmd_frame_marker(s, SURFACECMD_FRAMEACTION_END, frameId))
1463 ret = fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_SURFCMDS, s,
1464 cmd->skipCompression);
1468 ret = update_force_flush(context);
1474static BOOL update_send_frame_acknowledge(rdpContext* context, UINT32 frameId)
1476 WINPR_ASSERT(context);
1477 rdpRdp* rdp = context->rdp;
1480 WINPR_ASSERT(rdp->settings);
1481 WINPR_ASSERT(rdp->settings->ReceivedCapabilities);
1482 WINPR_ASSERT(rdp->settings->ReceivedCapabilitiesSize > CAPSET_TYPE_FRAME_ACKNOWLEDGE);
1483 if (rdp->settings->ReceivedCapabilities[CAPSET_TYPE_FRAME_ACKNOWLEDGE])
1485 UINT16 sec_flags = 0;
1486 wStream* s = rdp_data_pdu_init(rdp, &sec_flags);
1491 Stream_Write_UINT32(s, frameId);
1492 return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_FRAME_ACKNOWLEDGE, rdp->mcs->userId,
1499static BOOL update_send_synchronize(rdpContext* context)
1502 WINPR_ASSERT(context);
1503 rdpRdp* rdp = context->rdp;
1507 s = fastpath_update_pdu_init(rdp->fastpath);
1513 ret = fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_SYNCHRONIZE, s, FALSE);
1518static BOOL update_send_desktop_resize(rdpContext* context)
1520 WINPR_ASSERT(context);
1521 return rdp_server_reactivate(context->rdp);
1524static BOOL update_send_bitmap_update(rdpContext* context,
const BITMAP_UPDATE* bitmapUpdate)
1527 WINPR_ASSERT(context);
1528 rdpRdp* rdp = context->rdp;
1529 rdpUpdate* update = context->update;
1532 if (!update_force_flush(context))
1536 s = fastpath_update_pdu_init(rdp->fastpath);
1541 if (!update_write_bitmap_update(update, s, bitmapUpdate) ||
1542 !fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_BITMAP, s,
1543 bitmapUpdate->skipCompression))
1549 ret = update_force_flush(context);
1556static BOOL update_send_play_sound(rdpContext* context,
const PLAY_SOUND_UPDATE* play_sound)
1558 UINT16 sec_flags = 0;
1560 WINPR_ASSERT(context);
1561 rdpRdp* rdp = context->rdp;
1564 WINPR_ASSERT(rdp->settings);
1565 WINPR_ASSERT(play_sound);
1566 WINPR_ASSERT(rdp->settings->ReceivedCapabilities);
1567 WINPR_ASSERT(rdp->settings->ReceivedCapabilitiesSize > CAPSET_TYPE_SOUND);
1568 if (!rdp->settings->ReceivedCapabilities[CAPSET_TYPE_SOUND])
1573 s = rdp_data_pdu_init(rdp, &sec_flags);
1578 Stream_Write_UINT32(s, play_sound->duration);
1579 Stream_Write_UINT32(s, play_sound->frequency);
1580 return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_PLAY_SOUND, rdp->mcs->userId, sec_flags);
1587static BOOL update_send_dstblt(rdpContext* context,
const DSTBLT_ORDER* dstblt)
1591 WINPR_ASSERT(context);
1592 WINPR_ASSERT(dstblt);
1596 const size_t headerLength = update_prepare_order_info(context, &orderInfo, ORDER_TYPE_DSTBLT);
1597 const size_t inf = update_approximate_dstblt_order(&orderInfo, dstblt);
1598 if (!update_check_flush(context, headerLength + inf))
1606 const size_t offset = Stream_GetPosition(s);
1608 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1611 Stream_Seek(s, headerLength);
1613 if (!update_write_dstblt_order(s, &orderInfo, dstblt))
1616 if (update_write_order_info(context, s, &orderInfo, offset) < 0)
1618 update->numberOrders++;
1622static BOOL update_send_patblt(rdpContext* context,
PATBLT_ORDER* patblt)
1627 WINPR_ASSERT(context);
1628 WINPR_ASSERT(patblt);
1631 const size_t headerLength = update_prepare_order_info(context, &orderInfo, ORDER_TYPE_PATBLT);
1632 if (!update_check_flush(context,
1633 headerLength + update_approximate_patblt_order(&orderInfo, patblt)))
1641 offset = Stream_GetPosition(s);
1643 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1646 Stream_Seek(s, headerLength);
1647 if (!update_write_patblt_order(s, &orderInfo, patblt))
1649 if (update_write_order_info(context, s, &orderInfo, offset) < 0)
1651 update->numberOrders++;
1655static BOOL update_send_scrblt(rdpContext* context,
const SCRBLT_ORDER* scrblt)
1659 WINPR_ASSERT(context);
1660 WINPR_ASSERT(scrblt);
1663 const size_t headerLength = update_prepare_order_info(context, &orderInfo, ORDER_TYPE_SCRBLT);
1664 const size_t inf = update_approximate_scrblt_order(&orderInfo, scrblt);
1665 if (!update_check_flush(context, headerLength + inf))
1673 const size_t offset = Stream_GetPosition(s);
1675 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1678 Stream_Seek(s, headerLength);
1679 if (!update_write_scrblt_order(s, &orderInfo, scrblt))
1681 if (update_write_order_info(context, s, &orderInfo, offset) < 0)
1683 update->numberOrders++;
1687static BOOL update_send_opaque_rect(rdpContext* context,
const OPAQUE_RECT_ORDER* opaque_rect)
1692 WINPR_ASSERT(context);
1693 WINPR_ASSERT(opaque_rect);
1696 const size_t headerLength =
1697 update_prepare_order_info(context, &orderInfo, ORDER_TYPE_OPAQUE_RECT);
1698 if (!update_check_flush(
1699 context, headerLength + update_approximate_opaque_rect_order(&orderInfo, opaque_rect)))
1707 offset = Stream_GetPosition(s);
1709 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1712 Stream_Seek(s, headerLength);
1713 if (!update_write_opaque_rect_order(s, &orderInfo, opaque_rect))
1715 if (update_write_order_info(context, s, &orderInfo, offset) < 0)
1717 update->numberOrders++;
1721static BOOL update_send_line_to(rdpContext* context,
const LINE_TO_ORDER* line_to)
1725 WINPR_ASSERT(context);
1726 WINPR_ASSERT(line_to);
1728 const size_t headerLength = update_prepare_order_info(context, &orderInfo, ORDER_TYPE_LINE_TO);
1729 const size_t inf = update_approximate_line_to_order(&orderInfo, line_to);
1730 if (!update_check_flush(context, headerLength + inf))
1738 const size_t offset = Stream_GetPosition(s);
1740 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1743 Stream_Seek(s, headerLength);
1744 if (!update_write_line_to_order(s, &orderInfo, line_to))
1746 if (update_write_order_info(context, s, &orderInfo, offset) < 0)
1748 update->numberOrders++;
1752static BOOL update_send_memblt(rdpContext* context,
MEMBLT_ORDER* memblt)
1757 WINPR_ASSERT(context);
1758 WINPR_ASSERT(memblt);
1760 const size_t headerLength = update_prepare_order_info(context, &orderInfo, ORDER_TYPE_MEMBLT);
1761 if (!update_check_flush(context,
1762 headerLength + update_approximate_memblt_order(&orderInfo, memblt)))
1770 offset = Stream_GetPosition(s);
1772 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1775 Stream_Seek(s, headerLength);
1776 if (!update_write_memblt_order(s, &orderInfo, memblt))
1778 if (update_write_order_info(context, s, &orderInfo, offset) < 0)
1780 update->numberOrders++;
1784static BOOL update_send_glyph_index(rdpContext* context,
GLYPH_INDEX_ORDER* glyph_index)
1788 WINPR_ASSERT(context);
1789 WINPR_ASSERT(glyph_index);
1792 const size_t headerLength =
1793 update_prepare_order_info(context, &orderInfo, ORDER_TYPE_GLYPH_INDEX);
1794 const size_t inf = update_approximate_glyph_index_order(&orderInfo, glyph_index);
1795 if (!update_check_flush(context, headerLength + inf))
1803 const size_t offset = Stream_GetPosition(s);
1805 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1808 Stream_Seek(s, headerLength);
1809 if (!update_write_glyph_index_order(s, &orderInfo, glyph_index))
1811 if (update_write_order_info(context, s, &orderInfo, offset) < 0)
1813 update->numberOrders++;
1821static BOOL update_send_cache_bitmap(rdpContext* context,
const CACHE_BITMAP_ORDER* cache_bitmap)
1823 const size_t headerLength = 6;
1824 UINT16 extraFlags = 0;
1826 WINPR_ASSERT(context);
1827 WINPR_ASSERT(cache_bitmap);
1830 const BYTE orderType = cache_bitmap->compressed ? ORDER_TYPE_CACHE_BITMAP_COMPRESSED
1831 : ORDER_TYPE_BITMAP_UNCOMPRESSED;
1833 update_approximate_cache_bitmap_order(cache_bitmap, cache_bitmap->compressed, &extraFlags);
1834 if (!update_check_flush(context, headerLength + inf))
1842 const size_t bm = Stream_GetPosition(s);
1844 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1847 Stream_Seek(s, headerLength);
1849 if (!update_write_cache_bitmap_order(s, cache_bitmap, cache_bitmap->compressed, &extraFlags))
1852 const size_t em = Stream_GetPosition(s);
1853 WINPR_ASSERT(em >= bm + 13);
1854 const size_t orderLength = (em - bm) - 13;
1855 WINPR_ASSERT(orderLength <= UINT16_MAX);
1857 Stream_SetPosition(s, bm);
1858 Stream_Write_UINT8(s, ORDER_STANDARD | ORDER_SECONDARY);
1859 Stream_Write_UINT16(s, (UINT16)orderLength);
1860 Stream_Write_UINT16(s, extraFlags);
1861 Stream_Write_UINT8(s, orderType);
1862 Stream_SetPosition(s, em);
1863 update->numberOrders++;
1869 const size_t headerLength = 6;
1870 UINT16 extraFlags = 0;
1872 WINPR_ASSERT(context);
1873 WINPR_ASSERT(cache_bitmap_v2);
1876 const BYTE orderType = cache_bitmap_v2->compressed ? ORDER_TYPE_BITMAP_COMPRESSED_V2
1877 : ORDER_TYPE_BITMAP_UNCOMPRESSED_V2;
1879 if (context->settings->NoBitmapCompressionHeader)
1880 cache_bitmap_v2->flags |= CBR2_NO_BITMAP_COMPRESSION_HDR;
1882 if (!update_check_flush(
1883 context, headerLength + update_approximate_cache_bitmap_v2_order(
1884 cache_bitmap_v2, cache_bitmap_v2->compressed, &extraFlags)))
1892 const size_t bm = Stream_GetPosition(s);
1894 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1897 Stream_Seek(s, headerLength);
1899 if (!update_write_cache_bitmap_v2_order(s, cache_bitmap_v2, cache_bitmap_v2->compressed,
1903 const size_t em = Stream_GetPosition(s);
1904 WINPR_ASSERT(em >= bm + 13);
1905 const size_t orderLength = (em - bm) - 13;
1906 WINPR_ASSERT(orderLength <= UINT16_MAX);
1908 Stream_SetPosition(s, bm);
1909 Stream_Write_UINT8(s, ORDER_STANDARD | ORDER_SECONDARY);
1910 Stream_Write_UINT16(s, (UINT16)orderLength);
1911 Stream_Write_UINT16(s, extraFlags);
1912 Stream_Write_UINT8(s, orderType);
1913 Stream_SetPosition(s, em);
1914 update->numberOrders++;
1920 const size_t headerLength = 6;
1921 UINT16 extraFlags = 0;
1923 WINPR_ASSERT(context);
1924 WINPR_ASSERT(cache_bitmap_v3);
1927 const BYTE orderType = ORDER_TYPE_BITMAP_COMPRESSED_V3;
1928 if (!update_check_flush(context, headerLength + update_approximate_cache_bitmap_v3_order(
1929 cache_bitmap_v3, &extraFlags)))
1937 const size_t bm = Stream_GetPosition(s);
1939 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1942 Stream_Seek(s, headerLength);
1944 if (!update_write_cache_bitmap_v3_order(s, cache_bitmap_v3, &extraFlags))
1947 const size_t em = Stream_GetPosition(s);
1948 WINPR_ASSERT(em >= bm + 13);
1949 const size_t orderLength = (em - bm) - 13;
1950 WINPR_ASSERT(orderLength <= UINT16_MAX);
1952 Stream_SetPosition(s, bm);
1953 Stream_Write_UINT8(s, ORDER_STANDARD | ORDER_SECONDARY);
1954 Stream_Write_UINT16(s, (UINT16)orderLength);
1955 Stream_Write_UINT16(s, extraFlags);
1956 Stream_Write_UINT8(s, orderType);
1957 Stream_SetPosition(s, em);
1958 update->numberOrders++;
1962static BOOL update_send_cache_color_table(rdpContext* context,
1966 size_t headerLength = 6;
1968 WINPR_ASSERT(context);
1969 WINPR_ASSERT(cache_color_table);
1972 const size_t inf = update_approximate_cache_color_table_order(cache_color_table, &flags);
1973 if (!update_check_flush(context, headerLength + inf))
1981 const size_t bm = Stream_GetPosition(s);
1983 if (!Stream_EnsureRemainingCapacity(s, headerLength))
1986 Stream_Seek(s, headerLength);
1988 if (!update_write_cache_color_table_order(s, cache_color_table, &flags))
1991 const size_t em = Stream_GetPosition(s);
1992 WINPR_ASSERT(em >= bm + 13);
1993 const size_t orderLength = (em - bm) - 13;
1994 WINPR_ASSERT(orderLength <= UINT16_MAX);
1995 Stream_SetPosition(s, bm);
1996 Stream_Write_UINT8(s, ORDER_STANDARD | ORDER_SECONDARY);
1997 Stream_Write_UINT16(s, (UINT16)orderLength);
1998 Stream_Write_UINT16(s, flags);
1999 Stream_Write_UINT8(s, ORDER_TYPE_CACHE_COLOR_TABLE);
2000 Stream_SetPosition(s, em);
2001 update->numberOrders++;
2005static BOOL update_send_cache_glyph(rdpContext* context,
const CACHE_GLYPH_ORDER* cache_glyph)
2008 const size_t headerLength = 6;
2010 WINPR_ASSERT(context);
2011 WINPR_ASSERT(cache_glyph);
2014 const size_t inf = update_approximate_cache_glyph_order(cache_glyph, &flags);
2015 if (!update_check_flush(context, headerLength + inf))
2023 const size_t bm = Stream_GetPosition(s);
2025 if (!Stream_EnsureRemainingCapacity(s, headerLength))
2028 Stream_Seek(s, headerLength);
2030 if (!update_write_cache_glyph_order(s, cache_glyph, &flags))
2033 const size_t em = Stream_GetPosition(s);
2034 WINPR_ASSERT(em >= bm + 13);
2035 const size_t orderLength = (em - bm) - 13;
2036 WINPR_ASSERT(orderLength <= UINT16_MAX);
2037 Stream_SetPosition(s, bm);
2038 Stream_Write_UINT8(s, ORDER_STANDARD | ORDER_SECONDARY);
2039 Stream_Write_UINT16(s, (UINT16)orderLength);
2040 Stream_Write_UINT16(s, flags);
2041 Stream_Write_UINT8(s, ORDER_TYPE_CACHE_GLYPH);
2042 Stream_SetPosition(s, em);
2043 update->numberOrders++;
2047static BOOL update_send_cache_glyph_v2(rdpContext* context,
2051 const size_t headerLength = 6;
2053 WINPR_ASSERT(context);
2054 WINPR_ASSERT(cache_glyph_v2);
2057 const size_t inf = update_approximate_cache_glyph_v2_order(cache_glyph_v2, &flags);
2058 if (!update_check_flush(context, headerLength + inf))
2066 const size_t bm = Stream_GetPosition(s);
2068 if (!Stream_EnsureRemainingCapacity(s, headerLength))
2071 Stream_Seek(s, headerLength);
2073 if (!update_write_cache_glyph_v2_order(s, cache_glyph_v2, &flags))
2076 const size_t em = Stream_GetPosition(s);
2077 WINPR_ASSERT(em >= bm + 13);
2078 const size_t orderLength = (em - bm) - 13;
2079 WINPR_ASSERT(orderLength <= UINT16_MAX);
2080 Stream_SetPosition(s, bm);
2081 Stream_Write_UINT8(s, ORDER_STANDARD | ORDER_SECONDARY);
2082 Stream_Write_UINT16(s, (UINT16)orderLength);
2083 Stream_Write_UINT16(s, flags);
2084 Stream_Write_UINT8(s, ORDER_TYPE_CACHE_GLYPH);
2085 Stream_SetPosition(s, em);
2086 update->numberOrders++;
2090static BOOL update_send_cache_brush(rdpContext* context,
const CACHE_BRUSH_ORDER* cache_brush)
2093 const size_t headerLength = 6;
2095 WINPR_ASSERT(context);
2096 WINPR_ASSERT(cache_brush);
2099 const size_t inf = update_approximate_cache_brush_order(cache_brush, &flags);
2100 if (!update_check_flush(context, headerLength + inf))
2108 const size_t bm = Stream_GetPosition(s);
2110 if (!Stream_EnsureRemainingCapacity(s, headerLength))
2113 Stream_Seek(s, headerLength);
2115 if (!update_write_cache_brush_order(s, cache_brush, &flags))
2118 const size_t em = Stream_GetPosition(s);
2122 const size_t orderLength = (em - bm) - 13;
2123 WINPR_ASSERT(orderLength <= UINT16_MAX);
2124 Stream_SetPosition(s, bm);
2125 Stream_Write_UINT8(s, ORDER_STANDARD | ORDER_SECONDARY);
2126 Stream_Write_UINT16(s, (UINT16)orderLength);
2127 Stream_Write_UINT16(s, flags);
2128 Stream_Write_UINT8(s, ORDER_TYPE_CACHE_BRUSH);
2129 Stream_SetPosition(s, em);
2130 update->numberOrders++;
2138static BOOL update_send_create_offscreen_bitmap_order(
2141 WINPR_ASSERT(context);
2142 WINPR_ASSERT(create_offscreen_bitmap);
2145 const size_t headerLength = 1;
2146 const size_t orderType = ORDER_TYPE_CREATE_OFFSCREEN_BITMAP;
2147 const size_t controlFlags = ORDER_SECONDARY | (orderType << 2);
2148 const size_t inf = update_approximate_create_offscreen_bitmap_order(create_offscreen_bitmap);
2149 if (!update_check_flush(context, headerLength + inf))
2157 const size_t bm = Stream_GetPosition(s);
2159 if (!Stream_EnsureRemainingCapacity(s, headerLength))
2162 Stream_Seek(s, headerLength);
2164 if (!update_write_create_offscreen_bitmap_order(s, create_offscreen_bitmap))
2167 const size_t em = Stream_GetPosition(s);
2168 Stream_SetPosition(s, bm);
2169 Stream_Write_UINT8(s,
2170 WINPR_ASSERTING_INT_CAST(uint8_t, controlFlags));
2171 Stream_SetPosition(s, em);
2172 update->numberOrders++;
2176static BOOL update_send_switch_surface_order(rdpContext* context,
2179 WINPR_ASSERT(context);
2180 WINPR_ASSERT(switch_surface);
2183 const size_t headerLength = 1;
2184 const size_t orderType = ORDER_TYPE_SWITCH_SURFACE;
2185 const size_t controlFlags = ORDER_SECONDARY | (orderType << 2);
2186 const size_t inf = update_approximate_switch_surface_order(switch_surface);
2187 if (!update_check_flush(context, headerLength + inf))
2195 const size_t bm = Stream_GetPosition(s);
2197 if (!Stream_EnsureRemainingCapacity(s, headerLength))
2200 Stream_Seek(s, headerLength);
2202 if (!update_write_switch_surface_order(s, switch_surface))
2205 const size_t em = Stream_GetPosition(s);
2206 Stream_SetPosition(s, bm);
2207 Stream_Write_UINT8(s,
2208 WINPR_ASSERTING_INT_CAST(uint8_t, controlFlags));
2209 Stream_SetPosition(s, em);
2210 update->numberOrders++;
2214static BOOL update_send_pointer_system(rdpContext* context,
2218 BYTE updateCode = 0;
2220 WINPR_ASSERT(context);
2221 rdpRdp* rdp = context->rdp;
2225 s = fastpath_update_pdu_init(rdp->fastpath);
2230 if (pointer_system->type == SYSPTR_NULL)
2231 updateCode = FASTPATH_UPDATETYPE_PTR_NULL;
2233 updateCode = FASTPATH_UPDATETYPE_PTR_DEFAULT;
2235 ret = fastpath_send_update_pdu(rdp->fastpath, updateCode, s, FALSE);
2240static BOOL update_send_pointer_position(rdpContext* context,
2244 WINPR_ASSERT(context);
2245 rdpRdp* rdp = context->rdp;
2249 s = fastpath_update_pdu_init(rdp->fastpath);
2254 if (!Stream_EnsureRemainingCapacity(s, 16))
2257 Stream_Write_UINT16(
2258 s, WINPR_ASSERTING_INT_CAST(uint16_t, pointerPosition->xPos));
2259 Stream_Write_UINT16(
2260 s, WINPR_ASSERTING_INT_CAST(uint16_t, pointerPosition->yPos));
2261 ret = fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_PTR_POSITION, s, FALSE);
2269 WINPR_ASSERT(pointer_color);
2270 if (!Stream_EnsureRemainingCapacity(s, 32 + pointer_color->lengthAndMask +
2271 pointer_color->lengthXorMask))
2274 Stream_Write_UINT16(s, pointer_color->cacheIndex);
2275 Stream_Write_UINT16(s, pointer_color->hotSpotX);
2276 Stream_Write_UINT16(s, pointer_color->hotSpotY);
2277 Stream_Write_UINT16(s, pointer_color->width);
2278 Stream_Write_UINT16(s, pointer_color->height);
2279 Stream_Write_UINT16(s, pointer_color->lengthAndMask);
2280 Stream_Write_UINT16(s, pointer_color->lengthXorMask);
2282 if (pointer_color->lengthXorMask > 0)
2283 Stream_Write(s, pointer_color->xorMaskData, pointer_color->lengthXorMask);
2285 if (pointer_color->lengthAndMask > 0)
2286 Stream_Write(s, pointer_color->andMaskData, pointer_color->lengthAndMask);
2288 Stream_Write_UINT8(s, 0);
2292static BOOL update_send_pointer_color(rdpContext* context,
2297 WINPR_ASSERT(context);
2298 rdpRdp* rdp = context->rdp;
2302 WINPR_ASSERT(pointer_color);
2303 s = fastpath_update_pdu_init(rdp->fastpath);
2308 if (!update_write_pointer_color(s, pointer_color))
2311 ret = fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_COLOR, s, FALSE);
2319 WINPR_ASSERT(pointer);
2321 if (!Stream_EnsureRemainingCapacity(s, 32 + pointer->lengthAndMask + pointer->lengthXorMask))
2324 Stream_Write_UINT16(s, pointer->xorBpp);
2325 Stream_Write_UINT16(s, pointer->cacheIndex);
2326 Stream_Write_UINT16(s, pointer->hotSpotX);
2327 Stream_Write_UINT16(s, pointer->hotSpotY);
2328 Stream_Write_UINT16(s, pointer->width);
2329 Stream_Write_UINT16(s, pointer->height);
2330 Stream_Write_UINT32(s, pointer->lengthAndMask);
2331 Stream_Write_UINT32(s, pointer->lengthXorMask);
2332 Stream_Write(s, pointer->xorMaskData, pointer->lengthXorMask);
2333 Stream_Write(s, pointer->andMaskData, pointer->lengthAndMask);
2334 Stream_Write_UINT8(s, 0);
2341 WINPR_ASSERT(context);
2342 rdpRdp* rdp = context->rdp;
2346 WINPR_ASSERT(pointer);
2347 s = fastpath_update_pdu_init(rdp->fastpath);
2352 if (!update_write_pointer_large(s, pointer))
2355 ret = fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_LARGE_POINTER, s, FALSE);
2361static BOOL update_send_pointer_new(rdpContext* context,
const POINTER_NEW_UPDATE* pointer_new)
2365 WINPR_ASSERT(context);
2366 rdpRdp* rdp = context->rdp;
2370 WINPR_ASSERT(pointer_new);
2371 s = fastpath_update_pdu_init(rdp->fastpath);
2376 if (!Stream_EnsureRemainingCapacity(s, 16))
2379 Stream_Write_UINT16(
2380 s, WINPR_ASSERTING_INT_CAST(uint16_t, pointer_new->xorBpp));
2381 update_write_pointer_color(s, &pointer_new->colorPtrAttr);
2382 ret = fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_POINTER, s, FALSE);
2388static BOOL update_send_pointer_cached(rdpContext* context,
2393 WINPR_ASSERT(context);
2394 rdpRdp* rdp = context->rdp;
2398 WINPR_ASSERT(pointer_cached);
2399 s = fastpath_update_pdu_init(rdp->fastpath);
2404 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(
2405 uint16_t, pointer_cached->cacheIndex));
2406 ret = fastpath_send_update_pdu(rdp->fastpath, FASTPATH_UPDATETYPE_CACHED, s, FALSE);
2411BOOL update_read_refresh_rect(rdpUpdate* update,
wStream* s)
2413 BYTE numberOfAreas = 0;
2417 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
2420 Stream_Read_UINT8(s, numberOfAreas);
2423 if (!Stream_CheckAndLogRequiredLengthOfSize(TAG, s, numberOfAreas, 8ull))
2426 for (BYTE index = 0; index < numberOfAreas; index++)
2430 Stream_Read_UINT16(s, area->left);
2431 Stream_Read_UINT16(s, area->top);
2432 Stream_Read_UINT16(s, area->right);
2433 Stream_Read_UINT16(s, area->bottom);
2436 WINPR_ASSERT(update->context);
2437 WINPR_ASSERT(update->context->settings);
2438 if (update->context->settings->RefreshRect)
2439 IFCALL(update->RefreshRect, update->context, numberOfAreas, areas);
2441 WLog_Print(up->log, WLOG_WARN,
"ignoring refresh rect request from client");
2446BOOL update_read_suppress_output(rdpUpdate* update,
wStream* s)
2451 BYTE allowDisplayUpdates = 0;
2456 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
2459 Stream_Read_UINT8(s, allowDisplayUpdates);
2462 if (allowDisplayUpdates > 0)
2464 if (!Stream_CheckAndLogRequiredLength(TAG, s,
sizeof(
RECTANGLE_16)))
2467 Stream_Read_UINT16(s, rect.left);
2468 Stream_Read_UINT16(s, rect.top);
2469 Stream_Read_UINT16(s, rect.right);
2470 Stream_Read_UINT16(s, rect.bottom);
2475 WINPR_ASSERT(update->context);
2476 WINPR_ASSERT(update->context->settings);
2477 if (update->context->settings->SuppressOutput)
2478 IFCALL(update->SuppressOutput, update->context, allowDisplayUpdates, prect);
2480 WLog_Print(up->log, WLOG_WARN,
"ignoring suppress output request from client");
2485static BOOL update_send_set_keyboard_indicators(rdpContext* context, UINT16 led_flags)
2487 UINT16 sec_flags = 0;
2490 WINPR_ASSERT(context);
2491 rdpRdp* rdp = context->rdp;
2492 s = rdp_data_pdu_init(rdp, &sec_flags);
2497 Stream_Write_UINT16(s, 0);
2498 Stream_Write_UINT16(s, led_flags);
2500 WINPR_ASSERT(rdp->mcs);
2501 return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_SET_KEYBOARD_INDICATORS, rdp->mcs->userId,
2505static BOOL update_send_set_keyboard_ime_status(rdpContext* context, UINT16 imeId, UINT32 imeState,
2508 UINT16 sec_flags = 0;
2511 WINPR_ASSERT(context);
2512 rdpRdp* rdp = context->rdp;
2513 s = rdp_data_pdu_init(rdp, &sec_flags);
2519 Stream_Write_UINT16(s, imeId);
2520 Stream_Write_UINT32(s, imeState);
2521 Stream_Write_UINT32(s, imeConvMode);
2523 WINPR_ASSERT(rdp->mcs);
2524 return rdp_send_data_pdu(rdp, s, DATA_PDU_TYPE_SET_KEYBOARD_IME_STATUS, rdp->mcs->userId,
2528static UINT16 update_calculate_new_or_existing_window(
const WINDOW_ORDER_INFO* orderInfo,
2531 size_t orderSize = 11;
2533 WINPR_ASSERT(orderInfo);
2534 WINPR_ASSERT(stateOrder);
2536 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_OWNER) != 0)
2539 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_STYLE) != 0)
2542 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_SHOW) != 0)
2545 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_TITLE) != 0)
2546 orderSize += 2 + stateOrder->titleInfo.length;
2548 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET) != 0)
2551 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE) != 0)
2554 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_X) != 0)
2557 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_Y) != 0)
2560 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_RP_CONTENT) != 0)
2563 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_ROOT_PARENT) != 0)
2566 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET) != 0)
2569 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA) != 0)
2572 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE) != 0)
2575 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS) != 0)
2577 const size_t len = 2ULL + stateOrder->numWindowRects *
sizeof(
RECTANGLE_16);
2581 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET) != 0)
2584 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY) != 0)
2587 const size_t len = 2ULL + stateOrder->numVisibilityRects *
sizeof(
RECTANGLE_16);
2591 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_OVERLAY_DESCRIPTION) != 0)
2592 orderSize += 2 + stateOrder->OverlayDescription.length;
2594 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_TASKBAR_BUTTON) != 0)
2597 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_ENFORCE_SERVER_ZORDER) != 0)
2600 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_APPBAR_STATE) != 0)
2603 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_APPBAR_EDGE) != 0)
2606 return WINPR_ASSERTING_INT_CAST(uint16_t, orderSize);
2609static BOOL update_write_order_field_flags(UINT32 fieldFlags,
const WINDOW_STATE_ORDER* stateOrder,
2612 WINPR_ASSERT(stateOrder);
2614 if ((fieldFlags & WINDOW_ORDER_FIELD_OWNER) != 0)
2615 Stream_Write_UINT32(s, stateOrder->ownerWindowId);
2617 if ((fieldFlags & WINDOW_ORDER_FIELD_STYLE) != 0)
2619 Stream_Write_UINT32(s, stateOrder->style);
2620 Stream_Write_UINT32(s, stateOrder->extendedStyle);
2623 if ((fieldFlags & WINDOW_ORDER_FIELD_SHOW) != 0)
2625 Stream_Write_UINT8(s, WINPR_ASSERTING_INT_CAST(uint8_t, stateOrder->showState));
2628 if ((fieldFlags & WINDOW_ORDER_FIELD_TITLE) != 0)
2630 Stream_Write_UINT16(s, stateOrder->titleInfo.length);
2631 Stream_Write(s, stateOrder->titleInfo.string, stateOrder->titleInfo.length);
2634 if ((fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_OFFSET) != 0)
2636 Stream_Write_INT32(s, stateOrder->clientOffsetX);
2637 Stream_Write_INT32(s, stateOrder->clientOffsetY);
2640 if ((fieldFlags & WINDOW_ORDER_FIELD_CLIENT_AREA_SIZE) != 0)
2642 Stream_Write_UINT32(s, stateOrder->clientAreaWidth);
2643 Stream_Write_UINT32(s, stateOrder->clientAreaHeight);
2646 if ((fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_X) != 0)
2648 Stream_Write_UINT32(s, stateOrder->resizeMarginLeft);
2649 Stream_Write_UINT32(s, stateOrder->resizeMarginRight);
2652 if ((fieldFlags & WINDOW_ORDER_FIELD_RESIZE_MARGIN_Y) != 0)
2654 Stream_Write_UINT32(s, stateOrder->resizeMarginTop);
2655 Stream_Write_UINT32(s, stateOrder->resizeMarginBottom);
2658 if ((fieldFlags & WINDOW_ORDER_FIELD_RP_CONTENT) != 0)
2660 Stream_Write_UINT8(s, WINPR_ASSERTING_INT_CAST(uint8_t, stateOrder->RPContent));
2663 if ((fieldFlags & WINDOW_ORDER_FIELD_ROOT_PARENT) != 0)
2665 Stream_Write_UINT32(s, stateOrder->rootParentHandle);
2668 if ((fieldFlags & WINDOW_ORDER_FIELD_WND_OFFSET) != 0)
2670 Stream_Write_INT32(s, stateOrder->windowOffsetX);
2671 Stream_Write_INT32(s, stateOrder->windowOffsetY);
2674 if ((fieldFlags & WINDOW_ORDER_FIELD_WND_CLIENT_DELTA) != 0)
2676 Stream_Write_INT32(s, stateOrder->windowClientDeltaX);
2677 Stream_Write_INT32(s, stateOrder->windowClientDeltaY);
2680 if ((fieldFlags & WINDOW_ORDER_FIELD_WND_SIZE) != 0)
2682 Stream_Write_UINT32(s, stateOrder->windowWidth);
2683 Stream_Write_UINT32(s, stateOrder->windowHeight);
2686 if ((fieldFlags & WINDOW_ORDER_FIELD_WND_RECTS) != 0)
2688 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, stateOrder->numWindowRects));
2689 Stream_Write(s, stateOrder->windowRects, stateOrder->numWindowRects *
sizeof(
RECTANGLE_16));
2692 if ((fieldFlags & WINDOW_ORDER_FIELD_VIS_OFFSET) != 0)
2694 Stream_Write_INT32(s, stateOrder->visibleOffsetX);
2695 Stream_Write_INT32(s, stateOrder->visibleOffsetY);
2698 if ((fieldFlags & WINDOW_ORDER_FIELD_VISIBILITY) != 0)
2700 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, stateOrder->numVisibilityRects));
2701 Stream_Write(s, stateOrder->visibilityRects,
2702 stateOrder->numVisibilityRects *
sizeof(
RECTANGLE_16));
2705 if ((fieldFlags & WINDOW_ORDER_FIELD_OVERLAY_DESCRIPTION) != 0)
2707 Stream_Write_UINT16(s, stateOrder->OverlayDescription.length);
2708 Stream_Write(s, stateOrder->OverlayDescription.string,
2709 stateOrder->OverlayDescription.length);
2712 if ((fieldFlags & WINDOW_ORDER_FIELD_TASKBAR_BUTTON) != 0)
2714 Stream_Write_UINT8(s, stateOrder->TaskbarButton);
2717 if ((fieldFlags & WINDOW_ORDER_FIELD_ENFORCE_SERVER_ZORDER) != 0)
2719 Stream_Write_UINT8(s, stateOrder->EnforceServerZOrder);
2722 if ((fieldFlags & WINDOW_ORDER_FIELD_APPBAR_STATE) != 0)
2724 Stream_Write_UINT8(s, stateOrder->AppBarState);
2727 if ((fieldFlags & WINDOW_ORDER_FIELD_APPBAR_EDGE) != 0)
2729 Stream_Write_UINT8(s, stateOrder->AppBarEdge);
2735static BOOL update_send_new_or_existing_window(rdpContext* context,
2739 BYTE controlFlags = ORDER_SECONDARY | (ORDER_TYPE_WINDOW << 2);
2740 UINT16 orderSize = update_calculate_new_or_existing_window(orderInfo, stateOrder);
2742 WINPR_ASSERT(context);
2743 WINPR_ASSERT(orderInfo);
2744 WINPR_ASSERT(stateOrder);
2748 if (!update_check_flush(context, orderSize))
2756 if (!Stream_EnsureRemainingCapacity(s, orderSize))
2759 Stream_Write_UINT8(s, controlFlags);
2760 Stream_Write_UINT16(s, orderSize);
2761 Stream_Write_UINT32(s, orderInfo->fieldFlags);
2762 Stream_Write_UINT32(s, orderInfo->windowId);
2764 if (!update_write_order_field_flags(orderInfo->fieldFlags, stateOrder, s))
2767 update->numberOrders++;
2771static BOOL update_send_window_create(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
2774 return update_send_new_or_existing_window(context, orderInfo, stateOrder);
2777static BOOL update_send_window_update(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
2780 return update_send_new_or_existing_window(context, orderInfo, stateOrder);
2784update_calculate_window_icon_order(WINPR_ATTR_UNUSED
const WINDOW_ORDER_INFO* orderInfo,
2787 UINT16 orderSize = 23;
2789 WINPR_ASSERT(iconOrder);
2790 ICON_INFO* iconInfo = iconOrder->iconInfo;
2791 WINPR_ASSERT(iconInfo);
2793 orderSize += iconInfo->cbBitsColor + iconInfo->cbBitsMask;
2795 if (iconInfo->bpp <= 8)
2796 orderSize += 2 + iconInfo->cbColorTable;
2801static BOOL update_send_window_icon(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
2804 BYTE controlFlags = ORDER_SECONDARY | (ORDER_TYPE_WINDOW << 2);
2806 WINPR_ASSERT(iconOrder);
2807 ICON_INFO* iconInfo = iconOrder->iconInfo;
2808 UINT16 orderSize = update_calculate_window_icon_order(orderInfo, iconOrder);
2810 WINPR_ASSERT(context);
2811 WINPR_ASSERT(orderInfo);
2812 WINPR_ASSERT(iconInfo);
2816 if (!update_check_flush(context, orderSize))
2821 if (!s || !iconInfo)
2824 if (!Stream_EnsureRemainingCapacity(s, orderSize))
2828 Stream_Write_UINT8(s, controlFlags);
2829 Stream_Write_UINT16(s, orderSize);
2830 Stream_Write_UINT32(s, orderInfo->fieldFlags);
2831 Stream_Write_UINT32(s, orderInfo->windowId);
2833 Stream_Write_UINT16(
2834 s, WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->cacheEntry));
2835 Stream_Write_UINT8(s,
2836 WINPR_ASSERTING_INT_CAST(uint8_t, iconInfo->cacheId));
2837 Stream_Write_UINT8(s, WINPR_ASSERTING_INT_CAST(uint8_t, iconInfo->bpp));
2838 Stream_Write_UINT16(s,
2839 WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->width));
2840 Stream_Write_UINT16(
2841 s, WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->height));
2843 if (iconInfo->bpp <= 8)
2845 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(
2846 uint16_t, iconInfo->cbColorTable));
2849 Stream_Write_UINT16(
2850 s, WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->cbBitsMask));
2851 Stream_Write_UINT16(
2852 s, WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->cbBitsColor));
2853 Stream_Write(s, iconInfo->bitsMask, iconInfo->cbBitsMask);
2855 if (iconInfo->bpp <= 8)
2857 Stream_Write(s, iconInfo->colorTable, iconInfo->cbColorTable);
2860 Stream_Write(s, iconInfo->bitsColor, iconInfo->cbBitsColor);
2862 update->numberOrders++;
2866static BOOL update_send_window_cached_icon(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
2869 BYTE controlFlags = ORDER_SECONDARY | (ORDER_TYPE_WINDOW << 2);
2870 UINT16 orderSize = 14;
2872 WINPR_ASSERT(cachedIconOrder);
2875 WINPR_ASSERT(context);
2876 WINPR_ASSERT(orderInfo);
2877 WINPR_ASSERT(cachedIcon);
2881 if (!update_check_flush(context, orderSize))
2888 if (!Stream_EnsureRemainingCapacity(s, orderSize))
2892 Stream_Write_UINT8(s, controlFlags);
2893 Stream_Write_UINT16(s, orderSize);
2894 Stream_Write_UINT32(s, orderInfo->fieldFlags);
2895 Stream_Write_UINT32(s, orderInfo->windowId);
2897 Stream_Write_UINT16(
2898 s, WINPR_ASSERTING_INT_CAST(uint16_t, cachedIcon->cacheEntry));
2900 s, WINPR_ASSERTING_INT_CAST(uint8_t, cachedIcon->cacheId));
2901 update->numberOrders++;
2905static BOOL update_send_window_delete(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo)
2907 BYTE controlFlags = ORDER_SECONDARY | (ORDER_TYPE_WINDOW << 2);
2908 UINT16 orderSize = 11;
2910 WINPR_ASSERT(context);
2911 WINPR_ASSERT(orderInfo);
2914 if (!update_check_flush(context, orderSize))
2922 if (!Stream_EnsureRemainingCapacity(s, orderSize))
2926 Stream_Write_UINT8(s, controlFlags);
2927 Stream_Write_UINT16(s, orderSize);
2928 Stream_Write_UINT32(s, orderInfo->fieldFlags);
2929 Stream_Write_UINT32(s, orderInfo->windowId);
2930 update->numberOrders++;
2934static UINT16 update_calculate_new_or_existing_notification_icons_order(
2937 UINT16 orderSize = 15;
2939 WINPR_ASSERT(orderInfo);
2940 WINPR_ASSERT(iconStateOrder);
2942 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_VERSION) != 0)
2945 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_TIP) != 0)
2947 orderSize += 2 + iconStateOrder->toolTip.length;
2950 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_INFO_TIP) != 0)
2953 orderSize += 12 + infoTip.text.length + infoTip.title.length;
2956 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_STATE) != 0)
2961 if ((orderInfo->fieldFlags & WINDOW_ORDER_ICON) != 0)
2963 ICON_INFO iconInfo = iconStateOrder->icon;
2966 if (iconInfo.bpp <= 8)
2967 orderSize += 2 + iconInfo.cbColorTable;
2969 orderSize += iconInfo.cbBitsMask + iconInfo.cbBitsColor;
2971 else if ((orderInfo->fieldFlags & WINDOW_ORDER_CACHED_ICON) != 0)
2979static BOOL update_send_new_or_existing_order_icon(
const ICON_INFO* iconInfo,
wStream* s)
2981 WINPR_ASSERT(iconInfo);
2983 if (!Stream_EnsureRemainingCapacity(s, 8))
2986 Stream_Write_UINT16(
2987 s, WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->cacheEntry));
2988 Stream_Write_UINT8(s,
2989 WINPR_ASSERTING_INT_CAST(uint8_t, iconInfo->cacheId));
2990 Stream_Write_UINT8(s, WINPR_ASSERTING_INT_CAST(uint8_t, iconInfo->bpp));
2991 Stream_Write_UINT16(s,
2992 WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->width));
2993 Stream_Write_UINT16(
2994 s, WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->height));
2996 if (iconInfo->bpp <= 8)
2998 if (!Stream_EnsureRemainingCapacity(s, 2))
3000 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(
3001 uint16_t, iconInfo->cbColorTable));
3004 if (!Stream_EnsureRemainingCapacity(s, 4ULL + iconInfo->cbBitsMask))
3006 Stream_Write_UINT16(
3007 s, WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->cbBitsMask));
3008 Stream_Write_UINT16(
3009 s, WINPR_ASSERTING_INT_CAST(uint16_t, iconInfo->cbBitsColor));
3010 Stream_Write(s, iconInfo->bitsMask, iconInfo->cbBitsMask);
3012 if (iconInfo->bpp <= 8)
3014 if (!Stream_EnsureRemainingCapacity(s, iconInfo->cbColorTable))
3016 Stream_Write(s, iconInfo->colorTable, iconInfo->cbColorTable);
3019 if (!Stream_EnsureRemainingCapacity(s, iconInfo->cbBitsColor))
3021 Stream_Write(s, iconInfo->bitsColor, iconInfo->cbBitsColor);
3026update_send_new_or_existing_notification_icons(rdpContext* context,
3030 BYTE controlFlags = ORDER_SECONDARY | (ORDER_TYPE_WINDOW << 2);
3031 BOOL versionFieldPresent = FALSE;
3032 const UINT16 orderSize =
3033 update_calculate_new_or_existing_notification_icons_order(orderInfo, iconStateOrder);
3035 WINPR_ASSERT(context);
3036 WINPR_ASSERT(orderInfo);
3037 WINPR_ASSERT(iconStateOrder);
3040 if (!update_check_flush(context, orderSize))
3047 if (!Stream_EnsureRemainingCapacity(s, orderSize))
3051 Stream_Write_UINT8(s, controlFlags);
3052 Stream_Write_UINT16(s, orderSize);
3053 Stream_Write_UINT32(s, orderInfo->fieldFlags);
3054 Stream_Write_UINT32(s, orderInfo->windowId);
3055 Stream_Write_UINT32(s, orderInfo->notifyIconId);
3058 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_VERSION) != 0)
3060 versionFieldPresent = TRUE;
3061 Stream_Write_UINT32(s, iconStateOrder->version);
3064 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_TIP) != 0)
3066 Stream_Write_UINT16(s, iconStateOrder->toolTip.length);
3067 Stream_Write(s, iconStateOrder->toolTip.string, iconStateOrder->toolTip.length);
3070 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_INFO_TIP) != 0)
3075 if (versionFieldPresent && iconStateOrder->version == 0)
3078 Stream_Write_UINT32(s, infoTip.timeout);
3079 Stream_Write_UINT32(s, infoTip.flags);
3080 Stream_Write_UINT16(s, infoTip.text.length);
3081 Stream_Write(s, infoTip.text.string, infoTip.text.length);
3082 Stream_Write_UINT16(s, infoTip.title.length);
3083 Stream_Write(s, infoTip.title.string, infoTip.title.length);
3086 if ((orderInfo->fieldFlags & WINDOW_ORDER_FIELD_NOTIFY_STATE) != 0)
3089 if (versionFieldPresent && iconStateOrder->version == 0)
3092 Stream_Write_UINT32(s, iconStateOrder->state);
3095 if ((orderInfo->fieldFlags & WINDOW_ORDER_ICON) != 0)
3097 const ICON_INFO* iconInfo = &iconStateOrder->icon;
3099 if (!update_send_new_or_existing_order_icon(iconInfo, s))
3102 else if ((orderInfo->fieldFlags & WINDOW_ORDER_CACHED_ICON) != 0)
3105 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(
3106 uint16_t, cachedIcon.cacheEntry));
3108 s, WINPR_ASSERTING_INT_CAST(uint8_t, cachedIcon.cacheId));
3111 update->numberOrders++;
3115static BOOL update_send_notify_icon_create(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
3118 return update_send_new_or_existing_notification_icons(context, orderInfo, iconStateOrder);
3121static BOOL update_send_notify_icon_update(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
3124 return update_send_new_or_existing_notification_icons(context, orderInfo, iconStateOrder);
3127static BOOL update_send_notify_icon_delete(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo)
3129 BYTE controlFlags = ORDER_SECONDARY | (ORDER_TYPE_WINDOW << 2);
3130 UINT16 orderSize = 15;
3132 WINPR_ASSERT(context);
3133 WINPR_ASSERT(orderInfo);
3136 if (!update_check_flush(context, orderSize))
3145 Stream_Write_UINT8(s, controlFlags);
3146 Stream_Write_UINT16(s, orderSize);
3147 Stream_Write_UINT32(s, orderInfo->fieldFlags);
3148 Stream_Write_UINT32(s, orderInfo->windowId);
3149 Stream_Write_UINT32(s, orderInfo->notifyIconId);
3150 update->numberOrders++;
3154static UINT16 update_calculate_monitored_desktop(
const WINDOW_ORDER_INFO* orderInfo,
3157 UINT16 orderSize = 7;
3159 WINPR_ASSERT(orderInfo);
3160 WINPR_ASSERT(monitoredDesktop);
3162 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND)
3167 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ZORDER)
3169 orderSize += 1 + (4 * monitoredDesktop->numWindowIds);
3175static BOOL update_send_monitored_desktop(rdpContext* context,
const WINDOW_ORDER_INFO* orderInfo,
3178 BYTE controlFlags = ORDER_SECONDARY | (ORDER_TYPE_WINDOW << 2);
3179 UINT16 orderSize = update_calculate_monitored_desktop(orderInfo, monitoredDesktop);
3181 WINPR_ASSERT(context);
3182 WINPR_ASSERT(orderInfo);
3183 WINPR_ASSERT(monitoredDesktop);
3187 if (!update_check_flush(context, orderSize))
3195 Stream_Write_UINT8(s, controlFlags);
3196 Stream_Write_UINT16(s, orderSize);
3197 Stream_Write_UINT32(s, orderInfo->fieldFlags);
3199 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ACTIVE_WND)
3201 Stream_Write_UINT32(s, monitoredDesktop->activeWindowId);
3204 if (orderInfo->fieldFlags & WINDOW_ORDER_FIELD_DESKTOP_ZORDER)
3207 s, WINPR_ASSERTING_INT_CAST(
3208 uint8_t, monitoredDesktop->numWindowIds));
3211 for (UINT32 i = 0; i < monitoredDesktop->numWindowIds; i++)
3213 Stream_Write_UINT32(s,
3214 WINPR_ASSERTING_INT_CAST(uint32_t, monitoredDesktop->windowIds[i]));
3218 update->numberOrders++;
3222static BOOL update_send_non_monitored_desktop(rdpContext* context,
3225 BYTE controlFlags = ORDER_SECONDARY | (ORDER_TYPE_WINDOW << 2);
3226 UINT16 orderSize = 7;
3228 WINPR_ASSERT(context);
3229 WINPR_ASSERT(orderInfo);
3232 if (!update_check_flush(context, orderSize))
3240 Stream_Write_UINT8(s, controlFlags);
3241 Stream_Write_UINT16(s, orderSize);
3242 Stream_Write_UINT32(s, orderInfo->fieldFlags);
3243 update->numberOrders++;
3247void update_register_server_callbacks(rdpUpdate* update)
3249 WINPR_ASSERT(update);
3251 update->BeginPaint = s_update_begin_paint;
3252 update->EndPaint = s_update_end_paint;
3253 update->SetBounds = update_set_bounds;
3254 update->Synchronize = update_send_synchronize;
3255 update->DesktopResize = update_send_desktop_resize;
3256 update->BitmapUpdate = update_send_bitmap_update;
3257 update->SurfaceBits = update_send_surface_bits;
3258 update->SurfaceFrameMarker = update_send_surface_frame_marker;
3259 update->SurfaceCommand = update_send_surface_command;
3260 update->SurfaceFrameBits = update_send_surface_frame_bits;
3261 update->PlaySound = update_send_play_sound;
3262 update->SetKeyboardIndicators = update_send_set_keyboard_indicators;
3263 update->SetKeyboardImeStatus = update_send_set_keyboard_ime_status;
3264 update->SaveSessionInfo = rdp_send_save_session_info;
3265 update->ServerStatusInfo = rdp_send_server_status_info;
3266 update->primary->DstBlt = update_send_dstblt;
3267 update->primary->PatBlt = update_send_patblt;
3268 update->primary->ScrBlt = update_send_scrblt;
3269 update->primary->OpaqueRect = update_send_opaque_rect;
3270 update->primary->LineTo = update_send_line_to;
3271 update->primary->MemBlt = update_send_memblt;
3272 update->primary->GlyphIndex = update_send_glyph_index;
3273 update->secondary->CacheBitmap = update_send_cache_bitmap;
3274 update->secondary->CacheBitmapV2 = update_send_cache_bitmap_v2;
3275 update->secondary->CacheBitmapV3 = update_send_cache_bitmap_v3;
3276 update->secondary->CacheColorTable = update_send_cache_color_table;
3277 update->secondary->CacheGlyph = update_send_cache_glyph;
3278 update->secondary->CacheGlyphV2 = update_send_cache_glyph_v2;
3279 update->secondary->CacheBrush = update_send_cache_brush;
3280 update->altsec->CreateOffscreenBitmap = update_send_create_offscreen_bitmap_order;
3281 update->altsec->SwitchSurface = update_send_switch_surface_order;
3282 update->pointer->PointerSystem = update_send_pointer_system;
3283 update->pointer->PointerPosition = update_send_pointer_position;
3284 update->pointer->PointerColor = update_send_pointer_color;
3285 update->pointer->PointerLarge = update_send_pointer_large;
3286 update->pointer->PointerNew = update_send_pointer_new;
3287 update->pointer->PointerCached = update_send_pointer_cached;
3288 update->window->WindowCreate = update_send_window_create;
3289 update->window->WindowUpdate = update_send_window_update;
3290 update->window->WindowIcon = update_send_window_icon;
3291 update->window->WindowCachedIcon = update_send_window_cached_icon;
3292 update->window->WindowDelete = update_send_window_delete;
3293 update->window->NotifyIconCreate = update_send_notify_icon_create;
3294 update->window->NotifyIconUpdate = update_send_notify_icon_update;
3295 update->window->NotifyIconDelete = update_send_notify_icon_delete;
3296 update->window->MonitoredDesktop = update_send_monitored_desktop;
3297 update->window->NonMonitoredDesktop = update_send_non_monitored_desktop;
3300void update_register_client_callbacks(rdpUpdate* update)
3302 WINPR_ASSERT(update);
3304 update->RefreshRect = update_send_refresh_rect;
3305 update->SuppressOutput = update_send_suppress_output;
3306 update->SurfaceFrameAcknowledge = update_send_frame_acknowledge;
3309int update_process_messages(rdpUpdate* update)
3311 return update_message_queue_process_pending_messages(update);
3314static void update_free_queued_message(
void* obj)
3316 wMessage* msg = (wMessage*)obj;
3317 update_message_queue_free_message(msg);
3325 free(window_state->OverlayDescription.string);
3326 free(window_state->titleInfo.string);
3327 free(window_state->windowRects);
3328 free(window_state->visibilityRects);
3332rdpUpdate* update_new(rdpRdp* rdp)
3334 const wObject cb = {
nullptr,
nullptr,
nullptr, update_free_queued_message,
nullptr };
3337 WINPR_ASSERT(rdp->context);
3344 update->common.context = rdp->context;
3345 update->log = WLog_Get(
"com.freerdp.core.update");
3346 InitializeCriticalSection(&(update->mux));
3347 update->common.pointer = (rdpPointerUpdate*)calloc(1,
sizeof(rdpPointerUpdate));
3349 if (!update->common.pointer)
3358 update->common.primary = &primary->common;
3367 update->common.secondary = &secondary->common;
3377 update->common.altsec = &altsec->common;
3379 update->common.window = (rdpWindowUpdate*)calloc(1,
sizeof(rdpWindowUpdate));
3381 if (!update->common.window)
3386 deleteList->sIndices = 64;
3387 deleteList->indices = calloc(deleteList->sIndices, 2);
3389 if (!deleteList->indices)
3392 deleteList->cIndices = 0;
3396 update->common.SuppressOutput = update_send_suppress_output;
3397 update->initialState = TRUE;
3398 update->common.autoCalculateBitmapData = TRUE;
3399 update->queue = MessageQueue_New(&cb);
3404 return &update->common;
3406 WINPR_PRAGMA_DIAG_PUSH
3407 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
3408 update_free(&update->common);
3409 WINPR_PRAGMA_DIAG_POP
3413void update_free(rdpUpdate* update)
3415 if (update !=
nullptr)
3422 free(deleteList->indices);
3424 free(update->pointer);
3426 if (update->primary)
3430 free(primary->polygon_cb.points);
3431 free(primary->polyline.points);
3432 free(primary->polygon_sc.points);
3433 free(primary->fast_glyph.glyphData.aj);
3437 free(update->secondary);
3441 free(update->window);
3443 MessageQueue_Free(up->queue);
3444 DeleteCriticalSection(&up->mux);
3447 Stream_Free(up->us, TRUE);
3452void rdp_update_lock(rdpUpdate* update)
3455 EnterCriticalSection(&up->mux);
3458void rdp_update_unlock(rdpUpdate* update)
3461 LeaveCriticalSection(&up->mux);
3464BOOL update_begin_paint(rdpUpdate* update)
3467 WINPR_ASSERT(update);
3468 rdp_update_lock(update);
3470 up->withinBeginEndPaint = TRUE;
3472 WINPR_ASSERT(update->context);
3474 BOOL rc = IFCALLRESULT(TRUE, update->BeginPaint, update->context);
3476 WLog_WARN(TAG,
"BeginPaint call failed");
3479 rdpGdi* gdi = update->context->gdi;
3483 if (gdi->hdc && gdi->primary && gdi->primary->hdc)
3485 HGDI_WND hwnd = gdi->primary->hdc->hwnd;
3487 WINPR_ASSERT(hwnd->invalid);
3489 hwnd->invalid->null = TRUE;
3496BOOL update_end_paint(rdpUpdate* update)
3500 WINPR_ASSERT(update);
3501 IFCALLRET(update->EndPaint, rc, update->context);
3503 WLog_WARN(TAG,
"EndPaint call failed");
3507 if (!up->withinBeginEndPaint)
3509 up->withinBeginEndPaint = FALSE;
3511 rdp_update_unlock(update);
This struct contains function pointer to initialize/free objects.