22#include <freerdp/config.h>
26#include <winpr/wtypes.h>
28#include <winpr/assert.h>
29#include <winpr/cast.h>
31#include <freerdp/api.h>
32#include <freerdp/log.h>
33#include <freerdp/graphics.h>
34#include <freerdp/codec/bitmap.h>
35#include <freerdp/gdi/gdi.h>
40#include "../cache/glyph.h"
41#include "../cache/bitmap.h"
42#include "../cache/brush.h"
43#include "../cache/cache.h"
45#define TAG FREERDP_TAG("core.orders")
49#define get_checked_uint16(value) get_checked_uint16_int((value), __FILE__, __func__, __LINE__)
50static inline UINT16 get_checked_uint16_int(UINT32 value, WINPR_ATTR_UNUSED
const char* file,
51 WINPR_ATTR_UNUSED
const char* fkt,
52 WINPR_ATTR_UNUSED
size_t line)
54 WINPR_ASSERT_AT(value <= UINT16_MAX, file, fkt, line);
58#define get_checked_uint8(value) get_checked_uint8_int((value), __FILE__, __func__, __LINE__)
59static inline UINT8 get_checked_uint8_int(UINT32 value, WINPR_ATTR_UNUSED
const char* file,
60 WINPR_ATTR_UNUSED
const char* fkt,
61 WINPR_ATTR_UNUSED
size_t line)
63 WINPR_ASSERT_AT(value <= UINT8_MAX, file, fkt, line);
67#define get_checked_int16(value) get_checked_int16_int((value), __FILE__, __func__, __LINE__)
68static inline INT16 get_checked_int16_int(INT32 value, WINPR_ATTR_UNUSED
const char* file,
69 WINPR_ATTR_UNUSED
const char* fkt,
70 WINPR_ATTR_UNUSED
size_t line)
72 WINPR_ASSERT_AT(value <= INT16_MAX, file, fkt, line);
73 WINPR_ASSERT_AT(value >= INT16_MIN, file, fkt, line);
77#define check_val_fits_int16(value) check_val_fits_int16_int((value), __FILE__, __func__, __LINE__)
78static inline BOOL check_val_fits_int16_int(INT32 value, WINPR_ATTR_UNUSED
const char* file,
79 WINPR_ATTR_UNUSED
const char* fkt,
80 WINPR_ATTR_UNUSED
size_t line)
82 const DWORD level = WLOG_WARN;
83 static wLog* log =
nullptr;
87 if (value < INT16_MIN)
89 if (WLog_IsLevelActive(log, level))
90 WLog_PrintTextMessage(log, level, line, file, fkt,
"value %" PRId32
" < %d", INT16_MIN,
95 if (value > INT16_MAX)
97 if (WLog_IsLevelActive(log, level))
98 WLog_PrintTextMessage(log, level, line, file, fkt,
"value %" PRId32
" > %d", INT16_MAX,
106#define gdi_rob3_code_string_checked(value) \
107 gdi_rob3_code_string_checked_int((value), __FILE__, __func__, __LINE__)
108static inline const char* gdi_rob3_code_string_checked_int(UINT32 rob,
109 WINPR_ATTR_UNUSED
const char* file,
110 WINPR_ATTR_UNUSED
const char* fkt,
111 WINPR_ATTR_UNUSED
size_t line)
113 WINPR_ASSERT_AT((rob) <= UINT8_MAX, file, fkt, line);
114 return gdi_rop3_code_string((BYTE)rob);
117#define gdi_rop3_code_checked(value) \
118 gdi_rop3_code_checked_int((value), __FILE__, __func__, __LINE__)
119static inline DWORD gdi_rop3_code_checked_int(UINT32 code, WINPR_ATTR_UNUSED
const char* file,
120 WINPR_ATTR_UNUSED
const char* fkt,
121 WINPR_ATTR_UNUSED
size_t line)
123 WINPR_ASSERT_AT(code <= UINT8_MAX, file, fkt, line);
124 return gdi_rop3_code((UINT8)code);
127static const char primary_order_str[] =
"Primary Drawing Order";
128static const char secondary_order_str[] =
"Secondary Drawing Order";
129static const char alt_sec_order_str[] =
"Alternate Secondary Drawing Order";
131BYTE get_primary_drawing_order_field_bytes(UINT32 orderType, BOOL* pValid)
138 return DSTBLT_ORDER_FIELD_BYTES;
140 return PATBLT_ORDER_FIELD_BYTES;
142 return SCRBLT_ORDER_FIELD_BYTES;
152 return DRAW_NINE_GRID_ORDER_FIELD_BYTES;
154 return MULTI_DRAW_NINE_GRID_ORDER_FIELD_BYTES;
156 return LINE_TO_ORDER_FIELD_BYTES;
158 return OPAQUE_RECT_ORDER_FIELD_BYTES;
160 return SAVE_BITMAP_ORDER_FIELD_BYTES;
164 return MEMBLT_ORDER_FIELD_BYTES;
166 return MEM3BLT_ORDER_FIELD_BYTES;
168 return MULTI_DSTBLT_ORDER_FIELD_BYTES;
170 return MULTI_PATBLT_ORDER_FIELD_BYTES;
172 return MULTI_SCRBLT_ORDER_FIELD_BYTES;
174 return MULTI_OPAQUE_RECT_ORDER_FIELD_BYTES;
176 return FAST_INDEX_ORDER_FIELD_BYTES;
178 return POLYGON_SC_ORDER_FIELD_BYTES;
180 return POLYGON_CB_ORDER_FIELD_BYTES;
182 return POLYLINE_ORDER_FIELD_BYTES;
186 return FAST_GLYPH_ORDER_FIELD_BYTES;
188 return ELLIPSE_SC_ORDER_FIELD_BYTES;
190 return ELLIPSE_CB_ORDER_FIELD_BYTES;
192 return GLYPH_INDEX_ORDER_FIELD_BYTES;
196 WLog_WARN(TAG,
"Invalid orderType 0x%08X received", orderType);
201static BYTE get_cbr2_bpp(UINT32 bpp, BOOL* pValid)
216 WLog_WARN(TAG,
"Invalid bpp %" PRIu32, bpp);
223static BYTE get_bmf_bpp(UINT32 bmf, BOOL* pValid)
228 switch (bmf & (uint32_t)(~CACHED_BRUSH))
241 WLog_WARN(TAG,
"Invalid bmf %" PRIu32, bmf);
247static BYTE get_bpp_bmf(UINT32 bpp, BOOL* pValid)
264 WLog_WARN(TAG,
"Invalid color depth %" PRIu32, bpp);
271static BOOL check_order_activated(wLog* log,
const rdpSettings* settings,
const char* orderName,
272 BOOL condition,
const char* extendedMessage)
276 if (settings->AllowUnanouncedOrdersFromServer)
278 WLog_Print(log, WLOG_WARN,
279 "%s - SERVER BUG: The support for this feature was not announced!",
282 WLog_Print(log, WLOG_WARN,
"%s", extendedMessage);
287 WLog_Print(log, WLOG_ERROR,
288 "%s - SERVER BUG: The support for this feature was not announced! Use "
289 "/relax-order-checks to ignore",
292 WLog_Print(log, WLOG_WARN,
"%s", extendedMessage);
300static BOOL check_alt_order_supported(wLog* log, rdpSettings* settings, BYTE orderType,
301 const char* orderName)
303 const char* extendedMessage =
nullptr;
304 BOOL condition = FALSE;
308 case ORDER_TYPE_CREATE_OFFSCREEN_BITMAP:
309 case ORDER_TYPE_SWITCH_SURFACE:
310 condition = settings->OffscreenSupportLevel != 0;
311 extendedMessage =
"Adding /cache:offscreen might mitigate";
314 case ORDER_TYPE_CREATE_NINE_GRID_BITMAP:
315 condition = settings->DrawNineGridEnabled;
318 case ORDER_TYPE_FRAME_MARKER:
319 condition = settings->FrameMarkerCommandEnabled;
322 case ORDER_TYPE_GDIPLUS_FIRST:
323 case ORDER_TYPE_GDIPLUS_NEXT:
324 case ORDER_TYPE_GDIPLUS_END:
325 case ORDER_TYPE_GDIPLUS_CACHE_FIRST:
326 case ORDER_TYPE_GDIPLUS_CACHE_NEXT:
327 case ORDER_TYPE_GDIPLUS_CACHE_END:
328 condition = settings->DrawGdiPlusCacheEnabled;
331 case ORDER_TYPE_WINDOW:
332 condition = settings->RemoteWndSupportLevel != WINDOW_LEVEL_NOT_SUPPORTED;
335 case ORDER_TYPE_STREAM_BITMAP_FIRST:
336 case ORDER_TYPE_STREAM_BITMAP_NEXT:
337 case ORDER_TYPE_COMPDESK_FIRST:
342 WLog_Print(log, WLOG_WARN,
"%s - %s UNKNOWN", orderName, alt_sec_order_str);
347 return check_order_activated(log, settings, orderName, condition, extendedMessage);
350static BOOL check_secondary_order_supported(wLog* log, rdpSettings* settings, BYTE orderType,
351 const char* orderName)
353 const char* extendedMessage =
nullptr;
354 BOOL condition = FALSE;
358 case ORDER_TYPE_BITMAP_UNCOMPRESSED:
359 case ORDER_TYPE_CACHE_BITMAP_COMPRESSED:
360 condition = settings->BitmapCacheEnabled;
361 extendedMessage =
"Adding /cache:bitmap might mitigate";
364 case ORDER_TYPE_BITMAP_UNCOMPRESSED_V2:
365 case ORDER_TYPE_BITMAP_COMPRESSED_V2:
366 condition = settings->BitmapCacheEnabled;
367 extendedMessage =
"Adding /cache:bitmap might mitigate";
370 case ORDER_TYPE_BITMAP_COMPRESSED_V3:
371 condition = settings->BitmapCacheV3Enabled;
372 extendedMessage =
"Adding /cache:bitmap might mitigate";
375 case ORDER_TYPE_CACHE_COLOR_TABLE:
376 condition = (settings->OrderSupport[NEG_MEMBLT_INDEX] ||
377 settings->OrderSupport[NEG_MEM3BLT_INDEX]);
380 case ORDER_TYPE_CACHE_GLYPH:
382 switch (settings->GlyphSupportLevel)
384 case GLYPH_SUPPORT_PARTIAL:
385 case GLYPH_SUPPORT_FULL:
386 case GLYPH_SUPPORT_ENCODE:
390 case GLYPH_SUPPORT_NONE:
398 case ORDER_TYPE_CACHE_BRUSH:
403 WLog_Print(log, WLOG_WARN,
"SECONDARY ORDER %s not supported", orderName);
407 return check_order_activated(log, settings, orderName, condition, extendedMessage);
410static BOOL check_primary_order_supported(wLog* log,
const rdpSettings* settings, UINT32 orderType,
411 const char* orderName)
413 const char* extendedMessage =
nullptr;
414 BOOL condition = FALSE;
418 case ORDER_TYPE_DSTBLT:
419 condition = settings->OrderSupport[NEG_DSTBLT_INDEX];
422 case ORDER_TYPE_SCRBLT:
423 condition = settings->OrderSupport[NEG_SCRBLT_INDEX];
426 case ORDER_TYPE_DRAW_NINE_GRID:
427 condition = settings->OrderSupport[NEG_DRAWNINEGRID_INDEX];
430 case ORDER_TYPE_MULTI_DRAW_NINE_GRID:
431 condition = settings->OrderSupport[NEG_MULTI_DRAWNINEGRID_INDEX];
434 case ORDER_TYPE_LINE_TO:
435 condition = settings->OrderSupport[NEG_LINETO_INDEX];
440 case ORDER_TYPE_PATBLT:
441 case ORDER_TYPE_OPAQUE_RECT:
442 condition = settings->OrderSupport[NEG_OPAQUE_RECT_INDEX] ||
443 settings->OrderSupport[NEG_PATBLT_INDEX];
446 case ORDER_TYPE_SAVE_BITMAP:
447 condition = settings->OrderSupport[NEG_SAVEBITMAP_INDEX];
450 case ORDER_TYPE_MEMBLT:
451 condition = settings->OrderSupport[NEG_MEMBLT_INDEX];
454 case ORDER_TYPE_MEM3BLT:
455 condition = settings->OrderSupport[NEG_MEM3BLT_INDEX];
458 case ORDER_TYPE_MULTI_DSTBLT:
459 condition = settings->OrderSupport[NEG_MULTIDSTBLT_INDEX];
462 case ORDER_TYPE_MULTI_PATBLT:
463 condition = settings->OrderSupport[NEG_MULTIPATBLT_INDEX];
466 case ORDER_TYPE_MULTI_SCRBLT:
467 condition = settings->OrderSupport[NEG_MULTIDSTBLT_INDEX];
470 case ORDER_TYPE_MULTI_OPAQUE_RECT:
471 condition = settings->OrderSupport[NEG_MULTIOPAQUERECT_INDEX];
474 case ORDER_TYPE_FAST_INDEX:
475 condition = settings->OrderSupport[NEG_FAST_INDEX_INDEX];
478 case ORDER_TYPE_POLYGON_SC:
479 condition = settings->OrderSupport[NEG_POLYGON_SC_INDEX];
482 case ORDER_TYPE_POLYGON_CB:
483 condition = settings->OrderSupport[NEG_POLYGON_CB_INDEX];
486 case ORDER_TYPE_POLYLINE:
487 condition = settings->OrderSupport[NEG_POLYLINE_INDEX];
490 case ORDER_TYPE_FAST_GLYPH:
491 condition = settings->OrderSupport[NEG_FAST_GLYPH_INDEX];
494 case ORDER_TYPE_ELLIPSE_SC:
495 condition = settings->OrderSupport[NEG_ELLIPSE_SC_INDEX];
498 case ORDER_TYPE_ELLIPSE_CB:
499 condition = settings->OrderSupport[NEG_ELLIPSE_CB_INDEX];
502 case ORDER_TYPE_GLYPH_INDEX:
503 condition = settings->OrderSupport[NEG_GLYPH_INDEX_INDEX];
507 WLog_Print(log, WLOG_ERROR,
"%s %s not supported", orderName, primary_order_str);
511 return check_order_activated(log, settings, orderName, condition, extendedMessage);
514WINPR_PRAGMA_DIAG_PUSH
515WINPR_PRAGMA_DIAG_IGNORED_FORMAT_NONLITERAL
516static const char* primary_order_string(UINT32 orderType)
518 const char* orders[] = {
"[0x%02" PRIx8
"] DstBlt",
519 "[0x%02" PRIx8
"] PatBlt",
520 "[0x%02" PRIx8
"] ScrBlt",
521 "[0x%02" PRIx8
"] UNUSED",
522 "[0x%02" PRIx8
"] UNUSED",
523 "[0x%02" PRIx8
"] UNUSED",
524 "[0x%02" PRIx8
"] UNUSED",
525 "[0x%02" PRIx8
"] DrawNineGrid",
526 "[0x%02" PRIx8
"] MultiDrawNineGrid",
527 "[0x%02" PRIx8
"] LineTo",
528 "[0x%02" PRIx8
"] OpaqueRect",
529 "[0x%02" PRIx8
"] SaveBitmap",
530 "[0x%02" PRIx8
"] UNUSED",
531 "[0x%02" PRIx8
"] MemBlt",
532 "[0x%02" PRIx8
"] Mem3Blt",
533 "[0x%02" PRIx8
"] MultiDstBlt",
534 "[0x%02" PRIx8
"] MultiPatBlt",
535 "[0x%02" PRIx8
"] MultiScrBlt",
536 "[0x%02" PRIx8
"] MultiOpaqueRect",
537 "[0x%02" PRIx8
"] FastIndex",
538 "[0x%02" PRIx8
"] PolygonSC",
539 "[0x%02" PRIx8
"] PolygonCB",
540 "[0x%02" PRIx8
"] Polyline",
541 "[0x%02" PRIx8
"] UNUSED",
542 "[0x%02" PRIx8
"] FastGlyph",
543 "[0x%02" PRIx8
"] EllipseSC",
544 "[0x%02" PRIx8
"] EllipseCB",
545 "[0x%02" PRIx8
"] GlyphIndex" };
546 const char* fmt =
"[0x%02" PRIx8
"] UNKNOWN";
547 static char buffer[64] = WINPR_C_ARRAY_INIT;
549 if (orderType < ARRAYSIZE(orders))
550 fmt = orders[orderType];
552 (void)sprintf_s(buffer, ARRAYSIZE(buffer), fmt, orderType);
555static const char* secondary_order_string(UINT32 orderType)
557 const char* orders[] = {
"[0x%02" PRIx8
"] Cache Bitmap",
558 "[0x%02" PRIx8
"] Cache Color Table",
559 "[0x%02" PRIx8
"] Cache Bitmap (Compressed)",
560 "[0x%02" PRIx8
"] Cache Glyph",
561 "[0x%02" PRIx8
"] Cache Bitmap V2",
562 "[0x%02" PRIx8
"] Cache Bitmap V2 (Compressed)",
563 "[0x%02" PRIx8
"] UNUSED",
564 "[0x%02" PRIx8
"] Cache Brush",
565 "[0x%02" PRIx8
"] Cache Bitmap V3" };
566 const char* fmt =
"[0x%02" PRIx8
"] UNKNOWN";
567 static char buffer[64] = WINPR_C_ARRAY_INIT;
569 if (orderType < ARRAYSIZE(orders))
570 fmt = orders[orderType];
572 (void)sprintf_s(buffer, ARRAYSIZE(buffer), fmt, orderType);
575static const char* altsec_order_string(BYTE orderType)
577 const char* orders[] = {
578 "[0x%02" PRIx8
"] Switch Surface",
"[0x%02" PRIx8
"] Create Offscreen Bitmap",
579 "[0x%02" PRIx8
"] Stream Bitmap First",
"[0x%02" PRIx8
"] Stream Bitmap Next",
580 "[0x%02" PRIx8
"] Create NineGrid Bitmap",
"[0x%02" PRIx8
"] Draw GDI+ First",
581 "[0x%02" PRIx8
"] Draw GDI+ Next",
"[0x%02" PRIx8
"] Draw GDI+ End",
582 "[0x%02" PRIx8
"] Draw GDI+ Cache First",
"[0x%02" PRIx8
"] Draw GDI+ Cache Next",
583 "[0x%02" PRIx8
"] Draw GDI+ Cache End",
"[0x%02" PRIx8
"] Windowing",
584 "[0x%02" PRIx8
"] Desktop Composition",
"[0x%02" PRIx8
"] Frame Marker"
586 const char* fmt =
"[0x%02" PRIx8
"] UNKNOWN";
587 static char buffer[64] = WINPR_C_ARRAY_INIT;
589 if (orderType < ARRAYSIZE(orders))
590 fmt = orders[orderType];
592 (void)sprintf_s(buffer, ARRAYSIZE(buffer), fmt, orderType);
597static inline BOOL update_read_coord(
wStream* s, INT32* coord, BOOL delta)
604 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
607 Stream_Read_INT8(s, lsi8);
612 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
615 Stream_Read_INT16(s, lsi16);
622#define update_write_coord(s, coord) \
623 update_write_coord_int((s), (coord), #coord, __FILE__, __func__, __LINE__)
625static inline BOOL update_write_coord_int(
wStream* s, INT32 coord,
const char* name,
626 const char* file,
const char* fkt,
size_t line)
628 if ((coord < 0) || (coord > UINT16_MAX))
630 const DWORD level = WLOG_WARN;
631 wLog* log = WLog_Get(TAG);
632 if (WLog_IsLevelActive(log, level))
634 WLog_PrintTextMessage(log, level, line, file, fkt,
"[%s] 0 <= %" PRId32
" <= %d", name,
640 Stream_Write_UINT16(s, (UINT16)coord);
643static inline BOOL update_read_color(
wStream* s, UINT32* color)
647 if (!Stream_CheckAndLogRequiredLength(TAG, s, 3))
651 Stream_Read_UINT8(s,
byte);
652 *color = (UINT32)
byte;
653 Stream_Read_UINT8(s,
byte);
654 *color |= ((UINT32)
byte << 8) & 0xFF00;
655 Stream_Read_UINT8(s,
byte);
656 *color |= ((UINT32)
byte << 16) & 0xFF0000;
659static inline BOOL update_write_color(
wStream* s, UINT32 color)
662 byte = (color & 0xFF);
663 Stream_Write_UINT8(s,
byte);
664 byte = ((color >> 8) & 0xFF);
665 Stream_Write_UINT8(s,
byte);
666 byte = ((color >> 16) & 0xFF);
667 Stream_Write_UINT8(s,
byte);
670static inline BOOL update_read_colorref(
wStream* s, UINT32* color)
674 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
678 Stream_Read_UINT8(s,
byte);
680 Stream_Read_UINT8(s,
byte);
681 *color |= ((UINT32)
byte << 8);
682 Stream_Read_UINT8(s,
byte);
683 *color |= ((UINT32)
byte << 16);
684 Stream_Seek_UINT8(s);
687static inline BOOL update_read_color_quad(
wStream* s, UINT32* color)
689 return update_read_colorref(s, color);
691static inline void update_write_color_quad(
wStream* s, UINT32 color)
694 byte = (color >> 16) & 0xFF;
695 Stream_Write_UINT8(s,
byte);
696 byte = (color >> 8) & 0xFF;
697 Stream_Write_UINT8(s,
byte);
699 Stream_Write_UINT8(s,
byte);
701static inline BOOL update_read_2byte_unsigned(
wStream* s, UINT32* value)
705 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
708 Stream_Read_UINT8(s,
byte);
712 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
715 *value = ((
byte & 0x7F) << 8) & 0xFFFF;
716 Stream_Read_UINT8(s,
byte);
721 *value = (
byte & 0x7F);
726static inline BOOL update_write_2byte_unsigned(
wStream* s, UINT32 value)
735 byte = ((value & 0x7F00) >> 8);
736 Stream_Write_UINT8(s,
byte | 0x80);
737 byte = (value & 0xFF);
738 Stream_Write_UINT8(s,
byte);
742 byte = (value & 0x7F);
743 Stream_Write_UINT8(s,
byte);
748static inline BOOL update_read_2byte_signed(
wStream* s, INT32* value)
753 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
756 Stream_Read_UINT8(s,
byte);
757 negative = (
byte & 0x40) != 0;
758 *value = (
byte & 0x3F);
762 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
765 Stream_Read_UINT8(s,
byte);
766 *value = (*value << 8) |
byte;
774static inline BOOL update_write_2byte_signed(
wStream* s, INT32 value)
777 BOOL negative = FALSE;
790 byte = ((value & 0x3F00) >> 8);
795 Stream_Write_UINT8(s,
byte | 0x80);
796 byte = (value & 0xFF);
797 Stream_Write_UINT8(s,
byte);
801 byte = (value & 0x3F);
806 Stream_Write_UINT8(s,
byte);
811static inline BOOL update_read_4byte_unsigned(
wStream* s, UINT32* value)
813 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
816 const UINT32
byte = Stream_Get_UINT8(s);
817 const BYTE count = WINPR_ASSERTING_INT_CAST(uint8_t, (
byte & 0xC0) >> 6);
819 if (!Stream_CheckAndLogRequiredLength(TAG, s, count))
825 *value = (
byte & 0x3F);
829 *value = ((
byte & 0x3F) << 8) & 0xFFFF;
830 *value |= Stream_Get_UINT8(s);
834 *value = ((
byte & 0x3F) << 16) & 0xFFFFFF;
835 *value |= ((Stream_Get_UINT8(s) << 8)) & 0xFFFF;
836 *value |= Stream_Get_UINT8(s);
840 *value = ((
byte & 0x3F) << 24) & 0xFF000000;
841 *value |= ((Stream_Get_UINT8(s) << 16)) & 0xFF0000;
842 *value |= ((Stream_Get_UINT8(s) << 8)) & 0xFF00;
843 *value |= Stream_Get_UINT8(s);
852static inline BOOL update_write_4byte_unsigned(
wStream* s, UINT32 value)
858 Stream_Write_UINT8(s, (UINT8)value);
860 else if (value <= 0x3FFF)
862 byte = (value >> 8) & 0x3F;
863 Stream_Write_UINT8(s,
byte | 0x40);
864 byte = (value & 0xFF);
865 Stream_Write_UINT8(s,
byte);
867 else if (value <= 0x3FFFFF)
869 byte = (value >> 16) & 0x3F;
870 Stream_Write_UINT8(s,
byte | 0x80);
871 byte = (value >> 8) & 0xFF;
872 Stream_Write_UINT8(s,
byte);
873 byte = (value & 0xFF);
874 Stream_Write_UINT8(s,
byte);
876 else if (value <= 0x3FFFFFFF)
878 byte = (value >> 24) & 0x3F;
879 Stream_Write_UINT8(s,
byte | 0xC0);
880 byte = (value >> 16) & 0xFF;
881 Stream_Write_UINT8(s,
byte);
882 byte = (value >> 8) & 0xFF;
883 Stream_Write_UINT8(s,
byte);
884 byte = (value & 0xFF);
885 Stream_Write_UINT8(s,
byte);
893static inline BOOL update_read_delta(
wStream* s, INT32* value)
898 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
901 Stream_Read_UINT8(s,
byte);
904 uvalue = WINPR_CXX_COMPAT_CAST(UINT32, (
byte | ~0x3F));
906 uvalue = (
byte & 0x3F);
910 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
913 Stream_Read_UINT8(s,
byte);
914 uvalue = (uvalue << 8) |
byte;
916 *value = (INT32)uvalue;
921static inline BOOL update_read_brush(
wStream* s, rdpBrush* brush, BYTE fieldFlags)
923 if (fieldFlags & ORDER_FIELD_01)
925 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
928 Stream_Read_UINT8(s, brush->x);
931 if (fieldFlags & ORDER_FIELD_02)
933 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
936 Stream_Read_UINT8(s, brush->y);
939 if (fieldFlags & ORDER_FIELD_03)
941 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
944 Stream_Read_UINT8(s, brush->style);
947 if (fieldFlags & ORDER_FIELD_04)
949 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
952 Stream_Read_UINT8(s, brush->hatch);
955 if (brush->style & CACHED_BRUSH)
958 brush->index = brush->hatch;
959 brush->bpp = get_bmf_bpp(brush->style, &rc);
966 if (fieldFlags & ORDER_FIELD_05)
968 if (!Stream_CheckAndLogRequiredLength(TAG, s, 7))
971 brush->data = (BYTE*)brush->p8x8;
972 Stream_Read_UINT8(s, brush->data[7]);
973 Stream_Read_UINT8(s, brush->data[6]);
974 Stream_Read_UINT8(s, brush->data[5]);
975 Stream_Read_UINT8(s, brush->data[4]);
976 Stream_Read_UINT8(s, brush->data[3]);
977 Stream_Read_UINT8(s, brush->data[2]);
978 Stream_Read_UINT8(s, brush->data[1]);
979 brush->data[0] = get_checked_uint8(brush->hatch);
984static inline BOOL update_write_brush(
wStream* s, rdpBrush* brush, BYTE fieldFlags)
986 if (fieldFlags & ORDER_FIELD_01)
988 if (!Stream_EnsureRemainingCapacity(s, 1))
990 Stream_Write_UINT8(s, get_checked_uint8(brush->x));
993 if (fieldFlags & ORDER_FIELD_02)
995 if (!Stream_EnsureRemainingCapacity(s, 1))
997 Stream_Write_UINT8(s, get_checked_uint8(brush->y));
1000 if (fieldFlags & ORDER_FIELD_03)
1002 if (!Stream_EnsureRemainingCapacity(s, 1))
1004 Stream_Write_UINT8(s, get_checked_uint8(brush->style));
1007 if (brush->style & CACHED_BRUSH)
1010 brush->hatch = brush->index;
1011 brush->bpp = get_bmf_bpp(brush->style, &rc);
1014 if (brush->bpp == 0)
1018 if (fieldFlags & ORDER_FIELD_04)
1020 if (!Stream_EnsureRemainingCapacity(s, 1))
1022 Stream_Write_UINT8(s, get_checked_uint8(brush->hatch));
1025 if (fieldFlags & ORDER_FIELD_05)
1027 brush->data = (BYTE*)brush->p8x8;
1028 if (!Stream_EnsureRemainingCapacity(s, 7))
1030 Stream_Write_UINT8(s, brush->data[7]);
1031 Stream_Write_UINT8(s, brush->data[6]);
1032 Stream_Write_UINT8(s, brush->data[5]);
1033 Stream_Write_UINT8(s, brush->data[4]);
1034 Stream_Write_UINT8(s, brush->data[3]);
1035 Stream_Write_UINT8(s, brush->data[2]);
1036 Stream_Write_UINT8(s, brush->data[1]);
1037 brush->data[0] = get_checked_uint8(brush->hatch);
1042static inline BOOL update_read_delta_rects(
wStream* s,
DELTA_RECT* rectangles,
const UINT32* nr)
1044 UINT32 number = *nr;
1046 BYTE* zeroBits =
nullptr;
1047 UINT32 zeroBitsSize = 0;
1051 WLog_WARN(TAG,
"Invalid number of delta rectangles %" PRIu32, number);
1055 zeroBitsSize = ((number + 1) / 2);
1057 if (!Stream_CheckAndLogRequiredLength(TAG, s, zeroBitsSize))
1060 Stream_GetPointer(s, zeroBits);
1061 Stream_Seek(s, zeroBitsSize);
1062 ZeroMemory(rectangles,
sizeof(
DELTA_RECT) * number);
1064 for (UINT32 i = 0; i < number; i++)
1067 flags = zeroBits[i / 2];
1069 if ((~flags & 0x80) && !update_read_delta(s, &rectangles[i].left))
1072 if ((~flags & 0x40) && !update_read_delta(s, &rectangles[i].top))
1077 if (!update_read_delta(s, &rectangles[i].width))
1081 rectangles[i].width = rectangles[i - 1].width;
1083 rectangles[i].width = 0;
1087 if (!update_read_delta(s, &rectangles[i].height))
1091 rectangles[i].height = rectangles[i - 1].height;
1093 rectangles[i].height = 0;
1097 rectangles[i].left += rectangles[i - 1].left;
1098 rectangles[i].top += rectangles[i - 1].top;
1107static inline BOOL update_read_delta_points(
wStream* s,
DELTA_POINT** points, UINT32 number,
1108 WINPR_ATTR_UNUSED INT16 x, WINPR_ATTR_UNUSED INT16 y)
1111 BYTE* zeroBits =
nullptr;
1112 UINT32 zeroBitsSize = ((number + 3) / 4);
1114 WINPR_ASSERT(points);
1119 *points = newpoints;
1121 if (!Stream_CheckAndLogRequiredLength(TAG, s, zeroBitsSize))
1124 Stream_GetPointer(s, zeroBits);
1125 Stream_Seek(s, zeroBitsSize);
1126 ZeroMemory(*points,
sizeof(
DELTA_POINT) * number);
1128 for (UINT32 i = 0; i < number; i++)
1131 flags = zeroBits[i / 4];
1133 if ((~flags & 0x80) && !update_read_delta(s, &newpoints[i].x))
1135 WLog_ERR(TAG,
"update_read_delta(x) failed");
1139 if ((~flags & 0x40) && !update_read_delta(s, &newpoints[i].y))
1141 WLog_ERR(TAG,
"update_read_delta(y) failed");
1151static BOOL order_field_flag_is_set(
const ORDER_INFO* orderInfo, BYTE number)
1153 const UINT32 mask = (UINT32)(1UL << ((UINT32)number - 1UL));
1154 const BOOL set = (orderInfo->fieldFlags & mask) != 0;
1158static inline BOOL read_order_field_byte(
const char* orderName,
const ORDER_INFO* orderInfo,
1159 wStream* s, BYTE number, UINT32* target, BOOL optional)
1161 WINPR_ASSERT(orderName);
1162 WINPR_ASSERT(orderInfo);
1163 WINPR_ASSERT(target);
1165 if (!order_field_flag_is_set(orderInfo, number))
1167 WLog_DBG(TAG,
"order %s field %" PRIu8
" not found [optional:%d]", orderName, number,
1171 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1173 Stream_Read_UINT8(s, *target);
1177static inline BOOL read_order_field_2bytes(
const char* orderName,
const ORDER_INFO* orderInfo,
1178 wStream* s, BYTE number, UINT32* target1,
1179 UINT32* target2, BOOL optional)
1181 WINPR_ASSERT(orderName);
1182 WINPR_ASSERT(orderInfo);
1183 WINPR_ASSERT(target1);
1184 WINPR_ASSERT(target2);
1186 if (!order_field_flag_is_set(orderInfo, number))
1188 WLog_DBG(TAG,
"order %s field %" PRIu8
" not found [optional:%d]", orderName, number,
1192 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1194 Stream_Read_UINT8(s, *target1);
1195 Stream_Read_UINT8(s, *target2);
1199static inline BOOL read_order_field_uint16(
const char* orderName,
const ORDER_INFO* orderInfo,
1200 wStream* s, BYTE number, UINT32* target, BOOL optional)
1202 WINPR_ASSERT(orderName);
1203 WINPR_ASSERT(orderInfo);
1204 WINPR_ASSERT(target);
1206 if (!order_field_flag_is_set(orderInfo, number))
1208 WLog_DBG(TAG,
"order %s field %" PRIu8
" not found [optional:%d]", orderName, number,
1213 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1216 Stream_Read_UINT16(s, *target);
1220static inline BOOL read_order_field_int16(
const char* orderName,
const ORDER_INFO* orderInfo,
1221 wStream* s, BYTE number, INT32* target, BOOL optional)
1223 WINPR_ASSERT(orderName);
1224 WINPR_ASSERT(orderInfo);
1225 WINPR_ASSERT(target);
1227 if (!order_field_flag_is_set(orderInfo, number))
1229 WLog_DBG(TAG,
"order %s field %" PRIu8
" not found [optional:%d]", orderName, number,
1234 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1237 Stream_Read_INT16(s, *target);
1241static inline BOOL read_order_field_uint32(
const char* orderName,
const ORDER_INFO* orderInfo,
1242 wStream* s, BYTE number, UINT32* target, BOOL optional)
1244 WINPR_ASSERT(orderName);
1245 WINPR_ASSERT(orderInfo);
1246 WINPR_ASSERT(target);
1248 if (!order_field_flag_is_set(orderInfo, number))
1250 WLog_DBG(TAG,
"order %s field %" PRIu8
" not found [optional:%d]", orderName, number,
1255 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
1258 Stream_Read_UINT32(s, *target);
1262static inline BOOL read_order_field_coord(
const char* orderName,
const ORDER_INFO* orderInfo,
1263 wStream* s, UINT32 NO, INT32* TARGET, BOOL optional)
1265 WINPR_ASSERT(orderName);
1266 WINPR_ASSERT(orderInfo);
1267 WINPR_ASSERT(TARGET);
1269 if (!order_field_flag_is_set(orderInfo, get_checked_uint8(NO)))
1271 WLog_DBG(TAG,
"order %s field %" PRIu8
" not found [optional:%d]", orderName,
1272 get_checked_uint8(NO), optional);
1276 return update_read_coord(s, TARGET, orderInfo->deltaCoordinates);
1279static inline BOOL read_order_field_color(
const char* orderName,
const ORDER_INFO* orderInfo,
1280 wStream* s, UINT32 NO, UINT32* TARGET, BOOL optional)
1282 WINPR_ASSERT(orderName);
1283 WINPR_ASSERT(orderInfo);
1284 WINPR_ASSERT(TARGET);
1286 if (!order_field_flag_is_set(orderInfo, get_checked_uint8(NO)))
1288 WLog_DBG(TAG,
"order %s field %" PRIu8
" not found [optional:%d]", orderName,
1289 get_checked_uint8(NO), optional);
1293 if (!update_read_color(s, TARGET))
1298static inline BOOL FIELD_SKIP_BUFFER16(
wStream* s, UINT32 TARGET_LEN)
1300 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1303 Stream_Read_UINT16(s, TARGET_LEN);
1305 if (!Stream_SafeSeek(s, TARGET_LEN))
1307 WLog_ERR(TAG,
"error skipping %" PRIu32
" bytes", TARGET_LEN);
1314static BOOL update_read_dstblt_order(
const char* orderName,
wStream* s,
const ORDER_INFO* orderInfo,
1317 return (read_order_field_coord(orderName, orderInfo, s, 1, &dstblt->nLeftRect, FALSE) &&
1318 read_order_field_coord(orderName, orderInfo, s, 2, &dstblt->nTopRect, FALSE) &&
1319 read_order_field_coord(orderName, orderInfo, s, 3, &dstblt->nWidth, FALSE) &&
1320 read_order_field_coord(orderName, orderInfo, s, 4, &dstblt->nHeight, FALSE) &&
1321 read_order_field_byte(orderName, orderInfo, s, 5, &dstblt->bRop, TRUE));
1326 WINPR_UNUSED(orderInfo);
1327 WINPR_UNUSED(dstblt);
1333 if (!Stream_EnsureRemainingCapacity(s, update_approximate_dstblt_order(orderInfo, dstblt)))
1336 orderInfo->fieldFlags = 0;
1337 orderInfo->fieldFlags |= ORDER_FIELD_01;
1338 if (!update_write_coord(s, dstblt->nLeftRect))
1340 orderInfo->fieldFlags |= ORDER_FIELD_02;
1341 if (!update_write_coord(s, dstblt->nTopRect))
1343 orderInfo->fieldFlags |= ORDER_FIELD_03;
1344 if (!update_write_coord(s, dstblt->nWidth))
1346 orderInfo->fieldFlags |= ORDER_FIELD_04;
1347 if (!update_write_coord(s, dstblt->nHeight))
1349 orderInfo->fieldFlags |= ORDER_FIELD_05;
1350 Stream_Write_UINT8(s, get_checked_uint8(dstblt->bRop));
1354static BOOL update_read_patblt_order(
const char* orderName,
wStream* s,
const ORDER_INFO* orderInfo,
1357 return (read_order_field_coord(orderName, orderInfo, s, 1, &patblt->nLeftRect, FALSE) &&
1358 read_order_field_coord(orderName, orderInfo, s, 2, &patblt->nTopRect, FALSE) &&
1359 read_order_field_coord(orderName, orderInfo, s, 3, &patblt->nWidth, FALSE) &&
1360 read_order_field_coord(orderName, orderInfo, s, 4, &patblt->nHeight, FALSE) &&
1361 read_order_field_byte(orderName, orderInfo, s, 5, &patblt->bRop, TRUE) &&
1362 read_order_field_color(orderName, orderInfo, s, 6, &patblt->backColor, TRUE) &&
1363 read_order_field_color(orderName, orderInfo, s, 7, &patblt->foreColor, TRUE) &&
1364 update_read_brush(s, &patblt->brush,
1365 get_checked_uint8((orderInfo->fieldFlags >> 7) & 0x1F)));
1370 WINPR_UNUSED(orderInfo);
1371 WINPR_UNUSED(patblt);
1377 if (!Stream_EnsureRemainingCapacity(s, update_approximate_patblt_order(orderInfo, patblt)))
1380 orderInfo->fieldFlags = 0;
1381 orderInfo->fieldFlags |= ORDER_FIELD_01;
1382 if (!update_write_coord(s, patblt->nLeftRect))
1384 orderInfo->fieldFlags |= ORDER_FIELD_02;
1385 if (!update_write_coord(s, patblt->nTopRect))
1387 orderInfo->fieldFlags |= ORDER_FIELD_03;
1388 if (!update_write_coord(s, patblt->nWidth))
1390 orderInfo->fieldFlags |= ORDER_FIELD_04;
1391 if (!update_write_coord(s, patblt->nHeight))
1393 orderInfo->fieldFlags |= ORDER_FIELD_05;
1394 Stream_Write_UINT8(s, get_checked_uint8(patblt->bRop));
1395 orderInfo->fieldFlags |= ORDER_FIELD_06;
1396 update_write_color(s, patblt->backColor);
1397 orderInfo->fieldFlags |= ORDER_FIELD_07;
1398 update_write_color(s, patblt->foreColor);
1399 orderInfo->fieldFlags |= ORDER_FIELD_08;
1400 orderInfo->fieldFlags |= ORDER_FIELD_09;
1401 orderInfo->fieldFlags |= ORDER_FIELD_10;
1402 orderInfo->fieldFlags |= ORDER_FIELD_11;
1403 orderInfo->fieldFlags |= ORDER_FIELD_12;
1404 update_write_brush(s, &patblt->brush, get_checked_uint8((orderInfo->fieldFlags >> 7) & 0x1F));
1408static BOOL update_read_scrblt_order(
const char* orderName,
wStream* s,
const ORDER_INFO* orderInfo,
1411 WINPR_ASSERT(orderInfo);
1412 WINPR_ASSERT(scrblt);
1413 return (read_order_field_coord(orderName, orderInfo, s, 1, &scrblt->nLeftRect, FALSE) &&
1414 read_order_field_coord(orderName, orderInfo, s, 2, &scrblt->nTopRect, FALSE) &&
1415 read_order_field_coord(orderName, orderInfo, s, 3, &scrblt->nWidth, FALSE) &&
1416 read_order_field_coord(orderName, orderInfo, s, 4, &scrblt->nHeight, FALSE) &&
1417 read_order_field_byte(orderName, orderInfo, s, 5, &scrblt->bRop, TRUE) &&
1418 read_order_field_coord(orderName, orderInfo, s, 6, &scrblt->nXSrc, FALSE) &&
1419 read_order_field_coord(orderName, orderInfo, s, 7, &scrblt->nYSrc, FALSE));
1424 WINPR_ASSERT(orderInfo);
1425 WINPR_ASSERT(scrblt);
1426 WINPR_UNUSED(orderInfo);
1427 WINPR_UNUSED(scrblt);
1433 WINPR_ASSERT(orderInfo);
1434 WINPR_ASSERT(scrblt);
1435 if (!Stream_EnsureRemainingCapacity(s, update_approximate_scrblt_order(orderInfo, scrblt)))
1438 orderInfo->fieldFlags = 0;
1439 orderInfo->fieldFlags |= ORDER_FIELD_01;
1440 if (!update_write_coord(s, scrblt->nLeftRect))
1442 orderInfo->fieldFlags |= ORDER_FIELD_02;
1443 if (!update_write_coord(s, scrblt->nTopRect))
1445 orderInfo->fieldFlags |= ORDER_FIELD_03;
1446 if (!update_write_coord(s, scrblt->nWidth))
1448 orderInfo->fieldFlags |= ORDER_FIELD_04;
1449 if (!update_write_coord(s, scrblt->nHeight))
1451 orderInfo->fieldFlags |= ORDER_FIELD_05;
1452 WINPR_ASSERT(scrblt->bRop <= UINT8_MAX);
1453 Stream_Write_UINT8(s, (UINT8)scrblt->bRop);
1454 orderInfo->fieldFlags |= ORDER_FIELD_06;
1455 if (!update_write_coord(s, scrblt->nXSrc))
1457 orderInfo->fieldFlags |= ORDER_FIELD_07;
1458 return (update_write_coord(s, scrblt->nYSrc));
1460static BOOL update_read_opaque_rect_order(
const char* orderName,
wStream* s,
1465 if (!read_order_field_coord(orderName, orderInfo, s, 1, &opaque_rect->nLeftRect, FALSE) ||
1466 !read_order_field_coord(orderName, orderInfo, s, 2, &opaque_rect->nTopRect, FALSE) ||
1467 !read_order_field_coord(orderName, orderInfo, s, 3, &opaque_rect->nWidth, FALSE) ||
1468 !read_order_field_coord(orderName, orderInfo, s, 4, &opaque_rect->nHeight, FALSE))
1471 if ((orderInfo->fieldFlags & ORDER_FIELD_05) != 0)
1473 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1476 Stream_Read_UINT8(s,
byte);
1477 opaque_rect->color = (opaque_rect->color & 0x00FFFF00) | ((UINT32)byte);
1480 if ((orderInfo->fieldFlags & ORDER_FIELD_06) != 0)
1482 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1485 Stream_Read_UINT8(s,
byte);
1486 opaque_rect->color = (opaque_rect->color & 0x00FF00FF) | ((UINT32)
byte << 8);
1489 if ((orderInfo->fieldFlags & ORDER_FIELD_07) != 0)
1491 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1494 Stream_Read_UINT8(s,
byte);
1495 opaque_rect->color = (opaque_rect->color & 0x0000FFFF) | ((UINT32)
byte << 16);
1501size_t update_approximate_opaque_rect_order(
ORDER_INFO* orderInfo,
1504 WINPR_UNUSED(orderInfo);
1505 WINPR_UNUSED(opaque_rect);
1513 size_t inf = update_approximate_opaque_rect_order(orderInfo, opaque_rect);
1515 if (!Stream_EnsureRemainingCapacity(s, inf))
1519 orderInfo->fieldFlags = 0;
1520 orderInfo->fieldFlags |= ORDER_FIELD_01;
1521 if (!update_write_coord(s, opaque_rect->nLeftRect))
1523 orderInfo->fieldFlags |= ORDER_FIELD_02;
1524 if (!update_write_coord(s, opaque_rect->nTopRect))
1526 orderInfo->fieldFlags |= ORDER_FIELD_03;
1527 if (!update_write_coord(s, opaque_rect->nWidth))
1529 orderInfo->fieldFlags |= ORDER_FIELD_04;
1530 if (!update_write_coord(s, opaque_rect->nHeight))
1532 orderInfo->fieldFlags |= ORDER_FIELD_05;
1533 byte = opaque_rect->color & 0x000000FF;
1534 Stream_Write_UINT8(s,
byte);
1535 orderInfo->fieldFlags |= ORDER_FIELD_06;
1536 byte = (opaque_rect->color & 0x0000FF00) >> 8;
1537 Stream_Write_UINT8(s,
byte);
1538 orderInfo->fieldFlags |= ORDER_FIELD_07;
1539 byte = (opaque_rect->color & 0x00FF0000) >> 16;
1540 Stream_Write_UINT8(s,
byte);
1544static BOOL update_read_draw_nine_grid_order(
const char* orderName,
wStream* s,
1548 return (read_order_field_coord(orderName, orderInfo, s, 1, &draw_nine_grid->srcLeft, FALSE) &&
1549 read_order_field_coord(orderName, orderInfo, s, 2, &draw_nine_grid->srcTop, FALSE) &&
1550 read_order_field_coord(orderName, orderInfo, s, 3, &draw_nine_grid->srcRight, FALSE) &&
1551 read_order_field_coord(orderName, orderInfo, s, 4, &draw_nine_grid->srcBottom, FALSE) &&
1552 read_order_field_uint16(orderName, orderInfo, s, 5, &draw_nine_grid->bitmapId, FALSE));
1555static BOOL update_read_multi_dstblt_order(
const char* orderName,
wStream* s,
1559 UINT32 numRectangles = multi_dstblt->numRectangles;
1560 if (!read_order_field_coord(orderName, orderInfo, s, 1, &multi_dstblt->nLeftRect, FALSE) ||
1561 !read_order_field_coord(orderName, orderInfo, s, 2, &multi_dstblt->nTopRect, FALSE) ||
1562 !read_order_field_coord(orderName, orderInfo, s, 3, &multi_dstblt->nWidth, FALSE) ||
1563 !read_order_field_coord(orderName, orderInfo, s, 4, &multi_dstblt->nHeight, FALSE) ||
1564 !read_order_field_byte(orderName, orderInfo, s, 5, &multi_dstblt->bRop, TRUE) ||
1565 !read_order_field_byte(orderName, orderInfo, s, 6, &numRectangles, TRUE))
1568 if ((orderInfo->fieldFlags & ORDER_FIELD_07) != 0)
1570 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1573 multi_dstblt->numRectangles = numRectangles;
1574 Stream_Read_UINT16(s, multi_dstblt->cbData);
1575 return update_read_delta_rects(s, multi_dstblt->rectangles, &multi_dstblt->numRectangles);
1577 if (numRectangles > multi_dstblt->numRectangles)
1579 WLog_ERR(TAG,
"%s numRectangles %" PRIu32
" > %" PRIu32, orderName, numRectangles,
1580 multi_dstblt->numRectangles);
1583 multi_dstblt->numRectangles = numRectangles;
1587static BOOL update_read_multi_patblt_order(
const char* orderName,
wStream* s,
1591 if (!read_order_field_coord(orderName, orderInfo, s, 1, &multi_patblt->nLeftRect, FALSE) ||
1592 !read_order_field_coord(orderName, orderInfo, s, 2, &multi_patblt->nTopRect, FALSE) ||
1593 !read_order_field_coord(orderName, orderInfo, s, 3, &multi_patblt->nWidth, FALSE) ||
1594 !read_order_field_coord(orderName, orderInfo, s, 4, &multi_patblt->nHeight, FALSE) ||
1595 !read_order_field_byte(orderName, orderInfo, s, 5, &multi_patblt->bRop, TRUE) ||
1596 !read_order_field_color(orderName, orderInfo, s, 6, &multi_patblt->backColor, TRUE) ||
1597 !read_order_field_color(orderName, orderInfo, s, 7, &multi_patblt->foreColor, TRUE))
1600 if (!update_read_brush(s, &multi_patblt->brush,
1601 get_checked_uint8((orderInfo->fieldFlags >> 7) & 0x1F)))
1604 UINT32 numRectangles = multi_patblt->numRectangles;
1605 if (!read_order_field_byte(orderName, orderInfo, s, 13, &numRectangles, TRUE))
1608 if ((orderInfo->fieldFlags & ORDER_FIELD_14) != 0)
1610 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1613 multi_patblt->numRectangles = numRectangles;
1614 Stream_Read_UINT16(s, multi_patblt->cbData);
1616 if (!update_read_delta_rects(s, multi_patblt->rectangles, &multi_patblt->numRectangles))
1620 if (numRectangles > multi_patblt->numRectangles)
1622 WLog_ERR(TAG,
"%s numRectangles %" PRIu32
" > %" PRIu32, orderName, numRectangles,
1623 multi_patblt->numRectangles);
1626 multi_patblt->numRectangles = numRectangles;
1631static BOOL update_read_multi_scrblt_order(
const char* orderName,
wStream* s,
1635 WINPR_ASSERT(orderInfo);
1636 WINPR_ASSERT(multi_scrblt);
1638 UINT32 numRectangles = multi_scrblt->numRectangles;
1639 if (!read_order_field_coord(orderName, orderInfo, s, 1, &multi_scrblt->nLeftRect, FALSE) ||
1640 !read_order_field_coord(orderName, orderInfo, s, 2, &multi_scrblt->nTopRect, FALSE) ||
1641 !read_order_field_coord(orderName, orderInfo, s, 3, &multi_scrblt->nWidth, FALSE) ||
1642 !read_order_field_coord(orderName, orderInfo, s, 4, &multi_scrblt->nHeight, FALSE) ||
1643 !read_order_field_byte(orderName, orderInfo, s, 5, &multi_scrblt->bRop, TRUE) ||
1644 !read_order_field_coord(orderName, orderInfo, s, 6, &multi_scrblt->nXSrc, FALSE) ||
1645 !read_order_field_coord(orderName, orderInfo, s, 7, &multi_scrblt->nYSrc, FALSE) ||
1646 !read_order_field_byte(orderName, orderInfo, s, 8, &numRectangles, TRUE))
1649 if ((orderInfo->fieldFlags & ORDER_FIELD_09) != 0)
1651 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1654 multi_scrblt->numRectangles = numRectangles;
1655 Stream_Read_UINT16(s, multi_scrblt->cbData);
1656 return update_read_delta_rects(s, multi_scrblt->rectangles, &multi_scrblt->numRectangles);
1659 if (numRectangles > multi_scrblt->numRectangles)
1661 WLog_ERR(TAG,
"%s numRectangles %" PRIu32
" > %" PRIu32, orderName, numRectangles,
1662 multi_scrblt->numRectangles);
1665 multi_scrblt->numRectangles = numRectangles;
1670static BOOL update_read_multi_opaque_rect_order(
const char* orderName,
wStream* s,
1675 if (!read_order_field_coord(orderName, orderInfo, s, 1, &multi_opaque_rect->nLeftRect, FALSE) ||
1676 !read_order_field_coord(orderName, orderInfo, s, 2, &multi_opaque_rect->nTopRect, FALSE) ||
1677 !read_order_field_coord(orderName, orderInfo, s, 3, &multi_opaque_rect->nWidth, FALSE) ||
1678 !read_order_field_coord(orderName, orderInfo, s, 4, &multi_opaque_rect->nHeight, FALSE))
1681 if ((orderInfo->fieldFlags & ORDER_FIELD_05) != 0)
1683 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1686 Stream_Read_UINT8(s,
byte);
1687 multi_opaque_rect->color = (multi_opaque_rect->color & 0x00FFFF00) | ((UINT32)byte);
1690 if ((orderInfo->fieldFlags & ORDER_FIELD_06) != 0)
1692 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1695 Stream_Read_UINT8(s,
byte);
1696 multi_opaque_rect->color = (multi_opaque_rect->color & 0x00FF00FF) | ((UINT32)
byte << 8);
1699 if ((orderInfo->fieldFlags & ORDER_FIELD_07) != 0)
1701 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1704 Stream_Read_UINT8(s,
byte);
1705 multi_opaque_rect->color = (multi_opaque_rect->color & 0x0000FFFF) | ((UINT32)
byte << 16);
1708 UINT32 numRectangles = multi_opaque_rect->numRectangles;
1709 if (!read_order_field_byte(orderName, orderInfo, s, 8, &numRectangles, TRUE))
1712 if ((orderInfo->fieldFlags & ORDER_FIELD_09) != 0)
1714 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1717 multi_opaque_rect->numRectangles = numRectangles;
1718 Stream_Read_UINT16(s, multi_opaque_rect->cbData);
1719 return update_read_delta_rects(s, multi_opaque_rect->rectangles,
1720 &multi_opaque_rect->numRectangles);
1722 if (numRectangles > multi_opaque_rect->numRectangles)
1724 WLog_ERR(TAG,
"%s numRectangles %" PRIu32
" > %" PRIu32, orderName, numRectangles,
1725 multi_opaque_rect->numRectangles);
1728 multi_opaque_rect->numRectangles = numRectangles;
1733static BOOL update_read_multi_draw_nine_grid_order(
const char* orderName,
wStream* s,
1737 UINT32 nDeltaEntries = multi_draw_nine_grid->nDeltaEntries;
1738 if (!read_order_field_coord(orderName, orderInfo, s, 1, &multi_draw_nine_grid->srcLeft,
1740 !read_order_field_coord(orderName, orderInfo, s, 2, &multi_draw_nine_grid->srcTop, FALSE) ||
1741 !read_order_field_coord(orderName, orderInfo, s, 3, &multi_draw_nine_grid->srcRight,
1743 !read_order_field_coord(orderName, orderInfo, s, 4, &multi_draw_nine_grid->srcBottom,
1745 !read_order_field_uint16(orderName, orderInfo, s, 5, &multi_draw_nine_grid->bitmapId,
1747 !read_order_field_byte(orderName, orderInfo, s, 6, &nDeltaEntries, TRUE))
1750 if ((orderInfo->fieldFlags & ORDER_FIELD_07) != 0)
1752 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
1755 multi_draw_nine_grid->nDeltaEntries = nDeltaEntries;
1756 Stream_Read_UINT16(s, multi_draw_nine_grid->cbData);
1757 return update_read_delta_rects(s, multi_draw_nine_grid->rectangles,
1758 &multi_draw_nine_grid->nDeltaEntries);
1761 if (nDeltaEntries > multi_draw_nine_grid->nDeltaEntries)
1763 WLog_ERR(TAG,
"%s nDeltaEntries %" PRIu32
" > %" PRIu32, orderName, nDeltaEntries,
1764 multi_draw_nine_grid->nDeltaEntries);
1767 multi_draw_nine_grid->nDeltaEntries = nDeltaEntries;
1771static BOOL update_read_line_to_order(
const char* orderName,
wStream* s,
1774 return (read_order_field_uint16(orderName, orderInfo, s, 1, &line_to->backMode, TRUE) &&
1775 read_order_field_coord(orderName, orderInfo, s, 2, &line_to->nXStart, FALSE) &&
1776 read_order_field_coord(orderName, orderInfo, s, 3, &line_to->nYStart, FALSE) &&
1777 read_order_field_coord(orderName, orderInfo, s, 4, &line_to->nXEnd, FALSE) &&
1778 read_order_field_coord(orderName, orderInfo, s, 5, &line_to->nYEnd, FALSE) &&
1779 read_order_field_color(orderName, orderInfo, s, 6, &line_to->backColor, TRUE) &&
1780 read_order_field_byte(orderName, orderInfo, s, 7, &line_to->bRop2, TRUE) &&
1781 read_order_field_byte(orderName, orderInfo, s, 8, &line_to->penStyle, TRUE) &&
1782 read_order_field_byte(orderName, orderInfo, s, 9, &line_to->penWidth, TRUE) &&
1783 read_order_field_color(orderName, orderInfo, s, 10, &line_to->penColor, TRUE));
1788 WINPR_UNUSED(orderInfo);
1789 WINPR_UNUSED(line_to);
1795 if (!Stream_EnsureRemainingCapacity(s, update_approximate_line_to_order(orderInfo, line_to)))
1798 orderInfo->fieldFlags = 0;
1799 orderInfo->fieldFlags |= ORDER_FIELD_01;
1800 Stream_Write_UINT16(s, get_checked_uint16(line_to->backMode));
1801 orderInfo->fieldFlags |= ORDER_FIELD_02;
1802 if (!update_write_coord(s, line_to->nXStart))
1804 orderInfo->fieldFlags |= ORDER_FIELD_03;
1805 if (!update_write_coord(s, line_to->nYStart))
1807 orderInfo->fieldFlags |= ORDER_FIELD_04;
1808 if (!update_write_coord(s, line_to->nXEnd))
1810 orderInfo->fieldFlags |= ORDER_FIELD_05;
1811 if (!update_write_coord(s, line_to->nYEnd))
1813 orderInfo->fieldFlags |= ORDER_FIELD_06;
1814 update_write_color(s, line_to->backColor);
1815 orderInfo->fieldFlags |= ORDER_FIELD_07;
1816 Stream_Write_UINT8(s, get_checked_uint8(line_to->bRop2));
1817 orderInfo->fieldFlags |= ORDER_FIELD_08;
1818 Stream_Write_UINT8(s, get_checked_uint8(line_to->penStyle));
1819 orderInfo->fieldFlags |= ORDER_FIELD_09;
1820 Stream_Write_UINT8(s, get_checked_uint8(line_to->penWidth));
1821 orderInfo->fieldFlags |= ORDER_FIELD_10;
1822 update_write_color(s, line_to->penColor);
1826static BOOL update_read_polyline_order(
const char* orderName,
wStream* s,
1830 UINT32 new_num = polyline->numDeltaEntries;
1831 if (!read_order_field_coord(orderName, orderInfo, s, 1, &polyline->xStart, FALSE) ||
1832 !read_order_field_coord(orderName, orderInfo, s, 2, &polyline->yStart, FALSE) ||
1833 !read_order_field_byte(orderName, orderInfo, s, 3, &polyline->bRop2, TRUE) ||
1834 !read_order_field_uint16(orderName, orderInfo, s, 4, &word, TRUE) ||
1835 !read_order_field_color(orderName, orderInfo, s, 5, &polyline->penColor, TRUE) ||
1836 !read_order_field_byte(orderName, orderInfo, s, 6, &new_num, TRUE))
1839 if ((orderInfo->fieldFlags & ORDER_FIELD_07) != 0)
1844 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1847 Stream_Read_UINT8(s, polyline->cbData);
1849 if (!check_val_fits_int16(polyline->xStart) || !check_val_fits_int16(polyline->yStart))
1852 polyline->numDeltaEntries = new_num;
1853 return update_read_delta_points(s, &polyline->points, polyline->numDeltaEntries,
1854 get_checked_int16(polyline->xStart),
1855 get_checked_int16(polyline->yStart));
1857 if (new_num > polyline->numDeltaEntries)
1859 WLog_ERR(TAG,
"%s numDeltaEntries %" PRIu32
" > %" PRIu32, orderName, new_num,
1860 polyline->numDeltaEntries);
1863 polyline->numDeltaEntries = new_num;
1868static BOOL update_read_memblt_order(
const char* orderName,
wStream* s,
const ORDER_INFO* orderInfo,
1871 if (!s || !orderInfo || !memblt)
1874 if (!read_order_field_uint16(orderName, orderInfo, s, 1, &memblt->cacheId, TRUE) ||
1875 !read_order_field_coord(orderName, orderInfo, s, 2, &memblt->nLeftRect, FALSE) ||
1876 !read_order_field_coord(orderName, orderInfo, s, 3, &memblt->nTopRect, FALSE) ||
1877 !read_order_field_coord(orderName, orderInfo, s, 4, &memblt->nWidth, FALSE) ||
1878 !read_order_field_coord(orderName, orderInfo, s, 5, &memblt->nHeight, FALSE) ||
1879 !read_order_field_byte(orderName, orderInfo, s, 6, &memblt->bRop, TRUE) ||
1880 !read_order_field_coord(orderName, orderInfo, s, 7, &memblt->nXSrc, FALSE) ||
1881 !read_order_field_coord(orderName, orderInfo, s, 8, &memblt->nYSrc, FALSE) ||
1882 !read_order_field_uint16(orderName, orderInfo, s, 9, &memblt->cacheIndex, TRUE))
1884 memblt->colorIndex = (memblt->cacheId >> 8);
1885 memblt->cacheId = (memblt->cacheId & 0xFF);
1886 memblt->bitmap =
nullptr;
1892 WINPR_UNUSED(orderInfo);
1893 WINPR_UNUSED(memblt);
1899 if (!Stream_EnsureRemainingCapacity(s, update_approximate_memblt_order(orderInfo, memblt)))
1902 const UINT16 cacheId = (UINT16)((memblt->cacheId & 0xFF) | ((memblt->colorIndex & 0xFF) << 8));
1903 orderInfo->fieldFlags |= ORDER_FIELD_01;
1904 Stream_Write_UINT16(s, cacheId);
1905 orderInfo->fieldFlags |= ORDER_FIELD_02;
1906 if (!update_write_coord(s, memblt->nLeftRect))
1908 orderInfo->fieldFlags |= ORDER_FIELD_03;
1909 if (!update_write_coord(s, memblt->nTopRect))
1911 orderInfo->fieldFlags |= ORDER_FIELD_04;
1912 if (!update_write_coord(s, memblt->nWidth))
1914 orderInfo->fieldFlags |= ORDER_FIELD_05;
1915 if (!update_write_coord(s, memblt->nHeight))
1917 orderInfo->fieldFlags |= ORDER_FIELD_06;
1918 Stream_Write_UINT8(s, get_checked_uint8(memblt->bRop));
1919 orderInfo->fieldFlags |= ORDER_FIELD_07;
1920 if (!update_write_coord(s, memblt->nXSrc))
1922 orderInfo->fieldFlags |= ORDER_FIELD_08;
1923 if (!update_write_coord(s, memblt->nYSrc))
1925 orderInfo->fieldFlags |= ORDER_FIELD_09;
1926 Stream_Write_UINT16(s, get_checked_uint16(memblt->cacheIndex));
1929static BOOL update_read_mem3blt_order(
const char* orderName,
wStream* s,
1932 if (!read_order_field_uint16(orderName, orderInfo, s, 1, &mem3blt->cacheId, TRUE) ||
1933 !read_order_field_coord(orderName, orderInfo, s, 2, &mem3blt->nLeftRect, FALSE) ||
1934 !read_order_field_coord(orderName, orderInfo, s, 3, &mem3blt->nTopRect, FALSE) ||
1935 !read_order_field_coord(orderName, orderInfo, s, 4, &mem3blt->nWidth, FALSE) ||
1936 !read_order_field_coord(orderName, orderInfo, s, 5, &mem3blt->nHeight, FALSE) ||
1937 !read_order_field_byte(orderName, orderInfo, s, 6, &mem3blt->bRop, TRUE) ||
1938 !read_order_field_coord(orderName, orderInfo, s, 7, &mem3blt->nXSrc, FALSE) ||
1939 !read_order_field_coord(orderName, orderInfo, s, 8, &mem3blt->nYSrc, FALSE) ||
1940 !read_order_field_color(orderName, orderInfo, s, 9, &mem3blt->backColor, TRUE) ||
1941 !read_order_field_color(orderName, orderInfo, s, 10, &mem3blt->foreColor, TRUE))
1944 if (!update_read_brush(s, &mem3blt->brush,
1945 get_checked_uint8((orderInfo->fieldFlags >> 10) & 0x1F)) ||
1946 !read_order_field_uint16(orderName, orderInfo, s, 16, &mem3blt->cacheIndex, TRUE))
1948 mem3blt->colorIndex = (mem3blt->cacheId >> 8);
1949 mem3blt->cacheId = (mem3blt->cacheId & 0xFF);
1950 mem3blt->bitmap =
nullptr;
1953static BOOL update_read_save_bitmap_order(
const char* orderName,
wStream* s,
1957 return (read_order_field_uint32(orderName, orderInfo, s, 1, &save_bitmap->savedBitmapPosition,
1959 read_order_field_coord(orderName, orderInfo, s, 2, &save_bitmap->nLeftRect, FALSE) &&
1960 read_order_field_coord(orderName, orderInfo, s, 3, &save_bitmap->nTopRect, FALSE) &&
1961 read_order_field_coord(orderName, orderInfo, s, 4, &save_bitmap->nRightRect, FALSE) &&
1962 read_order_field_coord(orderName, orderInfo, s, 5, &save_bitmap->nBottomRect, FALSE) &&
1963 read_order_field_byte(orderName, orderInfo, s, 6, &save_bitmap->operation, TRUE));
1965static BOOL update_read_glyph_index_order(
const char* orderName,
wStream* s,
1969 if (!read_order_field_byte(orderName, orderInfo, s, 1, &glyph_index->cacheId, TRUE) ||
1970 !read_order_field_byte(orderName, orderInfo, s, 2, &glyph_index->flAccel, TRUE) ||
1971 !read_order_field_byte(orderName, orderInfo, s, 3, &glyph_index->ulCharInc, TRUE) ||
1972 !read_order_field_byte(orderName, orderInfo, s, 4, &glyph_index->fOpRedundant, TRUE) ||
1973 !read_order_field_color(orderName, orderInfo, s, 5, &glyph_index->backColor, TRUE) ||
1974 !read_order_field_color(orderName, orderInfo, s, 6, &glyph_index->foreColor, TRUE) ||
1975 !read_order_field_int16(orderName, orderInfo, s, 7, &glyph_index->bkLeft, TRUE) ||
1976 !read_order_field_int16(orderName, orderInfo, s, 8, &glyph_index->bkTop, TRUE) ||
1977 !read_order_field_int16(orderName, orderInfo, s, 9, &glyph_index->bkRight, TRUE) ||
1978 !read_order_field_int16(orderName, orderInfo, s, 10, &glyph_index->bkBottom, TRUE) ||
1979 !read_order_field_int16(orderName, orderInfo, s, 11, &glyph_index->opLeft, TRUE) ||
1980 !read_order_field_int16(orderName, orderInfo, s, 12, &glyph_index->opTop, TRUE) ||
1981 !read_order_field_int16(orderName, orderInfo, s, 13, &glyph_index->opRight, TRUE) ||
1982 !read_order_field_int16(orderName, orderInfo, s, 14, &glyph_index->opBottom, TRUE) ||
1983 !update_read_brush(s, &glyph_index->brush,
1984 get_checked_uint8((orderInfo->fieldFlags >> 14) & 0x1F)) ||
1985 !read_order_field_int16(orderName, orderInfo, s, 20, &glyph_index->x, TRUE) ||
1986 !read_order_field_int16(orderName, orderInfo, s, 21, &glyph_index->y, TRUE))
1989 if ((orderInfo->fieldFlags & ORDER_FIELD_22) != 0)
1991 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1994 Stream_Read_UINT8(s, glyph_index->cbData);
1996 if (!Stream_CheckAndLogRequiredLength(TAG, s, glyph_index->cbData))
1999 CopyMemory(glyph_index->data, Stream_ConstPointer(s), glyph_index->cbData);
2000 Stream_Seek(s, glyph_index->cbData);
2006size_t update_approximate_glyph_index_order(
ORDER_INFO* orderInfo,
2009 WINPR_UNUSED(orderInfo);
2010 WINPR_UNUSED(glyph_index);
2017 size_t inf = update_approximate_glyph_index_order(orderInfo, glyph_index);
2019 if (!Stream_EnsureRemainingCapacity(s, inf))
2022 if (!Stream_EnsureRemainingCapacity(s, 4))
2024 orderInfo->fieldFlags = 0;
2025 orderInfo->fieldFlags |= ORDER_FIELD_01;
2026 Stream_Write_UINT8(s, get_checked_uint8(glyph_index->cacheId));
2027 orderInfo->fieldFlags |= ORDER_FIELD_02;
2028 Stream_Write_UINT8(s, get_checked_uint8(glyph_index->flAccel));
2029 orderInfo->fieldFlags |= ORDER_FIELD_03;
2030 Stream_Write_UINT8(s, get_checked_uint8(glyph_index->ulCharInc));
2031 orderInfo->fieldFlags |= ORDER_FIELD_04;
2032 Stream_Write_UINT8(s, get_checked_uint8(glyph_index->fOpRedundant));
2033 orderInfo->fieldFlags |= ORDER_FIELD_05;
2034 if (!update_write_color(s, get_checked_uint8(glyph_index->backColor)))
2036 orderInfo->fieldFlags |= ORDER_FIELD_06;
2037 if (!update_write_color(s, glyph_index->foreColor))
2040 if (!Stream_EnsureRemainingCapacity(s, 14))
2042 orderInfo->fieldFlags |= ORDER_FIELD_07;
2043 Stream_Write_INT16(s, get_checked_int16(glyph_index->bkLeft));
2044 orderInfo->fieldFlags |= ORDER_FIELD_08;
2045 Stream_Write_INT16(s, get_checked_int16(glyph_index->bkTop));
2046 orderInfo->fieldFlags |= ORDER_FIELD_09;
2047 Stream_Write_INT16(s, get_checked_int16(glyph_index->bkRight));
2048 orderInfo->fieldFlags |= ORDER_FIELD_10;
2049 Stream_Write_INT16(s, get_checked_int16(glyph_index->bkBottom));
2050 orderInfo->fieldFlags |= ORDER_FIELD_11;
2051 Stream_Write_INT16(s, get_checked_int16(glyph_index->opLeft));
2052 orderInfo->fieldFlags |= ORDER_FIELD_12;
2053 Stream_Write_INT16(s, get_checked_int16(glyph_index->opTop));
2054 orderInfo->fieldFlags |= ORDER_FIELD_13;
2055 Stream_Write_INT16(s, get_checked_int16(glyph_index->opRight));
2056 orderInfo->fieldFlags |= ORDER_FIELD_14;
2057 Stream_Write_INT16(s, get_checked_int16(glyph_index->opBottom));
2058 orderInfo->fieldFlags |= ORDER_FIELD_15;
2059 orderInfo->fieldFlags |= ORDER_FIELD_16;
2060 orderInfo->fieldFlags |= ORDER_FIELD_17;
2061 orderInfo->fieldFlags |= ORDER_FIELD_18;
2062 orderInfo->fieldFlags |= ORDER_FIELD_19;
2063 if (!update_write_brush(s, &glyph_index->brush,
2064 get_checked_uint8((orderInfo->fieldFlags >> 14) & 0x1F)))
2067 if (!Stream_EnsureRemainingCapacity(s, 5ULL + glyph_index->cbData))
2069 orderInfo->fieldFlags |= ORDER_FIELD_20;
2070 Stream_Write_INT16(s, get_checked_int16(glyph_index->x));
2071 orderInfo->fieldFlags |= ORDER_FIELD_21;
2072 Stream_Write_INT16(s, get_checked_int16(glyph_index->y));
2073 orderInfo->fieldFlags |= ORDER_FIELD_22;
2074 Stream_Write_UINT8(s, get_checked_uint8(glyph_index->cbData));
2075 Stream_Write(s, glyph_index->data, glyph_index->cbData);
2078static BOOL update_read_fast_index_order(
const char* orderName,
wStream* s,
2081 if (!read_order_field_byte(orderName, orderInfo, s, 1, &fast_index->cacheId, TRUE) ||
2082 !read_order_field_2bytes(orderName, orderInfo, s, 2, &fast_index->ulCharInc,
2083 &fast_index->flAccel, TRUE) ||
2084 !read_order_field_color(orderName, orderInfo, s, 3, &fast_index->backColor, TRUE) ||
2085 !read_order_field_color(orderName, orderInfo, s, 4, &fast_index->foreColor, TRUE) ||
2086 !read_order_field_coord(orderName, orderInfo, s, 5, &fast_index->bkLeft, FALSE) ||
2087 !read_order_field_coord(orderName, orderInfo, s, 6, &fast_index->bkTop, FALSE) ||
2088 !read_order_field_coord(orderName, orderInfo, s, 7, &fast_index->bkRight, FALSE) ||
2089 !read_order_field_coord(orderName, orderInfo, s, 8, &fast_index->bkBottom, FALSE) ||
2090 !read_order_field_coord(orderName, orderInfo, s, 9, &fast_index->opLeft, FALSE) ||
2091 !read_order_field_coord(orderName, orderInfo, s, 10, &fast_index->opTop, FALSE) ||
2092 !read_order_field_coord(orderName, orderInfo, s, 11, &fast_index->opRight, FALSE) ||
2093 !read_order_field_coord(orderName, orderInfo, s, 12, &fast_index->opBottom, FALSE) ||
2094 !read_order_field_coord(orderName, orderInfo, s, 13, &fast_index->x, FALSE) ||
2095 !read_order_field_coord(orderName, orderInfo, s, 14, &fast_index->y, FALSE))
2098 if ((orderInfo->fieldFlags & ORDER_FIELD_15) != 0)
2100 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
2103 Stream_Read_UINT8(s, fast_index->cbData);
2105 if (!Stream_CheckAndLogRequiredLength(TAG, s, fast_index->cbData))
2108 CopyMemory(fast_index->data, Stream_ConstPointer(s), fast_index->cbData);
2109 Stream_Seek(s, fast_index->cbData);
2114static BOOL update_read_fast_glyph_order(
const char* orderName,
wStream* s,
2118 if (!read_order_field_byte(orderName, orderInfo, s, 1, &fastGlyph->cacheId, TRUE))
2120 if (fastGlyph->cacheId > 9)
2122 if (!read_order_field_2bytes(orderName, orderInfo, s, 2, &fastGlyph->ulCharInc,
2123 &fastGlyph->flAccel, TRUE) ||
2124 !read_order_field_color(orderName, orderInfo, s, 3, &fastGlyph->backColor, TRUE) ||
2125 !read_order_field_color(orderName, orderInfo, s, 4, &fastGlyph->foreColor, TRUE) ||
2126 !read_order_field_coord(orderName, orderInfo, s, 5, &fastGlyph->bkLeft, FALSE) ||
2127 !read_order_field_coord(orderName, orderInfo, s, 6, &fastGlyph->bkTop, FALSE) ||
2128 !read_order_field_coord(orderName, orderInfo, s, 7, &fastGlyph->bkRight, FALSE) ||
2129 !read_order_field_coord(orderName, orderInfo, s, 8, &fastGlyph->bkBottom, FALSE) ||
2130 !read_order_field_coord(orderName, orderInfo, s, 9, &fastGlyph->opLeft, FALSE) ||
2131 !read_order_field_coord(orderName, orderInfo, s, 10, &fastGlyph->opTop, FALSE) ||
2132 !read_order_field_coord(orderName, orderInfo, s, 11, &fastGlyph->opRight, FALSE) ||
2133 !read_order_field_coord(orderName, orderInfo, s, 12, &fastGlyph->opBottom, FALSE) ||
2134 !read_order_field_coord(orderName, orderInfo, s, 13, &fastGlyph->x, FALSE) ||
2135 !read_order_field_coord(orderName, orderInfo, s, 14, &fastGlyph->y, FALSE))
2138 if ((orderInfo->fieldFlags & ORDER_FIELD_15) != 0)
2140 const BYTE* src =
nullptr;
2143 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
2146 Stream_Read_UINT8(s, fastGlyph->cbData);
2148 src = Stream_ConstPointer(s);
2149 if (!Stream_SafeSeek(s, fastGlyph->cbData) || (fastGlyph->cbData == 0))
2152 CopyMemory(fastGlyph->data, src, fastGlyph->cbData);
2153 sub = Stream_StaticInit(&subbuffer, fastGlyph->data, fastGlyph->cbData);
2155 Stream_Read_UINT8(sub, glyph->cacheIndex);
2157 if (fastGlyph->cbData > 1)
2159 if (!update_read_2byte_signed(sub, &glyph->x) ||
2160 !update_read_2byte_signed(sub, &glyph->y) ||
2161 !update_read_2byte_unsigned(sub, &glyph->cx) ||
2162 !update_read_2byte_unsigned(sub, &glyph->cy))
2165 if ((glyph->cx == 0) || (glyph->cy == 0))
2167 WLog_ERR(TAG,
"GLYPH_DATA_V2::cx=%" PRIu32
", GLYPH_DATA_V2::cy=%" PRIu32,
2168 glyph->cx, glyph->cy);
2172 const size_t slen = Stream_GetRemainingLength(sub);
2173 if (slen > UINT32_MAX)
2175 glyph->cb = (UINT32)slen;
2178 BYTE* new_aj = (BYTE*)realloc(glyph->aj, glyph->cb);
2184 Stream_Read(sub, glyph->aj, glyph->cb);
2189 glyph->aj =
nullptr;
2197static BOOL update_read_polygon_sc_order(
const char* orderName,
wStream* s,
2200 UINT32 num = polygon_sc->numPoints;
2201 if (!read_order_field_coord(orderName, orderInfo, s, 1, &polygon_sc->xStart, FALSE) ||
2202 !read_order_field_coord(orderName, orderInfo, s, 2, &polygon_sc->yStart, FALSE) ||
2203 !read_order_field_byte(orderName, orderInfo, s, 3, &polygon_sc->bRop2, TRUE) ||
2204 !read_order_field_byte(orderName, orderInfo, s, 4, &polygon_sc->fillMode, TRUE) ||
2205 !read_order_field_color(orderName, orderInfo, s, 5, &polygon_sc->brushColor, TRUE) ||
2206 !read_order_field_byte(orderName, orderInfo, s, 6, &num, TRUE))
2209 if ((orderInfo->fieldFlags & ORDER_FIELD_07) != 0)
2214 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
2217 Stream_Read_UINT8(s, polygon_sc->cbData);
2219 if (!check_val_fits_int16(polygon_sc->xStart) || !check_val_fits_int16(polygon_sc->yStart))
2222 polygon_sc->numPoints = num;
2223 return update_read_delta_points(s, &polygon_sc->points, polygon_sc->numPoints,
2224 get_checked_int16(polygon_sc->xStart),
2225 get_checked_int16(polygon_sc->yStart));
2227 if (num > polygon_sc->numPoints)
2229 WLog_ERR(TAG,
"%s numPoints %" PRIu32
" > %" PRIu32, orderName, num, polygon_sc->numPoints);
2232 polygon_sc->numPoints = num;
2236static BOOL update_read_polygon_cb_order(
const char* orderName,
wStream* s,
2239 UINT32 num = polygon_cb->numPoints;
2240 if (!read_order_field_coord(orderName, orderInfo, s, 1, &polygon_cb->xStart, FALSE) ||
2241 !read_order_field_coord(orderName, orderInfo, s, 2, &polygon_cb->yStart, FALSE) ||
2242 !read_order_field_byte(orderName, orderInfo, s, 3, &polygon_cb->bRop2, TRUE) ||
2243 !read_order_field_byte(orderName, orderInfo, s, 4, &polygon_cb->fillMode, TRUE) ||
2244 !read_order_field_color(orderName, orderInfo, s, 5, &polygon_cb->backColor, TRUE) ||
2245 !read_order_field_color(orderName, orderInfo, s, 6, &polygon_cb->foreColor, TRUE))
2248 if (!update_read_brush(s, &polygon_cb->brush,
2249 get_checked_uint8((orderInfo->fieldFlags >> 6) & 0x1F)))
2252 if (!read_order_field_byte(orderName, orderInfo, s, 12, &num, TRUE))
2255 if ((orderInfo->fieldFlags & ORDER_FIELD_13) != 0)
2260 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
2263 Stream_Read_UINT8(s, polygon_cb->cbData);
2264 polygon_cb->numPoints = num;
2266 if (!check_val_fits_int16(polygon_cb->xStart) || !check_val_fits_int16(polygon_cb->yStart))
2269 if (!update_read_delta_points(s, &polygon_cb->points, polygon_cb->numPoints,
2270 get_checked_int16(polygon_cb->xStart),
2271 get_checked_int16(polygon_cb->yStart)))
2275 if (num > polygon_cb->numPoints)
2277 WLog_ERR(TAG,
"%s numPoints %" PRIu32
" > %" PRIu32, orderName, num, polygon_cb->numPoints);
2280 polygon_cb->numPoints = num;
2282 polygon_cb->backMode = (polygon_cb->bRop2 & 0x80) ? BACKMODE_TRANSPARENT : BACKMODE_OPAQUE;
2283 polygon_cb->bRop2 = (polygon_cb->bRop2 & 0x1F);
2286static BOOL update_read_ellipse_sc_order(
const char* orderName,
wStream* s,
2289 return (read_order_field_coord(orderName, orderInfo, s, 1, &ellipse_sc->leftRect, FALSE) &&
2290 read_order_field_coord(orderName, orderInfo, s, 2, &ellipse_sc->topRect, FALSE) &&
2291 read_order_field_coord(orderName, orderInfo, s, 3, &ellipse_sc->rightRect, FALSE) &&
2292 read_order_field_coord(orderName, orderInfo, s, 4, &ellipse_sc->bottomRect, FALSE) &&
2293 read_order_field_byte(orderName, orderInfo, s, 5, &ellipse_sc->bRop2, TRUE) &&
2294 read_order_field_byte(orderName, orderInfo, s, 6, &ellipse_sc->fillMode, TRUE) &&
2295 read_order_field_color(orderName, orderInfo, s, 7, &ellipse_sc->color, TRUE));
2297static BOOL update_read_ellipse_cb_order(
const char* orderName,
wStream* s,
2300 return (read_order_field_coord(orderName, orderInfo, s, 1, &ellipse_cb->leftRect, FALSE) &&
2301 read_order_field_coord(orderName, orderInfo, s, 2, &ellipse_cb->topRect, FALSE) &&
2302 read_order_field_coord(orderName, orderInfo, s, 3, &ellipse_cb->rightRect, FALSE) &&
2303 read_order_field_coord(orderName, orderInfo, s, 4, &ellipse_cb->bottomRect, FALSE) &&
2304 read_order_field_byte(orderName, orderInfo, s, 5, &ellipse_cb->bRop2, TRUE) &&
2305 read_order_field_byte(orderName, orderInfo, s, 6, &ellipse_cb->fillMode, TRUE) &&
2306 read_order_field_color(orderName, orderInfo, s, 7, &ellipse_cb->backColor, TRUE) &&
2307 read_order_field_color(orderName, orderInfo, s, 8, &ellipse_cb->foreColor, TRUE) &&
2308 update_read_brush(s, &ellipse_cb->brush,
2309 get_checked_uint8((orderInfo->fieldFlags >> 8) & 0x1F)));
2313WINPR_ATTR_MALLOC(free_cache_bitmap_order, 2)
2316 BOOL compressed, UINT16 flags)
2329 if (!Stream_CheckAndLogRequiredLength(TAG, s, 9))
2332 Stream_Read_UINT8(s, cache_bitmap->cacheId);
2333 Stream_Seek_UINT8(s);
2334 Stream_Read_UINT8(s, cache_bitmap->bitmapWidth);
2335 Stream_Read_UINT8(s, cache_bitmap->bitmapHeight);
2336 Stream_Read_UINT8(s, cache_bitmap->bitmapBpp);
2338 if ((cache_bitmap->bitmapBpp < 1) || (cache_bitmap->bitmapBpp > 32))
2340 WLog_Print(up->log, WLOG_ERROR,
"invalid bitmap bpp %" PRIu32
"", cache_bitmap->bitmapBpp);
2344 Stream_Read_UINT16(s, cache_bitmap->bitmapLength);
2345 Stream_Read_UINT16(s, cache_bitmap->cacheIndex);
2349 if ((flags & NO_BITMAP_COMPRESSION_HDR) == 0)
2351 BYTE* bitmapComprHdr = (BYTE*)&(cache_bitmap->bitmapComprHdr);
2353 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
2356 Stream_Read(s, bitmapComprHdr, 8);
2357 if (cache_bitmap->bitmapLength < 8)
2359 cache_bitmap->bitmapLength -= 8;
2363 if (cache_bitmap->bitmapLength == 0)
2366 if (!Stream_CheckAndLogRequiredLength(TAG, s, cache_bitmap->bitmapLength))
2369 cache_bitmap->bitmapDataStream = malloc(cache_bitmap->bitmapLength);
2371 if (!cache_bitmap->bitmapDataStream)
2374 Stream_Read(s, cache_bitmap->bitmapDataStream, cache_bitmap->bitmapLength);
2375 cache_bitmap->compressed = compressed;
2376 return cache_bitmap;
2378 WINPR_PRAGMA_DIAG_PUSH
2379 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
2380 free_cache_bitmap_order(update->context, cache_bitmap);
2381 WINPR_PRAGMA_DIAG_POP
2386 BOOL compressed,
const UINT16* flags)
2388 WINPR_ASSERT(cache_bitmap);
2389 WINPR_UNUSED(compressed);
2390 WINPR_UNUSED(flags);
2391 return 64 + cache_bitmap->bitmapLength;
2395 BOOL compressed, UINT16* flags)
2397 UINT32 bitmapLength = cache_bitmap->bitmapLength;
2398 size_t inf = update_approximate_cache_bitmap_order(cache_bitmap, compressed, flags);
2400 if (!Stream_EnsureRemainingCapacity(s, inf))
2403 *flags = NO_BITMAP_COMPRESSION_HDR;
2405 if ((*flags & NO_BITMAP_COMPRESSION_HDR) == 0)
2408 Stream_Write_UINT8(s, get_checked_uint8(cache_bitmap->cacheId));
2409 Stream_Write_UINT8(s, 0);
2410 Stream_Write_UINT8(s, get_checked_uint8(cache_bitmap->bitmapWidth));
2411 Stream_Write_UINT8(s,
2412 get_checked_uint8(cache_bitmap->bitmapHeight));
2413 Stream_Write_UINT8(s, get_checked_uint8(cache_bitmap->bitmapBpp));
2414 Stream_Write_UINT16(s, get_checked_uint16(bitmapLength));
2415 Stream_Write_UINT16(s, get_checked_uint16(cache_bitmap->cacheIndex));
2419 if ((*flags & NO_BITMAP_COMPRESSION_HDR) == 0)
2421 const BYTE* bitmapComprHdr = (
const BYTE*)&(cache_bitmap->bitmapComprHdr);
2422 Stream_Write(s, bitmapComprHdr, 8);
2426 Stream_Write(s, cache_bitmap->bitmapDataStream, bitmapLength);
2430 Stream_Write(s, cache_bitmap->bitmapDataStream, bitmapLength);
2436WINPR_ATTR_MALLOC(free_cache_bitmap_v2_order, 2)
2439 BOOL compressed, UINT16 flags)
2442 BYTE bitsPerPixelId = 0;
2450 if (!cache_bitmap_v2)
2453 cache_bitmap_v2->cacheId = flags & 0x0003;
2454 cache_bitmap_v2->flags = (flags & 0xFF80) >> 7;
2455 bitsPerPixelId = (flags & 0x0078) >> 3;
2456 cache_bitmap_v2->bitmapBpp = get_cbr2_bpp(bitsPerPixelId, &rc);
2460 if (cache_bitmap_v2->flags & CBR2_PERSISTENT_KEY_PRESENT)
2462 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
2465 Stream_Read_UINT32(s, cache_bitmap_v2->key1);
2466 Stream_Read_UINT32(s, cache_bitmap_v2->key2);
2469 if (cache_bitmap_v2->flags & CBR2_HEIGHT_SAME_AS_WIDTH)
2471 if (!update_read_2byte_unsigned(s, &cache_bitmap_v2->bitmapWidth))
2474 cache_bitmap_v2->bitmapHeight = cache_bitmap_v2->bitmapWidth;
2478 if (!update_read_2byte_unsigned(s, &cache_bitmap_v2->bitmapWidth) ||
2479 !update_read_2byte_unsigned(s, &cache_bitmap_v2->bitmapHeight))
2483 if (!update_read_4byte_unsigned(s, &cache_bitmap_v2->bitmapLength) ||
2484 !update_read_2byte_unsigned(s, &cache_bitmap_v2->cacheIndex))
2487 if (cache_bitmap_v2->flags & CBR2_DO_NOT_CACHE)
2488 cache_bitmap_v2->cacheIndex = BITMAP_CACHE_WAITING_LIST_INDEX;
2492 if (!(cache_bitmap_v2->flags & CBR2_NO_BITMAP_COMPRESSION_HDR))
2494 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
2498 s, cache_bitmap_v2->cbCompFirstRowSize);
2500 s, cache_bitmap_v2->cbCompMainBodySize);
2501 Stream_Read_UINT16(s, cache_bitmap_v2->cbScanWidth);
2503 s, cache_bitmap_v2->cbUncompressedSize);
2504 cache_bitmap_v2->bitmapLength = cache_bitmap_v2->cbCompMainBodySize;
2508 if (cache_bitmap_v2->bitmapLength == 0)
2511 if (!Stream_CheckAndLogRequiredLength(TAG, s, cache_bitmap_v2->bitmapLength))
2514 if (cache_bitmap_v2->bitmapLength == 0)
2517 cache_bitmap_v2->bitmapDataStream = malloc(cache_bitmap_v2->bitmapLength);
2519 if (!cache_bitmap_v2->bitmapDataStream)
2522 Stream_Read(s, cache_bitmap_v2->bitmapDataStream, cache_bitmap_v2->bitmapLength);
2523 cache_bitmap_v2->compressed = compressed;
2524 return cache_bitmap_v2;
2526 WINPR_PRAGMA_DIAG_PUSH
2527 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
2528 free_cache_bitmap_v2_order(update->context, cache_bitmap_v2);
2529 WINPR_PRAGMA_DIAG_POP
2534 BOOL compressed,
const UINT16* flags)
2536 WINPR_ASSERT(cache_bitmap_v2);
2537 WINPR_UNUSED(flags);
2538 WINPR_UNUSED(compressed);
2540 return 64 + cache_bitmap_v2->bitmapLength;
2544 BOOL compressed, UINT16* flags)
2547 BYTE bitsPerPixelId = 0;
2549 if (!Stream_EnsureRemainingCapacity(
2550 s, update_approximate_cache_bitmap_v2_order(cache_bitmap_v2, compressed, flags)))
2553 bitsPerPixelId = get_bpp_bmf(cache_bitmap_v2->bitmapBpp, &rc);
2556 WINPR_ASSERT(cache_bitmap_v2->cacheId <= 3);
2557 WINPR_ASSERT(bitsPerPixelId <= 0x0f);
2558 WINPR_ASSERT(cache_bitmap_v2->flags <= 0x1FF);
2559 *flags = (UINT16)((cache_bitmap_v2->cacheId & 0x0003) | ((bitsPerPixelId << 3) & 0xFFFF) |
2560 ((cache_bitmap_v2->flags << 7) & 0xFF80));
2562 if (cache_bitmap_v2->flags & CBR2_PERSISTENT_KEY_PRESENT)
2564 Stream_Write_UINT32(s, cache_bitmap_v2->key1);
2565 Stream_Write_UINT32(s, cache_bitmap_v2->key2);
2568 if (cache_bitmap_v2->flags & CBR2_HEIGHT_SAME_AS_WIDTH)
2570 if (!update_write_2byte_unsigned(s, cache_bitmap_v2->bitmapWidth))
2575 if (!update_write_2byte_unsigned(s, cache_bitmap_v2->bitmapWidth) ||
2576 !update_write_2byte_unsigned(s, cache_bitmap_v2->bitmapHeight))
2580 if (cache_bitmap_v2->flags & CBR2_DO_NOT_CACHE)
2581 cache_bitmap_v2->cacheIndex = BITMAP_CACHE_WAITING_LIST_INDEX;
2583 if (!update_write_4byte_unsigned(s, cache_bitmap_v2->bitmapLength) ||
2584 !update_write_2byte_unsigned(s, cache_bitmap_v2->cacheIndex))
2589 if (!(cache_bitmap_v2->flags & CBR2_NO_BITMAP_COMPRESSION_HDR))
2591 Stream_Write_UINT16(
2592 s, get_checked_uint16(
2593 cache_bitmap_v2->cbCompFirstRowSize));
2594 Stream_Write_UINT16(
2595 s, get_checked_uint16(
2596 cache_bitmap_v2->cbCompMainBodySize));
2597 Stream_Write_UINT16(
2598 s, get_checked_uint16(cache_bitmap_v2->cbScanWidth));
2599 Stream_Write_UINT16(
2600 s, get_checked_uint16(
2601 cache_bitmap_v2->cbUncompressedSize));
2602 cache_bitmap_v2->bitmapLength = cache_bitmap_v2->cbCompMainBodySize;
2605 if (!Stream_EnsureRemainingCapacity(s, cache_bitmap_v2->bitmapLength))
2608 Stream_Write(s, cache_bitmap_v2->bitmapDataStream, cache_bitmap_v2->bitmapLength);
2612 if (!Stream_EnsureRemainingCapacity(s, cache_bitmap_v2->bitmapLength))
2615 Stream_Write(s, cache_bitmap_v2->bitmapDataStream, cache_bitmap_v2->bitmapLength);
2618 cache_bitmap_v2->compressed = compressed;
2622WINPR_ATTR_MALLOC(free_cache_bitmap_v3_order, 2)
2628 BYTE bitsPerPixelId = 0;
2631 BYTE* new_data =
nullptr;
2640 if (!cache_bitmap_v3)
2643 cache_bitmap_v3->cacheId = flags & 0x00000003;
2644 cache_bitmap_v3->flags = (flags & 0x0000FF80) >> 7;
2645 bitsPerPixelId = (flags & 0x00000078) >> 3;
2646 cache_bitmap_v3->bpp = get_cbr2_bpp(bitsPerPixelId, &rc);
2650 if (!Stream_CheckAndLogRequiredLength(TAG, s, 21))
2653 Stream_Read_UINT16(s, cache_bitmap_v3->cacheIndex);
2654 Stream_Read_UINT32(s, cache_bitmap_v3->key1);
2655 Stream_Read_UINT32(s, cache_bitmap_v3->key2);
2656 bitmapData = &cache_bitmap_v3->bitmapData;
2657 Stream_Read_UINT8(s, bitmapData->bpp);
2659 if ((bitmapData->bpp < 1) || (bitmapData->bpp > 32))
2661 WLog_Print(up->log, WLOG_ERROR,
"invalid bpp value %" PRIu32
"", bitmapData->bpp);
2665 Stream_Seek_UINT8(s);
2666 Stream_Seek_UINT8(s);
2667 Stream_Read_UINT8(s, bitmapData->codecID);
2668 Stream_Read_UINT16(s, bitmapData->width);
2669 Stream_Read_UINT16(s, bitmapData->height);
2670 Stream_Read_UINT32(s, new_len);
2672 if ((new_len == 0) || (!Stream_CheckAndLogRequiredLength(TAG, s, new_len)))
2675 new_data = (BYTE*)realloc(bitmapData->data, new_len);
2680 bitmapData->data = new_data;
2681 bitmapData->length = new_len;
2682 Stream_Read(s, bitmapData->data, bitmapData->length);
2683 return cache_bitmap_v3;
2685 WINPR_PRAGMA_DIAG_PUSH
2686 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
2687 free_cache_bitmap_v3_order(update->context, cache_bitmap_v3);
2688 WINPR_PRAGMA_DIAG_POP
2693 WINPR_ATTR_UNUSED UINT16* flags)
2696 return 64 + bitmapData->length;
2703 BYTE bitsPerPixelId = 0;
2706 if (!Stream_EnsureRemainingCapacity(
2707 s, update_approximate_cache_bitmap_v3_order(cache_bitmap_v3, flags)))
2710 bitmapData = &cache_bitmap_v3->bitmapData;
2711 bitsPerPixelId = get_bpp_bmf(cache_bitmap_v3->bpp, &rc);
2714 *flags = (cache_bitmap_v3->cacheId & 0x00000003) |
2715 ((cache_bitmap_v3->flags << 7) & 0x0000FF80) | ((bitsPerPixelId << 3) & 0x00000078);
2716 Stream_Write_UINT16(s,
2717 get_checked_uint16(cache_bitmap_v3->cacheIndex));
2718 Stream_Write_UINT32(s, cache_bitmap_v3->key1);
2719 Stream_Write_UINT32(s, cache_bitmap_v3->key2);
2720 Stream_Write_UINT8(s, get_checked_uint8(bitmapData->bpp));
2721 Stream_Write_UINT8(s, 0);
2722 Stream_Write_UINT8(s, 0);
2723 Stream_Write_UINT8(s, get_checked_uint8(bitmapData->codecID));
2724 Stream_Write_UINT16(s, get_checked_uint16(bitmapData->width));
2725 Stream_Write_UINT16(s, get_checked_uint16(bitmapData->height));
2726 Stream_Write_UINT32(s, bitmapData->length);
2727 Stream_Write(s, bitmapData->data, bitmapData->length);
2731WINPR_ATTR_MALLOC(free_cache_color_table_order, 2)
2734 WINPR_ATTR_UNUSED UINT16 flags)
2736 UINT32* colorTable =
nullptr;
2739 if (!cache_color_table)
2742 if (!Stream_CheckAndLogRequiredLength(TAG, s, 3))
2745 Stream_Read_UINT8(s, cache_color_table->cacheIndex);
2746 Stream_Read_UINT16(s, cache_color_table->numberColors);
2748 if (cache_color_table->numberColors != 256)
2754 if (!Stream_CheckAndLogRequiredLengthOfSize(TAG, s, cache_color_table->numberColors, 4ull))
2757 colorTable = (UINT32*)&cache_color_table->colorTable;
2759 for (UINT32 i = 0; i < cache_color_table->numberColors; i++)
2760 update_read_color_quad(s, &colorTable[i]);
2762 return cache_color_table;
2764 WINPR_PRAGMA_DIAG_PUSH
2765 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
2766 free_cache_color_table_order(update->context, cache_color_table);
2767 WINPR_PRAGMA_DIAG_POP
2772 const UINT16* flags)
2774 WINPR_UNUSED(cache_color_table);
2775 WINPR_UNUSED(flags);
2777 return 16 + (256 * 4);
2780BOOL update_write_cache_color_table_order(
wStream* s,
2785 const UINT32* colorTable =
nullptr;
2787 if (cache_color_table->numberColors != 256)
2790 inf = update_approximate_cache_color_table_order(cache_color_table, flags);
2792 if (!Stream_EnsureRemainingCapacity(s, inf))
2795 Stream_Write_UINT8(s,
2796 get_checked_uint8(cache_color_table->cacheIndex));
2797 Stream_Write_UINT16(
2798 s, get_checked_uint16(cache_color_table->numberColors));
2799 colorTable = (
const UINT32*)&cache_color_table->colorTable;
2801 for (
size_t i = 0; i < cache_color_table->numberColors; i++)
2803 update_write_color_quad(s, colorTable[i]);
2812 WINPR_ASSERT(update);
2815 if (!cache_glyph_order)
2818 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
2821 Stream_Read_UINT8(s, cache_glyph_order->cacheId);
2822 Stream_Read_UINT8(s, cache_glyph_order->cGlyphs);
2824 for (UINT32 i = 0; i < cache_glyph_order->cGlyphs; i++)
2826 GLYPH_DATA* glyph = &cache_glyph_order->glyphData[i];
2828 if (!Stream_CheckAndLogRequiredLength(TAG, s, 10))
2831 Stream_Read_UINT16(s, glyph->cacheIndex);
2832 Stream_Read_INT16(s, glyph->x);
2833 Stream_Read_INT16(s, glyph->y);
2834 Stream_Read_UINT16(s, glyph->cx);
2835 Stream_Read_UINT16(s, glyph->cy);
2836 glyph->cb = ((glyph->cx + 7) / 8) * glyph->cy;
2837 glyph->cb += ((glyph->cb % 4) > 0) ? 4 - (glyph->cb % 4) : 0;
2839 if (!Stream_CheckAndLogRequiredLength(TAG, s, glyph->cb))
2842 glyph->aj = (BYTE*)malloc(glyph->cb);
2847 Stream_Read(s, glyph->aj, glyph->cb);
2850 if ((flags & CG_GLYPH_UNICODE_PRESENT) && (cache_glyph_order->cGlyphs > 0))
2852 cache_glyph_order->unicodeCharacters = calloc(cache_glyph_order->cGlyphs,
sizeof(WCHAR));
2854 if (!cache_glyph_order->unicodeCharacters)
2857 if (!Stream_CheckAndLogRequiredLengthOfSize(TAG, s, cache_glyph_order->cGlyphs,
2861 if (!Stream_Read_UTF16_String(s, cache_glyph_order->unicodeCharacters,
2862 cache_glyph_order->cGlyphs))
2866 return cache_glyph_order;
2868 free_cache_glyph_order(update->context, cache_glyph_order);
2872size_t update_approximate_cache_glyph_order(
const CACHE_GLYPH_ORDER* cache_glyph,
2873 const UINT16* flags)
2875 WINPR_ASSERT(cache_glyph);
2876 WINPR_UNUSED(flags);
2877 return 2 + cache_glyph->cGlyphs * 32;
2883 size_t inf = update_approximate_cache_glyph_order(cache_glyph, flags);
2885 if (!Stream_EnsureRemainingCapacity(s, inf))
2888 Stream_Write_UINT8(s, get_checked_uint8(cache_glyph->cacheId));
2889 Stream_Write_UINT8(s, get_checked_uint8(cache_glyph->cGlyphs));
2891 for (UINT32 i = 0; i < cache_glyph->cGlyphs; i++)
2894 glyph = &cache_glyph->glyphData[i];
2895 Stream_Write_UINT16(s, get_checked_uint16(glyph->cacheIndex));
2896 Stream_Write_INT16(s, glyph->x);
2897 Stream_Write_INT16(s, glyph->y);
2898 Stream_Write_UINT16(s, get_checked_uint16(glyph->cx));
2899 Stream_Write_UINT16(s, get_checked_uint16(glyph->cy));
2900 cb = ((glyph->cx + 7) / 8) * glyph->cy;
2901 cb += ((cb % 4) > 0) ? 4 - (cb % 4) : 0;
2902 Stream_Write(s, glyph->aj, cb);
2905 if (*flags & CG_GLYPH_UNICODE_PRESENT)
2907 Stream_Zero(s, 2ULL * cache_glyph->cGlyphs);
2918 if (!cache_glyph_v2)
2921 cache_glyph_v2->cacheId = (flags & 0x000F);
2922 cache_glyph_v2->flags = (flags & 0x00F0) >> 4;
2923 cache_glyph_v2->cGlyphs = (flags & 0xFF00) >> 8;
2925 for (UINT32 i = 0; i < cache_glyph_v2->cGlyphs; i++)
2929 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
2932 Stream_Read_UINT8(s, glyph->cacheIndex);
2934 if (!update_read_2byte_signed(s, &glyph->x) || !update_read_2byte_signed(s, &glyph->y) ||
2935 !update_read_2byte_unsigned(s, &glyph->cx) ||
2936 !update_read_2byte_unsigned(s, &glyph->cy))
2941 glyph->cb = ((glyph->cx + 7) / 8) * glyph->cy;
2942 glyph->cb += ((glyph->cb % 4) > 0) ? 4 - (glyph->cb % 4) : 0;
2944 if (!Stream_CheckAndLogRequiredLength(TAG, s, glyph->cb))
2947 glyph->aj = (BYTE*)malloc(glyph->cb);
2952 Stream_Read(s, glyph->aj, glyph->cb);
2955 if ((flags & CG_GLYPH_UNICODE_PRESENT) && (cache_glyph_v2->cGlyphs > 0))
2957 cache_glyph_v2->unicodeCharacters = calloc(cache_glyph_v2->cGlyphs,
sizeof(WCHAR));
2959 if (!cache_glyph_v2->unicodeCharacters)
2962 if (!Stream_CheckAndLogRequiredLengthOfSize(TAG, s, cache_glyph_v2->cGlyphs,
sizeof(WCHAR)))
2965 if (!Stream_Read_UTF16_String(s, cache_glyph_v2->unicodeCharacters,
2966 cache_glyph_v2->cGlyphs))
2970 return cache_glyph_v2;
2972 free_cache_glyph_v2_order(update->context, cache_glyph_v2);
2977 const UINT16* flags)
2979 WINPR_ASSERT(cache_glyph_v2);
2980 WINPR_UNUSED(flags);
2981 return 8 + cache_glyph_v2->cGlyphs * 32;
2987 size_t inf = update_approximate_cache_glyph_v2_order(cache_glyph_v2, flags);
2989 if (!Stream_EnsureRemainingCapacity(s, inf))
2992 WINPR_ASSERT(cache_glyph_v2->cacheId <= 0x0F);
2993 WINPR_ASSERT(cache_glyph_v2->flags <= 0x0F);
2994 WINPR_ASSERT(cache_glyph_v2->cGlyphs <= 0xFF);
2995 *flags = (UINT16)((cache_glyph_v2->cacheId & 0x000F) | ((cache_glyph_v2->flags & 0x000F) << 4) |
2996 ((cache_glyph_v2->cGlyphs & 0x00FF) << 8));
2998 for (UINT32 i = 0; i < cache_glyph_v2->cGlyphs; i++)
3002 Stream_Write_UINT8(s, get_checked_uint8(glyph->cacheIndex));
3004 if (!update_write_2byte_signed(s, glyph->x) || !update_write_2byte_signed(s, glyph->y) ||
3005 !update_write_2byte_unsigned(s, glyph->cx) ||
3006 !update_write_2byte_unsigned(s, glyph->cy))
3011 cb = ((glyph->cx + 7) / 8) * glyph->cy;
3012 cb += ((cb % 4) > 0) ? 4 - (cb % 4) : 0;
3013 Stream_Write(s, glyph->aj, cb);
3016 if (*flags & CG_GLYPH_UNICODE_PRESENT)
3018 Stream_Zero(s, 2ULL * cache_glyph_v2->cGlyphs);
3023static BOOL update_decompress_brush(
wStream* s, BYTE* output,
size_t outSize, BYTE bpp)
3026 const BYTE* palette = Stream_PointerAs(s,
const BYTE) + 16;
3027 const size_t bytesPerPixel = ((bpp + 1) / 8);
3029 if (!Stream_CheckAndLogRequiredLengthOfSize(TAG, s, 4ULL + bytesPerPixel, 4ULL))
3032 for (
size_t y = 0; y < 8; y++)
3034 for (
size_t x = 0; x < 8; x++)
3037 Stream_Read_UINT8(s,
byte);
3039 const uint32_t index = ((
byte >> ((3 - (x % 4)) * 2)) & 0x03);
3041 for (
size_t k = 0; k < bytesPerPixel; k++)
3043 const size_t dstIndex = ((8ULL * (7ULL - y) + x) * bytesPerPixel) + k;
3044 const size_t srcIndex = (index * bytesPerPixel) + k;
3045 if (dstIndex >= outSize)
3047 output[dstIndex] = palette[srcIndex];
3054static BOOL update_compress_brush(WINPR_ATTR_UNUSED
wStream* s, WINPR_ATTR_UNUSED
const BYTE* input,
3055 WINPR_ATTR_UNUSED BYTE bpp)
3060 WINPR_ATTR_UNUSED UINT16 flags)
3063 BYTE iBitmapFormat = 0;
3064 BOOL compressed = FALSE;
3071 if (!Stream_CheckAndLogRequiredLength(TAG, s, 6))
3074 Stream_Read_UINT8(s, cache_brush->index);
3075 Stream_Read_UINT8(s, iBitmapFormat);
3077 cache_brush->bpp = get_bmf_bpp(iBitmapFormat, &rc);
3081 Stream_Read_UINT8(s, cache_brush->cx);
3082 Stream_Read_UINT8(s, cache_brush->cy);
3085 Stream_Read_UINT8(s, cache_brush->style);
3086 Stream_Read_UINT8(s, cache_brush->length);
3088 if ((cache_brush->cx == 8) && (cache_brush->cy == 8))
3090 if (cache_brush->bpp == 1)
3092 if (cache_brush->length != 8)
3094 WLog_Print(up->log, WLOG_ERROR,
"incompatible 1bpp brush of length:%" PRIu32
"",
3095 cache_brush->length);
3099 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
3103 for (
int i = 7; i >= 0; i--)
3104 Stream_Read_UINT8(s, cache_brush->data[i]);
3108 if ((iBitmapFormat == BMF_8BPP) && (cache_brush->length == 20))
3110 else if ((iBitmapFormat == BMF_16BPP) && (cache_brush->length == 24))
3112 else if ((iBitmapFormat == BMF_24BPP) && (cache_brush->length == 28))
3114 else if ((iBitmapFormat == BMF_32BPP) && (cache_brush->length == 32))
3117 if (compressed != FALSE)
3120 if (!update_decompress_brush(s, cache_brush->data,
sizeof(cache_brush->data),
3121 get_checked_uint8(cache_brush->bpp)))
3127 UINT32 scanline = (cache_brush->bpp / 8) * 8;
3129 if (!Stream_CheckAndLogRequiredLengthOfSize(TAG, s, scanline, 8ull))
3132 for (
int i = 7; i >= 0; i--)
3134 Stream_Read(s, &cache_brush->data[1LL * i * scanline], scanline);
3142 free_cache_brush_order(update->context, cache_brush);
3146size_t update_approximate_cache_brush_order(
const CACHE_BRUSH_ORDER* cache_brush,
3147 const UINT16* flags)
3149 WINPR_UNUSED(cache_brush);
3150 WINPR_UNUSED(flags);
3157 BYTE iBitmapFormat = 0;
3159 BOOL compressed = FALSE;
3161 if (!Stream_EnsureRemainingCapacity(s,
3162 update_approximate_cache_brush_order(cache_brush, flags)))
3165 iBitmapFormat = get_bpp_bmf(cache_brush->bpp, &rc);
3168 Stream_Write_UINT8(s, get_checked_uint8(cache_brush->index));
3169 Stream_Write_UINT8(s, iBitmapFormat);
3170 Stream_Write_UINT8(s, get_checked_uint8(cache_brush->cx));
3171 Stream_Write_UINT8(s, get_checked_uint8(cache_brush->cy));
3172 Stream_Write_UINT8(s, get_checked_uint8(cache_brush->style));
3173 Stream_Write_UINT8(s, get_checked_uint8(cache_brush->length));
3175 if ((cache_brush->cx == 8) && (cache_brush->cy == 8))
3177 if (cache_brush->bpp == 1)
3179 if (cache_brush->length != 8)
3181 WLog_ERR(TAG,
"incompatible 1bpp brush of length:%" PRIu32
"", cache_brush->length);
3185 for (
int i = 7; i >= 0; i--)
3187 Stream_Write_UINT8(s, cache_brush->data[i]);
3192 if ((iBitmapFormat == BMF_8BPP) && (cache_brush->length == 20))
3194 else if ((iBitmapFormat == BMF_16BPP) && (cache_brush->length == 24))
3196 else if ((iBitmapFormat == BMF_32BPP) && (cache_brush->length == 32))
3199 if (compressed != FALSE)
3202 if (!update_compress_brush(s, cache_brush->data,
3203 get_checked_uint8(cache_brush->bpp)))
3209 const size_t scanline = 8ULL * (cache_brush->bpp / 8);
3211 for (
size_t i = 0; i <= 7; i++)
3213 Stream_Write(s, &cache_brush->data[1LL * (7 - i) * scanline], scanline);
3223update_read_create_offscreen_bitmap_order(
wStream* s,
3227 BOOL deleteListPresent = 0;
3230 if (!Stream_CheckAndLogRequiredLength(TAG, s, 6))
3233 Stream_Read_UINT16(s, flags);
3234 create_offscreen_bitmap->id = flags & 0x7FFF;
3235 deleteListPresent = (flags & 0x8000) != 0;
3236 Stream_Read_UINT16(s, create_offscreen_bitmap->cx);
3237 Stream_Read_UINT16(s, create_offscreen_bitmap->cy);
3238 deleteList = &(create_offscreen_bitmap->deleteList);
3240 if ((create_offscreen_bitmap->cx == 0) || (create_offscreen_bitmap->cy == 0))
3242 WLog_ERR(TAG,
"Invalid OFFSCREEN_DELETE_LIST: cx=%" PRIu16
", cy=%" PRIu16,
3243 create_offscreen_bitmap->cx, create_offscreen_bitmap->cy);
3247 if (deleteListPresent)
3249 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
3252 Stream_Read_UINT16(s, deleteList->cIndices);
3254 if (deleteList->cIndices > deleteList->sIndices)
3256 UINT16* new_indices =
nullptr;
3257 new_indices = (UINT16*)realloc(deleteList->indices, 2ULL * deleteList->cIndices);
3262 deleteList->sIndices = deleteList->cIndices;
3263 deleteList->indices = new_indices;
3266 if (!Stream_CheckAndLogRequiredLengthOfSize(TAG, s, deleteList->cIndices, 2ull))
3269 for (UINT32 i = 0; i < deleteList->cIndices; i++)
3271 Stream_Read_UINT16(s, deleteList->indices[i]);
3276 deleteList->cIndices = 0;
3282size_t update_approximate_create_offscreen_bitmap_order(
3287 WINPR_ASSERT(create_offscreen_bitmap);
3289 deleteList = &(create_offscreen_bitmap->deleteList);
3290 WINPR_ASSERT(deleteList);
3292 return 32ull + deleteList->cIndices * 2ull;
3295BOOL update_write_create_offscreen_bitmap_order(
3299 BOOL deleteListPresent = 0;
3302 if (!Stream_EnsureRemainingCapacity(
3303 s, update_approximate_create_offscreen_bitmap_order(create_offscreen_bitmap)))
3306 deleteList = &(create_offscreen_bitmap->deleteList);
3307 flags = create_offscreen_bitmap->id & 0x7FFF;
3308 deleteListPresent = (deleteList->cIndices > 0);
3310 if (deleteListPresent)
3313 Stream_Write_UINT16(s, flags);
3314 Stream_Write_UINT16(s, get_checked_uint16(create_offscreen_bitmap->cx));
3315 Stream_Write_UINT16(s, get_checked_uint16(create_offscreen_bitmap->cy));
3317 if (deleteListPresent)
3319 Stream_Write_UINT16(s, get_checked_uint16(deleteList->cIndices));
3321 for (
size_t i = 0; i < deleteList->cIndices; i++)
3323 Stream_Write_UINT16(s, deleteList->indices[i]);
3331 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
3334 Stream_Read_UINT16(s, switch_surface->bitmapId);
3337size_t update_approximate_switch_surface_order(
3344 size_t inf = update_approximate_switch_surface_order(switch_surface);
3346 if (!Stream_EnsureRemainingCapacity(s, inf))
3349 WINPR_ASSERT(switch_surface->bitmapId <= UINT16_MAX);
3350 Stream_Write_UINT16(s, (UINT16)switch_surface->bitmapId);
3354update_read_create_nine_grid_bitmap_order(
wStream* s,
3359 if (!Stream_CheckAndLogRequiredLength(TAG, s, 19))
3362 Stream_Read_UINT8(s, create_nine_grid_bitmap->bitmapBpp);
3364 if ((create_nine_grid_bitmap->bitmapBpp < 1) || (create_nine_grid_bitmap->bitmapBpp > 32))
3366 WLog_ERR(TAG,
"invalid bpp value %" PRIu32
"", create_nine_grid_bitmap->bitmapBpp);
3370 Stream_Read_UINT16(s, create_nine_grid_bitmap->bitmapId);
3371 nineGridInfo = &(create_nine_grid_bitmap->nineGridInfo);
3372 Stream_Read_UINT32(s, nineGridInfo->flFlags);
3373 Stream_Read_UINT16(s, nineGridInfo->ulLeftWidth);
3374 Stream_Read_UINT16(s, nineGridInfo->ulRightWidth);
3375 Stream_Read_UINT16(s, nineGridInfo->ulTopHeight);
3376 Stream_Read_UINT16(s, nineGridInfo->ulBottomHeight);
3377 update_read_colorref(s, &nineGridInfo->crTransparent);
3382 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
3385 Stream_Read_UINT32(s, frame_marker->action);
3388static BOOL update_read_stream_bitmap_first_order(
wStream* s,
3391 if (!Stream_CheckAndLogRequiredLength(TAG, s, 10))
3394 Stream_Read_UINT8(s, stream_bitmap_first->bitmapFlags);
3395 Stream_Read_UINT8(s, stream_bitmap_first->bitmapBpp);
3397 if ((stream_bitmap_first->bitmapBpp < 1) || (stream_bitmap_first->bitmapBpp > 32))
3399 WLog_ERR(TAG,
"invalid bpp value %" PRIu32
"", stream_bitmap_first->bitmapBpp);
3403 Stream_Read_UINT16(s, stream_bitmap_first->bitmapType);
3404 Stream_Read_UINT16(s, stream_bitmap_first->bitmapWidth);
3405 Stream_Read_UINT16(s, stream_bitmap_first->bitmapHeight);
3407 if (stream_bitmap_first->bitmapFlags & STREAM_BITMAP_V2)
3409 if (!Stream_CheckAndLogRequiredLength(TAG, s, 4))
3412 Stream_Read_UINT32(s, stream_bitmap_first->bitmapSize);
3416 if (!Stream_CheckAndLogRequiredLength(TAG, s, 2))
3419 Stream_Read_UINT16(s, stream_bitmap_first->bitmapSize);
3422 FIELD_SKIP_BUFFER16(
3423 s, stream_bitmap_first->bitmapBlockSize);
3426static BOOL update_read_stream_bitmap_next_order(
wStream* s,
3429 if (!Stream_CheckAndLogRequiredLength(TAG, s, 5))
3432 Stream_Read_UINT8(s, stream_bitmap_next->bitmapFlags);
3433 Stream_Read_UINT16(s, stream_bitmap_next->bitmapType);
3434 FIELD_SKIP_BUFFER16(
3435 s, stream_bitmap_next->bitmapBlockSize);
3438static BOOL update_read_draw_gdiplus_first_order(
wStream* s,
3441 if (!Stream_CheckAndLogRequiredLength(TAG, s, 11))
3444 Stream_Seek_UINT8(s);
3445 Stream_Read_UINT16(s, draw_gdiplus_first->cbSize);
3446 Stream_Read_UINT32(s, draw_gdiplus_first->cbTotalSize);
3447 Stream_Read_UINT32(s, draw_gdiplus_first->cbTotalEmfSize);
3448 return Stream_SafeSeek(s, draw_gdiplus_first->cbSize);
3450static BOOL update_read_draw_gdiplus_next_order(
wStream* s,
3453 if (!Stream_CheckAndLogRequiredLength(TAG, s, 3))
3456 Stream_Seek_UINT8(s);
3457 FIELD_SKIP_BUFFER16(s, draw_gdiplus_next->cbSize);
3462 if (!Stream_CheckAndLogRequiredLength(TAG, s, 11))
3465 Stream_Seek_UINT8(s);
3466 Stream_Read_UINT16(s, draw_gdiplus_end->cbSize);
3467 Stream_Read_UINT32(s, draw_gdiplus_end->cbTotalSize);
3468 Stream_Read_UINT32(s, draw_gdiplus_end->cbTotalEmfSize);
3469 return Stream_SafeSeek(s, draw_gdiplus_end->cbSize);
3472update_read_draw_gdiplus_cache_first_order(
wStream* s,
3475 if (!Stream_CheckAndLogRequiredLength(TAG, s, 11))
3478 Stream_Read_UINT8(s, draw_gdiplus_cache_first->flags);
3479 Stream_Read_UINT16(s, draw_gdiplus_cache_first->cacheType);
3480 Stream_Read_UINT16(s, draw_gdiplus_cache_first->cacheIndex);
3481 Stream_Read_UINT16(s, draw_gdiplus_cache_first->cbSize);
3482 Stream_Read_UINT32(s, draw_gdiplus_cache_first->cbTotalSize);
3483 return Stream_SafeSeek(s, draw_gdiplus_cache_first->cbSize);
3486update_read_draw_gdiplus_cache_next_order(
wStream* s,
3489 if (!Stream_CheckAndLogRequiredLength(TAG, s, 7))
3492 Stream_Read_UINT8(s, draw_gdiplus_cache_next->flags);
3493 Stream_Read_UINT16(s, draw_gdiplus_cache_next->cacheType);
3494 Stream_Read_UINT16(s, draw_gdiplus_cache_next->cacheIndex);
3495 FIELD_SKIP_BUFFER16(s, draw_gdiplus_cache_next->cbSize);
3499update_read_draw_gdiplus_cache_end_order(
wStream* s,
3502 if (!Stream_CheckAndLogRequiredLength(TAG, s, 11))
3505 Stream_Read_UINT8(s, draw_gdiplus_cache_end->flags);
3506 Stream_Read_UINT16(s, draw_gdiplus_cache_end->cacheType);
3507 Stream_Read_UINT16(s, draw_gdiplus_cache_end->cacheIndex);
3508 Stream_Read_UINT16(s, draw_gdiplus_cache_end->cbSize);
3509 Stream_Read_UINT32(s, draw_gdiplus_cache_end->cbTotalSize);
3510 return Stream_SafeSeek(s, draw_gdiplus_cache_end->cbSize);
3512static BOOL update_read_field_flags(
wStream* s, UINT32* fieldFlags, BYTE flags, BYTE fieldBytes)
3514 if (flags & ORDER_ZERO_FIELD_BYTE_BIT0)
3517 if (flags & ORDER_ZERO_FIELD_BYTE_BIT1)
3525 if (!Stream_CheckAndLogRequiredLength(TAG, s, fieldBytes))
3530 for (
size_t i = 0; i < fieldBytes; i++)
3532 const UINT32
byte = Stream_Get_UINT8(s);
3533 *fieldFlags |= (
byte << (i * 8ULL)) & 0xFFFFFFFF;
3538BOOL update_write_field_flags(
wStream* s, UINT32 fieldFlags, WINPR_ATTR_UNUSED BYTE flags,
3543 if (fieldBytes == 1)
3545 byte = fieldFlags & 0xFF;
3546 Stream_Write_UINT8(s,
byte);
3548 else if (fieldBytes == 2)
3550 byte = fieldFlags & 0xFF;
3551 Stream_Write_UINT8(s,
byte);
3552 byte = (fieldFlags >> 8) & 0xFF;
3553 Stream_Write_UINT8(s,
byte);
3555 else if (fieldBytes == 3)
3557 byte = fieldFlags & 0xFF;
3558 Stream_Write_UINT8(s,
byte);
3559 byte = (fieldFlags >> 8) & 0xFF;
3560 Stream_Write_UINT8(s,
byte);
3561 byte = (fieldFlags >> 16) & 0xFF;
3562 Stream_Write_UINT8(s,
byte);
3571static BOOL update_read_bounds(
wStream* s, rdpBounds* bounds)
3575 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
3578 Stream_Read_UINT8(s, flags);
3580 if (flags & BOUND_LEFT)
3582 if (!update_read_coord(s, &bounds->left, FALSE))
3585 else if (flags & BOUND_DELTA_LEFT)
3587 if (!update_read_coord(s, &bounds->left, TRUE))
3591 if (flags & BOUND_TOP)
3593 if (!update_read_coord(s, &bounds->top, FALSE))
3596 else if (flags & BOUND_DELTA_TOP)
3598 if (!update_read_coord(s, &bounds->top, TRUE))
3602 if (flags & BOUND_RIGHT)
3604 if (!update_read_coord(s, &bounds->right, FALSE))
3607 else if (flags & BOUND_DELTA_RIGHT)
3609 if (!update_read_coord(s, &bounds->right, TRUE))
3613 if (flags & BOUND_BOTTOM)
3615 if (!update_read_coord(s, &bounds->bottom, FALSE))
3618 else if (flags & BOUND_DELTA_BOTTOM)
3620 if (!update_read_coord(s, &bounds->bottom, TRUE))
3628 WINPR_ASSERT(orderInfo);
3630 if (!(orderInfo->controlFlags & ORDER_BOUNDS))
3633 if (orderInfo->controlFlags & ORDER_ZERO_BOUNDS_DELTAS)
3636 Stream_Write_UINT8(s, get_checked_uint8(orderInfo->boundsFlags));
3638 if (orderInfo->boundsFlags & BOUND_LEFT)
3640 if (!update_write_coord(s, orderInfo->bounds.left))
3643 else if (orderInfo->boundsFlags & BOUND_DELTA_LEFT)
3647 if (orderInfo->boundsFlags & BOUND_TOP)
3649 if (!update_write_coord(s, orderInfo->bounds.top))
3652 else if (orderInfo->boundsFlags & BOUND_DELTA_TOP)
3656 if (orderInfo->boundsFlags & BOUND_RIGHT)
3658 if (!update_write_coord(s, orderInfo->bounds.right))
3661 else if (orderInfo->boundsFlags & BOUND_DELTA_RIGHT)
3665 if (orderInfo->boundsFlags & BOUND_BOTTOM)
3667 if (!update_write_coord(s, orderInfo->bounds.bottom))
3670 else if (orderInfo->boundsFlags & BOUND_DELTA_BOTTOM)
3677static BOOL read_primary_order(wLog* log,
const char* orderName,
wStream* s,
3678 const ORDER_INFO* orderInfo, rdpPrimaryUpdate* primary_pub)
3683 if (!s || !orderInfo || !orderName)
3686 switch (orderInfo->orderType)
3688 case ORDER_TYPE_DSTBLT:
3689 rc = update_read_dstblt_order(orderName, s, orderInfo, &(primary->dstblt));
3692 case ORDER_TYPE_PATBLT:
3693 rc = update_read_patblt_order(orderName, s, orderInfo, &(primary->patblt));
3696 case ORDER_TYPE_SCRBLT:
3697 rc = update_read_scrblt_order(orderName, s, orderInfo, &(primary->scrblt));
3700 case ORDER_TYPE_OPAQUE_RECT:
3701 rc = update_read_opaque_rect_order(orderName, s, orderInfo, &(primary->opaque_rect));
3704 case ORDER_TYPE_DRAW_NINE_GRID:
3705 rc = update_read_draw_nine_grid_order(orderName, s, orderInfo,
3706 &(primary->draw_nine_grid));
3709 case ORDER_TYPE_MULTI_DSTBLT:
3710 rc = update_read_multi_dstblt_order(orderName, s, orderInfo, &(primary->multi_dstblt));
3713 case ORDER_TYPE_MULTI_PATBLT:
3714 rc = update_read_multi_patblt_order(orderName, s, orderInfo, &(primary->multi_patblt));
3717 case ORDER_TYPE_MULTI_SCRBLT:
3718 rc = update_read_multi_scrblt_order(orderName, s, orderInfo, &(primary->multi_scrblt));
3721 case ORDER_TYPE_MULTI_OPAQUE_RECT:
3722 rc = update_read_multi_opaque_rect_order(orderName, s, orderInfo,
3723 &(primary->multi_opaque_rect));
3726 case ORDER_TYPE_MULTI_DRAW_NINE_GRID:
3727 rc = update_read_multi_draw_nine_grid_order(orderName, s, orderInfo,
3728 &(primary->multi_draw_nine_grid));
3731 case ORDER_TYPE_LINE_TO:
3732 rc = update_read_line_to_order(orderName, s, orderInfo, &(primary->line_to));
3735 case ORDER_TYPE_POLYLINE:
3736 rc = update_read_polyline_order(orderName, s, orderInfo, &(primary->polyline));
3739 case ORDER_TYPE_MEMBLT:
3740 rc = update_read_memblt_order(orderName, s, orderInfo, &(primary->memblt));
3743 case ORDER_TYPE_MEM3BLT:
3744 rc = update_read_mem3blt_order(orderName, s, orderInfo, &(primary->mem3blt));
3747 case ORDER_TYPE_SAVE_BITMAP:
3748 rc = update_read_save_bitmap_order(orderName, s, orderInfo, &(primary->save_bitmap));
3751 case ORDER_TYPE_GLYPH_INDEX:
3752 rc = update_read_glyph_index_order(orderName, s, orderInfo, &(primary->glyph_index));
3755 case ORDER_TYPE_FAST_INDEX:
3756 rc = update_read_fast_index_order(orderName, s, orderInfo, &(primary->fast_index));
3759 case ORDER_TYPE_FAST_GLYPH:
3760 rc = update_read_fast_glyph_order(orderName, s, orderInfo, &(primary->fast_glyph));
3763 case ORDER_TYPE_POLYGON_SC:
3764 rc = update_read_polygon_sc_order(orderName, s, orderInfo, &(primary->polygon_sc));
3767 case ORDER_TYPE_POLYGON_CB:
3768 rc = update_read_polygon_cb_order(orderName, s, orderInfo, &(primary->polygon_cb));
3771 case ORDER_TYPE_ELLIPSE_SC:
3772 rc = update_read_ellipse_sc_order(orderName, s, orderInfo, &(primary->ellipse_sc));
3775 case ORDER_TYPE_ELLIPSE_CB:
3776 rc = update_read_ellipse_cb_order(orderName, s, orderInfo, &(primary->ellipse_cb));
3780 WLog_Print(log, WLOG_WARN,
"%s %s not supported, ignoring", primary_order_str,
3788 WLog_Print(log, WLOG_ERROR,
"%s %s failed", primary_order_str, orderName);
3795static BOOL update_recv_primary_order(rdpUpdate* update,
wStream* s, BYTE flags)
3800 rdpContext* context = update->context;
3805 ORDER_INFO* orderInfo = &(primary->order_info);
3806 WINPR_ASSERT(orderInfo);
3807 WINPR_ASSERT(context);
3809 const rdpSettings* settings = context->settings;
3810 WINPR_ASSERT(settings);
3812 const BOOL defaultReturn =
3815 if (flags & ORDER_TYPE_CHANGE)
3817 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
3820 Stream_Read_UINT8(s, orderInfo->orderType);
3823 const char* orderName = primary_order_string(orderInfo->orderType);
3824 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
3826 if (!check_primary_order_supported(up->log, settings, orderInfo->orderType, orderName))
3829 field = get_primary_drawing_order_field_bytes(orderInfo->orderType, &rc);
3833 if (!update_read_field_flags(s, &(orderInfo->fieldFlags), flags, field))
3835 WLog_Print(up->log, WLOG_ERROR,
"update_read_field_flags() failed");
3839 if (flags & ORDER_BOUNDS)
3841 if (!(flags & ORDER_ZERO_BOUNDS_DELTAS))
3843 if (!update_read_bounds(s, &orderInfo->bounds))
3845 WLog_Print(up->log, WLOG_ERROR,
"update_read_bounds() failed");
3850 rc = IFCALLRESULT(defaultReturn, update->SetBounds, context, &orderInfo->bounds);
3856 orderInfo->deltaCoordinates = (flags & ORDER_DELTA_COORDINATES) != 0;
3858 if (!read_primary_order(up->log, orderName, s, orderInfo, &primary->common))
3861 rc = IFCALLRESULT(TRUE, primary->common.OrderInfo, context, orderInfo, orderName);
3865 switch (orderInfo->orderType)
3867 case ORDER_TYPE_DSTBLT:
3869 WLog_Print(up->log, WLOG_DEBUG,
"%s %s rop=%s [0x%08" PRIx32
"]", primary_order_str,
3870 orderName, gdi_rob3_code_string_checked(primary->dstblt.bRop),
3871 gdi_rop3_code_checked(primary->dstblt.bRop));
3872 rc = IFCALLRESULT(defaultReturn, primary->common.DstBlt, context, &primary->dstblt);
3876 case ORDER_TYPE_PATBLT:
3878 WINPR_ASSERT(primary->patblt.bRop <= UINT8_MAX);
3879 WLog_Print(up->log, WLOG_DEBUG,
"%s %s rop=%s [0x%08" PRIx32
"]", primary_order_str,
3880 orderName, gdi_rob3_code_string_checked(primary->patblt.bRop),
3881 gdi_rop3_code_checked(primary->patblt.bRop));
3882 rc = IFCALLRESULT(defaultReturn, primary->common.PatBlt, context, &primary->patblt);
3886 case ORDER_TYPE_SCRBLT:
3888 WINPR_ASSERT(primary->scrblt.bRop <= UINT8_MAX);
3889 WLog_Print(up->log, WLOG_DEBUG,
"%s %s rop=%s [0x%08" PRIx32
"]", primary_order_str,
3890 orderName, gdi_rob3_code_string_checked((UINT8)primary->scrblt.bRop),
3891 gdi_rop3_code_checked(primary->scrblt.bRop));
3892 rc = IFCALLRESULT(defaultReturn, primary->common.ScrBlt, context, &primary->scrblt);
3896 case ORDER_TYPE_OPAQUE_RECT:
3898 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
3899 rc = IFCALLRESULT(defaultReturn, primary->common.OpaqueRect, context,
3900 &primary->opaque_rect);
3904 case ORDER_TYPE_DRAW_NINE_GRID:
3906 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
3907 rc = IFCALLRESULT(defaultReturn, primary->common.DrawNineGrid, context,
3908 &primary->draw_nine_grid);
3912 case ORDER_TYPE_MULTI_DSTBLT:
3914 WLog_Print(up->log, WLOG_DEBUG,
"%s %s rop=%s [0x%08" PRIx32
"]", primary_order_str,
3915 orderName, gdi_rob3_code_string_checked(primary->multi_dstblt.bRop),
3916 gdi_rop3_code_checked(primary->multi_dstblt.bRop));
3917 rc = IFCALLRESULT(defaultReturn, primary->common.MultiDstBlt, context,
3918 &primary->multi_dstblt);
3922 case ORDER_TYPE_MULTI_PATBLT:
3924 WLog_Print(up->log, WLOG_DEBUG,
"%s %s rop=%s [0x%08" PRIx32
"]", primary_order_str,
3925 orderName, gdi_rob3_code_string_checked(primary->multi_patblt.bRop),
3926 gdi_rop3_code_checked(primary->multi_patblt.bRop));
3927 rc = IFCALLRESULT(defaultReturn, primary->common.MultiPatBlt, context,
3928 &primary->multi_patblt);
3932 case ORDER_TYPE_MULTI_SCRBLT:
3934 WLog_Print(up->log, WLOG_DEBUG,
"%s %s rop=%s [0x%08" PRIx32
"]", primary_order_str,
3935 orderName, gdi_rob3_code_string_checked(primary->multi_scrblt.bRop),
3936 gdi_rop3_code_checked(primary->multi_scrblt.bRop));
3937 rc = IFCALLRESULT(defaultReturn, primary->common.MultiScrBlt, context,
3938 &primary->multi_scrblt);
3942 case ORDER_TYPE_MULTI_OPAQUE_RECT:
3944 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
3945 rc = IFCALLRESULT(defaultReturn, primary->common.MultiOpaqueRect, context,
3946 &primary->multi_opaque_rect);
3950 case ORDER_TYPE_MULTI_DRAW_NINE_GRID:
3952 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
3953 rc = IFCALLRESULT(defaultReturn, primary->common.MultiDrawNineGrid, context,
3954 &primary->multi_draw_nine_grid);
3958 case ORDER_TYPE_LINE_TO:
3960 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
3961 rc = IFCALLRESULT(defaultReturn, primary->common.LineTo, context, &primary->line_to);
3965 case ORDER_TYPE_POLYLINE:
3967 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
3968 rc = IFCALLRESULT(defaultReturn, primary->common.Polyline, context, &primary->polyline);
3972 case ORDER_TYPE_MEMBLT:
3974 WINPR_ASSERT(primary->memblt.bRop <= UINT8_MAX);
3975 WLog_Print(up->log, WLOG_DEBUG,
"%s %s rop=%s [0x%08" PRIx32
"]", primary_order_str,
3976 orderName, gdi_rob3_code_string_checked(primary->memblt.bRop),
3977 gdi_rop3_code_checked(primary->memblt.bRop));
3978 rc = IFCALLRESULT(defaultReturn, primary->common.MemBlt, context, &primary->memblt);
3982 case ORDER_TYPE_MEM3BLT:
3984 WINPR_ASSERT(primary->mem3blt.bRop <= UINT8_MAX);
3985 WLog_Print(up->log, WLOG_DEBUG,
"%s %s rop=%s [0x%08" PRIx32
"]", primary_order_str,
3986 orderName, gdi_rob3_code_string_checked(primary->mem3blt.bRop),
3987 gdi_rop3_code_checked(primary->mem3blt.bRop));
3988 rc = IFCALLRESULT(defaultReturn, primary->common.Mem3Blt, context, &primary->mem3blt);
3992 case ORDER_TYPE_SAVE_BITMAP:
3994 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
3995 rc = IFCALLRESULT(defaultReturn, primary->common.SaveBitmap, context,
3996 &primary->save_bitmap);
4000 case ORDER_TYPE_GLYPH_INDEX:
4002 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
4003 rc = IFCALLRESULT(defaultReturn, primary->common.GlyphIndex, context,
4004 &primary->glyph_index);
4008 case ORDER_TYPE_FAST_INDEX:
4010 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
4011 rc = IFCALLRESULT(defaultReturn, primary->common.FastIndex, context,
4012 &primary->fast_index);
4016 case ORDER_TYPE_FAST_GLYPH:
4018 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
4019 rc = IFCALLRESULT(defaultReturn, primary->common.FastGlyph, context,
4020 &primary->fast_glyph);
4024 case ORDER_TYPE_POLYGON_SC:
4026 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
4027 rc = IFCALLRESULT(defaultReturn, primary->common.PolygonSC, context,
4028 &primary->polygon_sc);
4032 case ORDER_TYPE_POLYGON_CB:
4034 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
4035 rc = IFCALLRESULT(defaultReturn, primary->common.PolygonCB, context,
4036 &primary->polygon_cb);
4040 case ORDER_TYPE_ELLIPSE_SC:
4042 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
4043 rc = IFCALLRESULT(defaultReturn, primary->common.EllipseSC, context,
4044 &primary->ellipse_sc);
4048 case ORDER_TYPE_ELLIPSE_CB:
4050 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", primary_order_str, orderName);
4051 rc = IFCALLRESULT(defaultReturn, primary->common.EllipseCB, context,
4052 &primary->ellipse_cb);
4057 WLog_Print(up->log, WLOG_WARN,
"%s %s not supported", primary_order_str, orderName);
4063 WLog_Print(up->log, WLOG_ERROR,
"%s %s failed", primary_order_str, orderName);
4067 if (flags & ORDER_BOUNDS)
4069 rc = IFCALLRESULT(defaultReturn, update->SetBounds, context,
nullptr);
4075static BOOL update_recv_secondary_order(rdpUpdate* update,
wStream* s, WINPR_ATTR_UNUSED BYTE flags)
4083 UINT16 extraFlags = 0;
4084 INT16 orderLength = 0;
4086 rdpContext* context = update->context;
4087 rdpSettings* settings = context->settings;
4088 rdpSecondaryUpdate* secondary = update->secondary;
4089 const char* name =
nullptr;
4090 BOOL defaultReturn = 0;
4094 if (!Stream_CheckAndLogRequiredLength(TAG, s, 5))
4097 Stream_Read_INT16(s, orderLength);
4098 Stream_Read_UINT16(s, extraFlags);
4099 Stream_Read_UINT8(s, orderType);
4101 start = Stream_GetPosition(s);
4102 name = secondary_order_string(orderType);
4103 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", secondary_order_str, name);
4104 rc = IFCALLRESULT(TRUE, secondary->CacheOrderInfo, context, orderLength, extraFlags, orderType,
4117 if (orderLength < 0)
4119 WLog_Print(up->log, WLOG_ERROR,
"orderLength %" PRIu16
" must be >= 7", orderLength);
4123 const size_t orderLengthFull = WINPR_ASSERTING_INT_CAST(
size_t, orderLength) + 7ULL;
4124 if (!Stream_CheckAndLogRequiredLength(TAG, s, orderLengthFull))
4127 if (!check_secondary_order_supported(up->log, settings, orderType, name))
4132 case ORDER_TYPE_BITMAP_UNCOMPRESSED:
4133 case ORDER_TYPE_CACHE_BITMAP_COMPRESSED:
4135 const BOOL compressed = (orderType == ORDER_TYPE_CACHE_BITMAP_COMPRESSED);
4137 update_read_cache_bitmap_order(update, s, compressed, extraFlags);
4141 rc = IFCALLRESULT(defaultReturn, secondary->CacheBitmap, context, order);
4142 free_cache_bitmap_order(context, order);
4147 case ORDER_TYPE_BITMAP_UNCOMPRESSED_V2:
4148 case ORDER_TYPE_BITMAP_COMPRESSED_V2:
4150 const BOOL compressed = (orderType == ORDER_TYPE_BITMAP_COMPRESSED_V2);
4152 update_read_cache_bitmap_v2_order(update, s, compressed, extraFlags);
4156 rc = IFCALLRESULT(defaultReturn, secondary->CacheBitmapV2, context, order);
4157 free_cache_bitmap_v2_order(context, order);
4162 case ORDER_TYPE_BITMAP_COMPRESSED_V3:
4168 rc = IFCALLRESULT(defaultReturn, secondary->CacheBitmapV3, context, order);
4169 free_cache_bitmap_v3_order(context, order);
4174 case ORDER_TYPE_CACHE_COLOR_TABLE:
4177 update_read_cache_color_table_order(update, s, extraFlags);
4181 rc = IFCALLRESULT(defaultReturn, secondary->CacheColorTable, context, order);
4182 free_cache_color_table_order(context, order);
4187 case ORDER_TYPE_CACHE_GLYPH:
4189 switch (settings->GlyphSupportLevel)
4191 case GLYPH_SUPPORT_PARTIAL:
4192 case GLYPH_SUPPORT_FULL:
4194 CACHE_GLYPH_ORDER* order = update_read_cache_glyph_order(update, s, extraFlags);
4198 rc = IFCALLRESULT(defaultReturn, secondary->CacheGlyph, context, order);
4199 free_cache_glyph_order(context, order);
4204 case GLYPH_SUPPORT_ENCODE:
4207 update_read_cache_glyph_v2_order(update, s, extraFlags);
4211 rc = IFCALLRESULT(defaultReturn, secondary->CacheGlyphV2, context, order);
4212 free_cache_glyph_v2_order(context, order);
4217 case GLYPH_SUPPORT_NONE:
4224 case ORDER_TYPE_CACHE_BRUSH:
4227 CACHE_BRUSH_ORDER* order = update_read_cache_brush_order(update, s, extraFlags);
4231 rc = IFCALLRESULT(defaultReturn, secondary->CacheBrush, context, order);
4232 free_cache_brush_order(context, order);
4238 WLog_Print(up->log, WLOG_WARN,
"%s %s not supported", secondary_order_str, name);
4244 WLog_Print(up->log, WLOG_ERROR,
"%s %s failed", secondary_order_str, name);
4247 end = start + WINPR_ASSERTING_INT_CAST(
size_t, orderLengthFull);
4248 pos = Stream_GetPosition(s);
4251 WLog_Print(up->log, WLOG_WARN,
"%s %s: read %" PRIuz
"bytes too much", secondary_order_str,
4258 WLog_Print(up->log, WLOG_DEBUG,
"%s %s: read %" PRIuz
"bytes short, skipping",
4259 secondary_order_str, name, diff);
4260 if (!Stream_SafeSeek(s, diff))
4266static BOOL read_altsec_order(wLog* log,
wStream* s, BYTE orderType, rdpAltSecUpdate* altsec_pub)
4275 case ORDER_TYPE_CREATE_OFFSCREEN_BITMAP:
4276 rc = update_read_create_offscreen_bitmap_order(s, &(altsec->create_offscreen_bitmap));
4279 case ORDER_TYPE_SWITCH_SURFACE:
4280 rc = update_read_switch_surface_order(s, &(altsec->switch_surface));
4283 case ORDER_TYPE_CREATE_NINE_GRID_BITMAP:
4284 rc = update_read_create_nine_grid_bitmap_order(s, &(altsec->create_nine_grid_bitmap));
4287 case ORDER_TYPE_FRAME_MARKER:
4288 rc = update_read_frame_marker_order(s, &(altsec->frame_marker));
4291 case ORDER_TYPE_STREAM_BITMAP_FIRST:
4292 rc = update_read_stream_bitmap_first_order(s, &(altsec->stream_bitmap_first));
4295 case ORDER_TYPE_STREAM_BITMAP_NEXT:
4296 rc = update_read_stream_bitmap_next_order(s, &(altsec->stream_bitmap_next));
4299 case ORDER_TYPE_GDIPLUS_FIRST:
4300 rc = update_read_draw_gdiplus_first_order(s, &(altsec->draw_gdiplus_first));
4303 case ORDER_TYPE_GDIPLUS_NEXT:
4304 rc = update_read_draw_gdiplus_next_order(s, &(altsec->draw_gdiplus_next));
4307 case ORDER_TYPE_GDIPLUS_END:
4308 rc = update_read_draw_gdiplus_end_order(s, &(altsec->draw_gdiplus_end));
4311 case ORDER_TYPE_GDIPLUS_CACHE_FIRST:
4312 rc = update_read_draw_gdiplus_cache_first_order(s, &(altsec->draw_gdiplus_cache_first));
4315 case ORDER_TYPE_GDIPLUS_CACHE_NEXT:
4316 rc = update_read_draw_gdiplus_cache_next_order(s, &(altsec->draw_gdiplus_cache_next));
4319 case ORDER_TYPE_GDIPLUS_CACHE_END:
4320 rc = update_read_draw_gdiplus_cache_end_order(s, &(altsec->draw_gdiplus_cache_end));
4323 case ORDER_TYPE_WINDOW:
4328 case ORDER_TYPE_COMPDESK_FIRST:
4338 WLog_Print(log, WLOG_ERROR,
"Read %s %s failed", alt_sec_order_str,
4339 altsec_order_string(orderType));
4345static BOOL update_recv_altsec_order(rdpUpdate* update,
wStream* s, BYTE flags)
4347 BYTE orderType = flags >> 2;
4350 rdpContext* context = update->context;
4351 rdpSettings* settings = context->settings;
4353 const char* orderName = altsec_order_string(orderType);
4356 WINPR_ASSERT(context);
4357 WINPR_ASSERT(settings);
4359 WLog_Print(up->log, WLOG_DEBUG,
"%s %s", alt_sec_order_str, orderName);
4361 rc = IFCALLRESULT(TRUE, altsec->common.DrawOrderInfo, context, orderType, orderName);
4365 if (!check_alt_order_supported(up->log, settings, orderType, orderName))
4368 if (!read_altsec_order(up->log, s, orderType, &altsec->common))
4373 case ORDER_TYPE_CREATE_OFFSCREEN_BITMAP:
4374 IFCALLRET(altsec->common.CreateOffscreenBitmap, rc, context,
4375 &(altsec->create_offscreen_bitmap));
4378 case ORDER_TYPE_SWITCH_SURFACE:
4379 IFCALLRET(altsec->common.SwitchSurface, rc, context, &(altsec->switch_surface));
4382 case ORDER_TYPE_CREATE_NINE_GRID_BITMAP:
4383 IFCALLRET(altsec->common.CreateNineGridBitmap, rc, context,
4384 &(altsec->create_nine_grid_bitmap));
4387 case ORDER_TYPE_FRAME_MARKER:
4388 IFCALLRET(altsec->common.FrameMarker, rc, context, &(altsec->frame_marker));
4391 case ORDER_TYPE_STREAM_BITMAP_FIRST:
4392 IFCALLRET(altsec->common.StreamBitmapFirst, rc, context,
4393 &(altsec->stream_bitmap_first));
4396 case ORDER_TYPE_STREAM_BITMAP_NEXT:
4397 IFCALLRET(altsec->common.StreamBitmapNext, rc, context, &(altsec->stream_bitmap_next));
4400 case ORDER_TYPE_GDIPLUS_FIRST:
4401 IFCALLRET(altsec->common.DrawGdiPlusFirst, rc, context, &(altsec->draw_gdiplus_first));
4404 case ORDER_TYPE_GDIPLUS_NEXT:
4405 IFCALLRET(altsec->common.DrawGdiPlusNext, rc, context, &(altsec->draw_gdiplus_next));
4408 case ORDER_TYPE_GDIPLUS_END:
4409 IFCALLRET(altsec->common.DrawGdiPlusEnd, rc, context, &(altsec->draw_gdiplus_end));
4412 case ORDER_TYPE_GDIPLUS_CACHE_FIRST:
4413 IFCALLRET(altsec->common.DrawGdiPlusCacheFirst, rc, context,
4414 &(altsec->draw_gdiplus_cache_first));
4417 case ORDER_TYPE_GDIPLUS_CACHE_NEXT:
4418 IFCALLRET(altsec->common.DrawGdiPlusCacheNext, rc, context,
4419 &(altsec->draw_gdiplus_cache_next));
4422 case ORDER_TYPE_GDIPLUS_CACHE_END:
4423 IFCALLRET(altsec->common.DrawGdiPlusCacheEnd, rc, context,
4424 &(altsec->draw_gdiplus_cache_end));
4427 case ORDER_TYPE_WINDOW:
4428 rc = update_recv_altsec_window_order(update, s);
4431 case ORDER_TYPE_COMPDESK_FIRST:
4441 WLog_Print(up->log, WLOG_ERROR,
"%s %s failed", alt_sec_order_str, orderName);
4446BOOL update_recv_order(rdpUpdate* update,
wStream* s)
4449 BYTE controlFlags = 0;
4452 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
4455 Stream_Read_UINT8(s, controlFlags);
4457 if (!(controlFlags & ORDER_STANDARD))
4458 rc = update_recv_altsec_order(update, s, controlFlags);
4459 else if (controlFlags & ORDER_SECONDARY)
4460 rc = update_recv_secondary_order(update, s, controlFlags);
4462 rc = update_recv_primary_order(update, s, controlFlags);
4465 WLog_Print(up->log, WLOG_ERROR,
"order flags %02" PRIx8
" failed", controlFlags);
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.