20#include <freerdp/config.h>
22#include <winpr/assert.h>
27#include <winpr/cast.h>
29#include <winpr/synch.h>
30#include <winpr/thread.h>
31#include <winpr/stream.h>
33#include <freerdp/freerdp.h>
34#include <freerdp/codec/color.h>
36#include <freerdp/channels/wtsvc.h>
37#include <freerdp/channels/log.h>
39#include "rdpgfx_common.h"
40#include "rdpgfx_main.h"
42#define TAG CHANNELS_TAG("rdpgfx.server")
44static BOOL rdpgfx_server_close(RdpgfxServerContext* context);
46#define checkCapsAreExchanged(context) \
47 checkCapsAreExchangedInt(context, __FILE__, __func__, __LINE__)
48WINPR_ATTR_NODISCARD
static BOOL checkCapsAreExchangedInt(RdpgfxServerContext* context,
49 const char* file,
const char* fkt,
52 WINPR_ASSERT(context);
53 WINPR_ASSERT(context->priv);
55 const DWORD level = WLOG_TRACE;
56 if (WLog_IsLevelActive(context->priv->log, level))
58 WLog_PrintTextMessage(context->priv->log, level, line, file, fkt,
59 "activeCapSet{Version=0x%08" PRIx32
", flags=0x%08" PRIx32
"}",
60 context->priv->activeCapSet.version,
61 context->priv->activeCapSet.flags);
63 return context->priv->activeCapSet.version > 0;
75WINPR_ATTR_NODISCARD
static inline size_t rdpgfx_pdu_length(
size_t dataLen)
77 return RDPGFX_HEADER_SIZE + dataLen;
80WINPR_ATTR_NODISCARD
static inline UINT rdpgfx_server_packet_init_header(
wStream* s, UINT16 cmdId,
85 .pduLength = WINPR_ASSERTING_INT_CAST(UINT32, pduLength) };
88 return rdpgfx_write_header(s, &header);
98WINPR_ATTR_NODISCARD
static inline BOOL rdpgfx_server_packet_complete_header(
wStream* s,
101 const size_t current = Stream_GetPosition(s);
102 const size_t cap = Stream_Capacity(s);
103 if (cap < start + RDPGFX_HEADER_SIZE)
105 if ((start > UINT32_MAX) || (current < start))
108 if (!Stream_SetPosition(s, start + RDPGFX_HEADER_SIZE -
sizeof(UINT32)))
110 Stream_Write_UINT32(s, (UINT32)(current - start));
111 return Stream_SetPosition(s, current);
121WINPR_ATTR_NODISCARD
static UINT rdpgfx_server_packet_send(RdpgfxServerContext* context,
wStream* s)
126 BYTE* pSrcData = Stream_Buffer(s);
127 const size_t SrcSize = Stream_GetPosition(s);
128 if (SrcSize > UINT32_MAX)
129 return ERROR_INTERNAL_ERROR;
131 WINPR_ASSERT(context);
132 WINPR_ASSERT(context->priv);
137 wStream* fs = Stream_New(
nullptr, SrcSize + 7 + (SrcSize / ZGFX_SEGMENTED_MAXSIZE + 1) * 4);
141 WLog_Print(context->priv->log, WLOG_ERROR,
"Stream_New failed!");
142 error = CHANNEL_RC_NO_MEMORY;
146 if (zgfx_compress_to_stream(context->priv->zgfx, fs, pSrcData, (UINT32)SrcSize, &flags) < 0)
148 WLog_Print(context->priv->log, WLOG_ERROR,
"zgfx_compress_to_stream failed!");
149 error = ERROR_INTERNAL_ERROR;
154 const size_t pos = Stream_GetPosition(fs);
155 WINPR_ASSERT(pos <= UINT32_MAX);
156 if (!WTSVirtualChannelWrite(context->priv->rdpgfx_channel, Stream_BufferAs(fs,
char),
157 (UINT32)pos, &written))
159 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelWrite failed!");
160 error = ERROR_INTERNAL_ERROR;
165 if (written < Stream_GetPosition(fs))
167 WLog_Print(context->priv->log, WLOG_WARN,
168 "Unexpected bytes written: %" PRIu32
"/%" PRIuz
"", written,
169 Stream_GetPosition(fs));
172 error = CHANNEL_RC_OK;
174 Stream_Free(fs, TRUE);
175 Stream_Free(s, TRUE);
191WINPR_ATTR_MALLOC(Stream_Free, 1)
192static
wStream* rdpgfx_server_single_packet_new(wLog* log, UINT16 cmdId,
size_t dataLen)
195 const size_t pduLength = rdpgfx_pdu_length(dataLen);
196 wStream* s = Stream_New(
nullptr, pduLength);
200 WLog_Print(log, WLOG_ERROR,
"Stream_New failed!");
204 if ((error = rdpgfx_server_packet_init_header(s, cmdId, pduLength)))
206 WLog_Print(log, WLOG_ERROR,
"Failed to init header with error %" PRIu32
"!", error);
212 Stream_Free(s, TRUE);
224WINPR_ATTR_NODISCARD
static inline UINT
225rdpgfx_server_single_packet_send(RdpgfxServerContext* context,
wStream* s)
228 if (!rdpgfx_server_packet_complete_header(s, 0))
229 return ERROR_INTERNAL_ERROR;
230 return rdpgfx_server_packet_send(context, s);
238WINPR_ATTR_NODISCARD
static UINT
239rdpgfx_send_caps_confirm_pdu(RdpgfxServerContext* context,
242 WINPR_ASSERT(context);
243 WINPR_ASSERT(context->priv);
244 WINPR_ASSERT(capsConfirm);
247 WINPR_ASSERT(capsSet);
249 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_CAPSCONFIRM,
250 RDPGFX_CAPSET_BASE_SIZE + capsSet->length);
254 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
255 return CHANNEL_RC_NO_MEMORY;
258 WLog_Print(context->priv->log, WLOG_DEBUG,
259 "CAPS version=0x%04" PRIx32
", flags=0x%04" PRIx32
", length=%" PRIu32,
260 capsSet->version, capsSet->flags, capsSet->length);
261 Stream_Write_UINT32(s, capsSet->version);
262 Stream_Write_UINT32(s, capsSet->length);
264 if (capsSet->length >= 4)
266 Stream_Write_UINT32(s, capsSet->flags);
267 Stream_Zero(s, capsSet->length - 4);
270 Stream_Zero(s, capsSet->length);
272 context->priv->activeCapSet = *capsSet;
273 return rdpgfx_server_single_packet_send(context, s);
281WINPR_ATTR_NODISCARD
static UINT
284 const size_t RDPGFX_RESET_GRAPHICS_PDU_SIZE = 340;
286 if (!checkCapsAreExchanged(context))
287 return CHANNEL_RC_NOT_INITIALIZED;
290 WINPR_ASSERT(context->priv);
293 if (pdu->monitorCount >= 16)
295 WLog_Print(context->priv->log, WLOG_ERROR,
296 "Monitor count MUST be less than or equal to 16: %" PRIu32
"",
298 return ERROR_INVALID_DATA;
302 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_RESETGRAPHICS,
303 RDPGFX_RESET_GRAPHICS_PDU_SIZE - RDPGFX_HEADER_SIZE);
307 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
308 return CHANNEL_RC_NO_MEMORY;
311 Stream_Write_UINT32(s, pdu->width);
312 Stream_Write_UINT32(s, pdu->height);
313 Stream_Write_UINT32(s, pdu->monitorCount);
315 for (UINT32 index = 0; index < pdu->monitorCount; index++)
317 const MONITOR_DEF* monitor = &(pdu->monitorDefArray[index]);
318 Stream_Write_INT32(s, monitor->left);
319 Stream_Write_INT32(s, monitor->top);
320 Stream_Write_INT32(s, monitor->right);
321 Stream_Write_INT32(s, monitor->bottom);
322 Stream_Write_UINT32(s, monitor->flags);
326 const size_t pos = Stream_GetPosition(s);
327 if (pos > RDPGFX_RESET_GRAPHICS_PDU_SIZE)
329 Stream_Free(s, TRUE);
330 return ERROR_INVALID_DATA;
332 if (!Stream_SafeSeek(s, RDPGFX_RESET_GRAPHICS_PDU_SIZE - pos))
334 Stream_Free(s, TRUE);
335 return ERROR_INVALID_DATA;
337 return rdpgfx_server_single_packet_send(context, s);
345WINPR_ATTR_NODISCARD
static UINT
346rdpgfx_send_evict_cache_entry_pdu(RdpgfxServerContext* context,
349 if (!checkCapsAreExchanged(context))
350 return CHANNEL_RC_NOT_INITIALIZED;
353 WINPR_ASSERT(context->priv);
355 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_EVICTCACHEENTRY, 2);
359 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
360 return CHANNEL_RC_NO_MEMORY;
363 Stream_Write_UINT16(s, pdu->cacheSlot);
364 return rdpgfx_server_single_packet_send(context, s);
372WINPR_ATTR_NODISCARD
static UINT
373rdpgfx_send_cache_import_reply_pdu(RdpgfxServerContext* context,
376 if (!checkCapsAreExchanged(context))
377 return CHANNEL_RC_NOT_INITIALIZED;
378 WINPR_ASSERT(context);
379 WINPR_ASSERT(context->priv);
382 WLog_Print(context->priv->log, WLOG_DEBUG,
"reply with %" PRIu16
" entries",
383 pdu->importedEntriesCount);
384 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_CACHEIMPORTREPLY,
385 2 + 2 * pdu->importedEntriesCount);
389 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
390 return CHANNEL_RC_NO_MEMORY;
394 Stream_Write_UINT16(s, pdu->importedEntriesCount);
396 for (UINT16 index = 0; index < pdu->importedEntriesCount; index++)
398 Stream_Write_UINT16(s, pdu->cacheSlots[index]);
401 return rdpgfx_server_single_packet_send(context, s);
404WINPR_ATTR_NODISCARD
static UINT
405rdpgfx_process_cache_import_offer_pdu(RdpgfxServerContext* context,
408 if (!checkCapsAreExchanged(context))
409 return CHANNEL_RC_NOT_INITIALIZED;
410 WINPR_ASSERT(context);
411 WINPR_ASSERT(context->priv);
412 WINPR_ASSERT(cacheImportOffer);
415 WLog_Print(context->priv->log, WLOG_DEBUG,
416 "received %" PRIu16
" entries, reply with %" PRIu16
" entries",
417 cacheImportOffer->cacheEntriesCount, reply.importedEntriesCount);
418 return IFCALLRESULT(CHANNEL_RC_OK, context->CacheImportReply, context, &reply);
426WINPR_ATTR_NODISCARD
static UINT
429 if (!checkCapsAreExchanged(context))
430 return CHANNEL_RC_NOT_INITIALIZED;
432 WINPR_ASSERT(context->priv);
433 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_CREATESURFACE, 7);
435 WINPR_ASSERT(context);
437 WINPR_ASSERT((pdu->pixelFormat == GFX_PIXEL_FORMAT_XRGB_8888) ||
438 (pdu->pixelFormat == GFX_PIXEL_FORMAT_ARGB_8888));
442 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
443 return CHANNEL_RC_NO_MEMORY;
446 Stream_Write_UINT16(s, pdu->surfaceId);
447 Stream_Write_UINT16(s, pdu->width);
448 Stream_Write_UINT16(s, pdu->height);
449 Stream_Write_UINT8(s, pdu->pixelFormat);
450 return rdpgfx_server_single_packet_send(context, s);
458WINPR_ATTR_NODISCARD
static UINT
461 if (!checkCapsAreExchanged(context))
462 return CHANNEL_RC_NOT_INITIALIZED;
465 WINPR_ASSERT(context->priv);
466 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_DELETESURFACE, 2);
470 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
471 return CHANNEL_RC_NO_MEMORY;
474 Stream_Write_UINT16(s, pdu->surfaceId);
475 return rdpgfx_server_single_packet_send(context, s);
478WINPR_ATTR_NODISCARD
static inline BOOL
481 if (!Stream_EnsureRemainingCapacity(s, 8))
485 Stream_Write_UINT32(s, pdu->timestamp);
486 Stream_Write_UINT32(s, pdu->frameId);
490WINPR_ATTR_NODISCARD
static inline BOOL rdpgfx_write_end_frame_pdu(
wStream* s,
493 if (!Stream_EnsureRemainingCapacity(s, 4))
496 Stream_Write_UINT32(s, pdu->frameId);
505WINPR_ATTR_NODISCARD
static UINT rdpgfx_send_start_frame_pdu(RdpgfxServerContext* context,
508 if (!checkCapsAreExchanged(context))
509 return CHANNEL_RC_NOT_INITIALIZED;
512 WINPR_ASSERT(context->priv);
513 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_STARTFRAME,
514 RDPGFX_START_FRAME_PDU_SIZE);
518 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
519 return CHANNEL_RC_NO_MEMORY;
522 if (!rdpgfx_write_start_frame_pdu(s, pdu))
524 Stream_Free(s, TRUE);
525 return ERROR_INTERNAL_ERROR;
527 return rdpgfx_server_single_packet_send(context, s);
535WINPR_ATTR_NODISCARD
static UINT rdpgfx_send_end_frame_pdu(RdpgfxServerContext* context,
538 if (!checkCapsAreExchanged(context))
539 return CHANNEL_RC_NOT_INITIALIZED;
542 WINPR_ASSERT(context->priv);
543 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_ENDFRAME,
544 RDPGFX_END_FRAME_PDU_SIZE);
548 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
549 return CHANNEL_RC_NO_MEMORY;
552 if (!rdpgfx_write_end_frame_pdu(s, pdu))
554 Stream_Free(s, TRUE);
555 return ERROR_INTERNAL_ERROR;
557 return rdpgfx_server_single_packet_send(context, s);
566WINPR_ATTR_NODISCARD
static inline UINT32
569 WINPR_ASSERT(havc420);
571 return sizeof(UINT32)
573 * havc420->meta.numRegionRects +
583WINPR_ATTR_NODISCARD
static inline UINT32
593 switch (cmd->codecId)
595 case RDPGFX_CODECID_CAPROGRESSIVE:
596 case RDPGFX_CODECID_CAPROGRESSIVE_V2:
597 return RDPGFX_WIRE_TO_SURFACE_PDU_2_SIZE + cmd->length;
599 case RDPGFX_CODECID_AVC420:
601 h264Size = rdpgfx_estimate_h264_avc420(havc420);
602 return RDPGFX_WIRE_TO_SURFACE_PDU_1_SIZE + h264Size;
604 case RDPGFX_CODECID_AVC444:
606 h264Size =
sizeof(UINT32);
608 havc420 = &(havc444->bitstream[0]);
609 h264Size += rdpgfx_estimate_h264_avc420(havc420);
612 if (havc444->LC == 0)
614 havc420 = &(havc444->bitstream[1]);
615 h264Size += rdpgfx_estimate_h264_avc420(havc420);
618 return RDPGFX_WIRE_TO_SURFACE_PDU_1_SIZE + h264Size;
621 return RDPGFX_WIRE_TO_SURFACE_PDU_1_SIZE + cmd->length;
632WINPR_ATTR_NODISCARD
static inline UINT16
637 if (cmd->codecId == RDPGFX_CODECID_CAPROGRESSIVE ||
638 cmd->codecId == RDPGFX_CODECID_CAPROGRESSIVE_V2)
640 return RDPGFX_CMDID_WIRETOSURFACE_2;
643 return RDPGFX_CMDID_WIRETOSURFACE_1;
651WINPR_ATTR_NODISCARD
static UINT rdpgfx_write_h264_metablock(wLog* log,
wStream* s,
656 UINT error = CHANNEL_RC_OK;
659 if (!Stream_EnsureRemainingCapacity(s, 4 + meta->numRegionRects * 10))
660 return ERROR_OUTOFMEMORY;
662 Stream_Write_UINT32(s, meta->numRegionRects);
664 for (UINT32 index = 0; index < meta->numRegionRects; index++)
666 regionRect = &(meta->regionRects[index]);
668 if ((error = rdpgfx_write_rect16(s, regionRect)))
670 WLog_Print(log, WLOG_ERROR,
"rdpgfx_write_rect16 failed with error %" PRIu32
"!",
676 for (UINT32 index = 0; index < meta->numRegionRects; index++)
678 quantQualityVal = &(meta->quantQualityVals[index]);
679 Stream_Write_UINT8(s, WINPR_ASSERTING_INT_CAST(
680 uint8_t, quantQualityVal->qp | (quantQualityVal->r << 6) |
681 (quantQualityVal->p << 7)));
683 Stream_Write_UINT8(s, quantQualityVal->qualityVal);
695WINPR_ATTR_NODISCARD
static inline UINT
698 WINPR_ASSERT(havc420);
699 const UINT error = rdpgfx_write_h264_metablock(log, s, &(havc420->meta));
701 if (error != CHANNEL_RC_OK)
703 WLog_Print(log, WLOG_ERROR,
"rdpgfx_write_h264_metablock failed with error %" PRIu32
"!",
708 if (!Stream_EnsureRemainingCapacity(s, havc420->length))
709 return ERROR_OUTOFMEMORY;
711 Stream_Write(s, havc420->data, havc420->length);
722WINPR_ATTR_NODISCARD
static UINT rdpgfx_write_surface_command(wLog* log,
wStream* s,
725 UINT error = CHANNEL_RC_OK;
728 UINT8 pixelFormat = 0;
733 case PIXEL_FORMAT_BGRX32:
734 pixelFormat = GFX_PIXEL_FORMAT_XRGB_8888;
737 case PIXEL_FORMAT_BGRA32:
738 pixelFormat = GFX_PIXEL_FORMAT_ARGB_8888;
742 WLog_Print(log, WLOG_ERROR,
"Format %s not supported!",
743 FreeRDPGetColorFormatName(cmd->format));
744 return ERROR_INVALID_DATA;
747 if (cmd->codecId == RDPGFX_CODECID_CAPROGRESSIVE ||
748 cmd->codecId == RDPGFX_CODECID_CAPROGRESSIVE_V2)
750 if (!Stream_EnsureRemainingCapacity(s, 13 + cmd->length))
751 return ERROR_INTERNAL_ERROR;
754 s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->surfaceId));
756 s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->codecId));
757 Stream_Write_UINT32(s, cmd->contextId);
758 Stream_Write_UINT8(s, pixelFormat);
759 Stream_Write_UINT32(s, cmd->length);
760 Stream_Write(s, cmd->data, cmd->length);
765 if (!Stream_EnsureRemainingCapacity(s, 17))
766 return ERROR_INTERNAL_ERROR;
768 s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->surfaceId));
770 s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->codecId));
771 Stream_Write_UINT8(s, pixelFormat);
772 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->left));
773 Stream_Write_UINT16(s, WINPR_ASSERTING_INT_CAST(uint16_t, cmd->top));
774 Stream_Write_UINT16(s,
775 WINPR_ASSERTING_INT_CAST(uint16_t, cmd->right));
776 Stream_Write_UINT16(s,
777 WINPR_ASSERTING_INT_CAST(uint16_t, cmd->bottom));
778 Stream_Write_UINT32(s, cmd->length);
779 const size_t bitmapDataStart = Stream_GetPosition(s);
781 if (cmd->codecId == RDPGFX_CODECID_AVC420)
784 error = rdpgfx_write_h264_avc420(log, s, havc420);
786 if (error != CHANNEL_RC_OK)
788 WLog_Print(log, WLOG_ERROR,
"rdpgfx_write_h264_avc420 failed!");
792 else if ((cmd->codecId == RDPGFX_CODECID_AVC444) ||
793 (cmd->codecId == RDPGFX_CODECID_AVC444v2))
796 havc420 = &(havc444->bitstream[0]);
797 if (!Stream_EnsureRemainingCapacity(s, 4))
798 return ERROR_INTERNAL_ERROR;
799 Stream_Write_UINT32(s, havc444->cbAvc420EncodedBitstream1 |
800 ((uint32_t)havc444->LC << 30UL));
802 error = rdpgfx_write_h264_avc420(log, s, havc420);
804 if (error != CHANNEL_RC_OK)
806 WLog_Print(log, WLOG_ERROR,
"rdpgfx_write_h264_avc420 failed!");
811 if (havc444->LC == 0)
813 havc420 = &(havc444->bitstream[1]);
814 error = rdpgfx_write_h264_avc420(log, s, havc420);
816 if (error != CHANNEL_RC_OK)
818 WLog_Print(log, WLOG_ERROR,
"rdpgfx_write_h264_avc420 failed!");
825 if (!Stream_EnsureRemainingCapacity(s, cmd->length))
826 return ERROR_INTERNAL_ERROR;
827 Stream_Write(s, cmd->data, cmd->length);
831 const size_t bitmapDataLength = Stream_GetPosition(s) - bitmapDataStart;
832 if (bitmapDataLength > UINT32_MAX)
833 return ERROR_INTERNAL_ERROR;
835 if (!Stream_SetPosition(s, bitmapDataStart -
sizeof(UINT32)))
836 return ERROR_INVALID_DATA;
837 if (!Stream_EnsureRemainingCapacity(s, 4))
838 return ERROR_INTERNAL_ERROR;
839 Stream_Write_UINT32(s, (UINT32)bitmapDataLength);
840 if (!Stream_SafeSeek(s, bitmapDataLength))
841 return ERROR_INTERNAL_ERROR;
854WINPR_ATTR_NODISCARD
static UINT rdpgfx_send_surface_command(RdpgfxServerContext* context,
857 if (!checkCapsAreExchanged(context))
858 return CHANNEL_RC_NOT_INITIALIZED;
859 UINT error = CHANNEL_RC_OK;
861 WINPR_ASSERT(context);
862 WINPR_ASSERT(context->priv);
866 rdpgfx_server_single_packet_new(context->priv->log, rdpgfx_surface_command_cmdid(cmd),
867 rdpgfx_estimate_surface_command(cmd));
871 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
872 return CHANNEL_RC_NO_MEMORY;
875 error = rdpgfx_write_surface_command(context->priv->log, s, cmd);
877 if (error != CHANNEL_RC_OK)
879 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_write_surface_command failed!");
883 return rdpgfx_server_single_packet_send(context, s);
885 Stream_Free(s, TRUE);
897WINPR_ATTR_NODISCARD
static UINT
903 if (!checkCapsAreExchanged(context))
904 return CHANNEL_RC_NOT_INITIALIZED;
906 WINPR_ASSERT(context->priv);
908 WINPR_ASSERT(startFrame);
909 WINPR_ASSERT(endFrame);
911 UINT error = CHANNEL_RC_OK;
912 size_t size = rdpgfx_pdu_length(rdpgfx_estimate_surface_command(cmd));
916 size += rdpgfx_pdu_length(RDPGFX_START_FRAME_PDU_SIZE);
921 size += rdpgfx_pdu_length(RDPGFX_END_FRAME_PDU_SIZE);
924 wStream* s = Stream_New(
nullptr, size);
928 WLog_Print(context->priv->log, WLOG_ERROR,
"Stream_New failed!");
929 return CHANNEL_RC_NO_MEMORY;
935 const size_t position = Stream_GetPosition(s);
936 error = rdpgfx_server_packet_init_header(s, RDPGFX_CMDID_STARTFRAME, 0);
938 if (error != CHANNEL_RC_OK)
940 WLog_Print(context->priv->log, WLOG_ERROR,
941 "Failed to init header with error %" PRIu32
"!", error);
945 if (!rdpgfx_write_start_frame_pdu(s, startFrame) ||
946 !rdpgfx_server_packet_complete_header(s, position))
952 const size_t pos = Stream_GetPosition(s);
953 error = rdpgfx_server_packet_init_header(s, rdpgfx_surface_command_cmdid(cmd),
956 if (error != CHANNEL_RC_OK)
958 WLog_Print(context->priv->log, WLOG_ERROR,
959 "Failed to init header with error %" PRIu32
"!", error);
963 error = rdpgfx_write_surface_command(context->priv->log, s, cmd);
965 if (error != CHANNEL_RC_OK)
967 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_write_surface_command failed!");
971 if (!rdpgfx_server_packet_complete_header(s, pos))
978 const size_t position = Stream_GetPosition(s);
979 error = rdpgfx_server_packet_init_header(s, RDPGFX_CMDID_ENDFRAME, 0);
981 if (error != CHANNEL_RC_OK)
983 WLog_Print(context->priv->log, WLOG_ERROR,
984 "Failed to init header with error %" PRIu32
"!", error);
988 if (!rdpgfx_write_end_frame_pdu(s, endFrame) ||
989 !rdpgfx_server_packet_complete_header(s, position))
993 return rdpgfx_server_packet_send(context, s);
995 Stream_Free(s, TRUE);
1004WINPR_ATTR_NODISCARD
static UINT
1005rdpgfx_send_delete_encoding_context_pdu(RdpgfxServerContext* context,
1008 if (!checkCapsAreExchanged(context))
1009 return CHANNEL_RC_NOT_INITIALIZED;
1011 WINPR_ASSERT(context->priv);
1014 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_DELETEENCODINGCONTEXT, 6);
1018 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1019 return CHANNEL_RC_NO_MEMORY;
1022 Stream_Write_UINT16(s, pdu->surfaceId);
1023 Stream_Write_UINT32(s, pdu->codecContextId);
1024 return rdpgfx_server_single_packet_send(context, s);
1032WINPR_ATTR_NODISCARD
static UINT rdpgfx_send_solid_fill_pdu(RdpgfxServerContext* context,
1035 if (!checkCapsAreExchanged(context))
1036 return CHANNEL_RC_NOT_INITIALIZED;
1038 WINPR_ASSERT(context->priv);
1041 UINT error = CHANNEL_RC_OK;
1043 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_SOLIDFILL,
1044 8 + 8 * pdu->fillRectCount);
1048 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1049 return CHANNEL_RC_NO_MEMORY;
1052 Stream_Write_UINT16(s, pdu->surfaceId);
1055 if ((error = rdpgfx_write_color32(s, &(pdu->fillPixel))))
1057 WLog_Print(context->priv->log, WLOG_ERROR,
1058 "rdpgfx_write_color32 failed with error %" PRIu32
"!", error);
1062 Stream_Write_UINT16(s, pdu->fillRectCount);
1064 for (UINT16 index = 0; index < pdu->fillRectCount; index++)
1066 fillRect = &(pdu->fillRects[index]);
1068 if ((error = rdpgfx_write_rect16(s, fillRect)))
1070 WLog_Print(context->priv->log, WLOG_ERROR,
1071 "rdpgfx_write_rect16 failed with error %" PRIu32
"!", error);
1076 return rdpgfx_server_single_packet_send(context, s);
1078 Stream_Free(s, TRUE);
1087WINPR_ATTR_NODISCARD
static UINT
1088rdpgfx_send_surface_to_surface_pdu(RdpgfxServerContext* context,
1091 if (!checkCapsAreExchanged(context))
1092 return CHANNEL_RC_NOT_INITIALIZED;
1095 WINPR_ASSERT(context->priv);
1097 UINT error = CHANNEL_RC_OK;
1099 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_SURFACETOSURFACE,
1100 14 + 4 * pdu->destPtsCount);
1104 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1105 return CHANNEL_RC_NO_MEMORY;
1108 Stream_Write_UINT16(s, pdu->surfaceIdSrc);
1109 Stream_Write_UINT16(s, pdu->surfaceIdDest);
1112 if ((error = rdpgfx_write_rect16(s, &(pdu->rectSrc))))
1114 WLog_Print(context->priv->log, WLOG_ERROR,
1115 "rdpgfx_write_rect16 failed with error %" PRIu32
"!", error);
1119 Stream_Write_UINT16(s, pdu->destPtsCount);
1121 for (UINT16 index = 0; index < pdu->destPtsCount; index++)
1123 destPt = &(pdu->destPts[index]);
1125 if ((error = rdpgfx_write_point16(s, destPt)))
1127 WLog_Print(context->priv->log, WLOG_ERROR,
1128 "rdpgfx_write_point16 failed with error %" PRIu32
"!", error);
1133 return rdpgfx_server_single_packet_send(context, s);
1135 Stream_Free(s, TRUE);
1144WINPR_ATTR_NODISCARD
static UINT
1145rdpgfx_send_surface_to_cache_pdu(RdpgfxServerContext* context,
1148 if (!checkCapsAreExchanged(context))
1149 return CHANNEL_RC_NOT_INITIALIZED;
1152 WINPR_ASSERT(context->priv);
1154 UINT error = CHANNEL_RC_OK;
1156 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_SURFACETOCACHE, 20);
1160 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1161 return CHANNEL_RC_NO_MEMORY;
1164 Stream_Write_UINT16(s, pdu->surfaceId);
1165 Stream_Write_UINT64(s, pdu->cacheKey);
1166 Stream_Write_UINT16(s, pdu->cacheSlot);
1169 if ((error = rdpgfx_write_rect16(s, &(pdu->rectSrc))))
1171 WLog_Print(context->priv->log, WLOG_ERROR,
1172 "rdpgfx_write_rect16 failed with error %" PRIu32
"!", error);
1176 return rdpgfx_server_single_packet_send(context, s);
1178 Stream_Free(s, TRUE);
1187WINPR_ATTR_NODISCARD
static UINT
1188rdpgfx_send_cache_to_surface_pdu(RdpgfxServerContext* context,
1191 if (!checkCapsAreExchanged(context))
1192 return CHANNEL_RC_NOT_INITIALIZED;
1195 WINPR_ASSERT(context->priv);
1197 UINT error = CHANNEL_RC_OK;
1199 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_CACHETOSURFACE,
1200 6 + 4 * pdu->destPtsCount);
1204 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1205 return CHANNEL_RC_NO_MEMORY;
1208 Stream_Write_UINT16(s, pdu->cacheSlot);
1209 Stream_Write_UINT16(s, pdu->surfaceId);
1210 Stream_Write_UINT16(s, pdu->destPtsCount);
1212 for (UINT16 index = 0; index < pdu->destPtsCount; index++)
1214 destPt = &(pdu->destPts[index]);
1216 if ((error = rdpgfx_write_point16(s, destPt)))
1218 WLog_Print(context->priv->log, WLOG_ERROR,
1219 "rdpgfx_write_point16 failed with error %" PRIu32
"", error);
1224 return rdpgfx_server_single_packet_send(context, s);
1226 Stream_Free(s, TRUE);
1235WINPR_ATTR_NODISCARD
static UINT
1236rdpgfx_send_map_surface_to_output_pdu(RdpgfxServerContext* context,
1239 if (!checkCapsAreExchanged(context))
1240 return CHANNEL_RC_NOT_INITIALIZED;
1243 WINPR_ASSERT(context->priv);
1245 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_MAPSURFACETOOUTPUT, 12);
1249 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1250 return CHANNEL_RC_NO_MEMORY;
1253 Stream_Write_UINT16(s, pdu->surfaceId);
1254 Stream_Write_UINT16(s, 0);
1255 Stream_Write_UINT32(s, pdu->outputOriginX);
1256 Stream_Write_UINT32(s, pdu->outputOriginY);
1257 return rdpgfx_server_single_packet_send(context, s);
1265WINPR_ATTR_NODISCARD
static UINT
1266rdpgfx_send_map_surface_to_window_pdu(RdpgfxServerContext* context,
1269 if (!checkCapsAreExchanged(context))
1270 return CHANNEL_RC_NOT_INITIALIZED;
1273 WINPR_ASSERT(context->priv);
1276 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_MAPSURFACETOWINDOW, 18);
1280 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1281 return CHANNEL_RC_NO_MEMORY;
1284 Stream_Write_UINT16(s, pdu->surfaceId);
1285 Stream_Write_UINT64(s, pdu->windowId);
1286 Stream_Write_UINT32(s, pdu->mappedWidth);
1287 Stream_Write_UINT32(s, pdu->mappedHeight);
1288 return rdpgfx_server_single_packet_send(context, s);
1291WINPR_ATTR_NODISCARD
static UINT
1292rdpgfx_send_map_surface_to_scaled_window_pdu(RdpgfxServerContext* context,
1295 if (!checkCapsAreExchanged(context))
1296 return CHANNEL_RC_NOT_INITIALIZED;
1299 WINPR_ASSERT(context->priv);
1300 wStream* s = rdpgfx_server_single_packet_new(context->priv->log,
1301 RDPGFX_CMDID_MAPSURFACETOSCALEDWINDOW, 26);
1305 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1306 return CHANNEL_RC_NO_MEMORY;
1309 Stream_Write_UINT16(s, pdu->surfaceId);
1310 Stream_Write_UINT64(s, pdu->windowId);
1311 Stream_Write_UINT32(s, pdu->mappedWidth);
1312 Stream_Write_UINT32(s, pdu->mappedHeight);
1313 Stream_Write_UINT32(s, pdu->targetWidth);
1314 Stream_Write_UINT32(s, pdu->targetHeight);
1315 return rdpgfx_server_single_packet_send(context, s);
1323WINPR_ATTR_NODISCARD
static UINT rdpgfx_recv_frame_acknowledge_pdu(RdpgfxServerContext* context,
1326 WINPR_ASSERT(context);
1328 if (!checkCapsAreExchanged(context))
1329 return CHANNEL_RC_NOT_INITIALIZED;
1331 WINPR_ASSERT(context->priv);
1334 UINT error = CHANNEL_RC_OK;
1336 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 12))
1337 return ERROR_INVALID_DATA;
1339 Stream_Read_UINT32(s, pdu.queueDepth);
1340 Stream_Read_UINT32(s, pdu.frameId);
1341 Stream_Read_UINT32(s, pdu.totalFramesDecoded);
1343 IFCALLRET(context->FrameAcknowledge, error, context, &pdu);
1346 WLog_Print(context->priv->log, WLOG_ERROR,
1347 "context->FrameAcknowledge failed with error %" PRIu32
"", error);
1357WINPR_ATTR_NODISCARD
static UINT rdpgfx_recv_cache_import_offer_pdu(RdpgfxServerContext* context,
1360 WINPR_ASSERT(context);
1361 if (!checkCapsAreExchanged(context))
1362 return CHANNEL_RC_NOT_INITIALIZED;
1364 WINPR_ASSERT(context->priv);
1368 UINT error = CHANNEL_RC_OK;
1370 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 2))
1371 return ERROR_INVALID_DATA;
1374 Stream_Read_UINT16(s, pdu.cacheEntriesCount);
1377 if (pdu.cacheEntriesCount >= 5462)
1379 WLog_Print(context->priv->log, WLOG_ERROR,
"Invalid cacheEntriesCount: %" PRIu16
"",
1380 pdu.cacheEntriesCount);
1381 return ERROR_INVALID_DATA;
1384 if (!Stream_CheckAndLogRequiredLengthOfSizeWLog(context->priv->log, s, pdu.cacheEntriesCount,
1386 return ERROR_INVALID_DATA;
1388 for (UINT16 index = 0; index < pdu.cacheEntriesCount; index++)
1390 cacheEntry = &(pdu.cacheEntries[index]);
1391 Stream_Read_UINT64(s, cacheEntry->cacheKey);
1392 Stream_Read_UINT32(s, cacheEntry->bitmapLength);
1395 IFCALLRET(context->CacheImportOffer, error, context, &pdu);
1398 WLog_Print(context->priv->log, WLOG_ERROR,
1399 "context->CacheImportOffer failed with error %" PRIu32
"", error);
1409WINPR_ATTR_NODISCARD
static UINT rdpgfx_recv_caps_advertise_pdu(RdpgfxServerContext* context,
1414 UINT error = ERROR_INVALID_DATA;
1417 return ERROR_BAD_ARGUMENTS;
1419 WINPR_ASSERT(context->priv);
1420 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 2))
1421 return ERROR_INVALID_DATA;
1423 Stream_Read_UINT16(s, pdu.capsSetCount);
1424 if (pdu.capsSetCount > 0)
1426 capsSets = calloc(pdu.capsSetCount, (RDPGFX_CAPSET_BASE_SIZE + 4));
1428 return ERROR_OUTOFMEMORY;
1431 pdu.capsSets = capsSets;
1433 for (UINT16 index = 0; index < pdu.capsSetCount; index++)
1437 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 8))
1440 Stream_Read_UINT32(s, capsSet->version);
1441 Stream_Read_UINT32(s, capsSet->length);
1443 if (capsSet->length >= 4)
1445 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 4))
1448 Stream_Peek_UINT32(s, capsSet->flags);
1451 if (!Stream_SafeSeek(s, capsSet->length))
1455 error = ERROR_BAD_CONFIGURATION;
1456 IFCALLRET(context->CapsAdvertise, error, context, &pdu);
1459 WLog_Print(context->priv->log, WLOG_ERROR,
1460 "context->CapsAdvertise failed with error %" PRIu32
"", error);
1472WINPR_ATTR_NODISCARD
static UINT rdpgfx_recv_qoe_frame_acknowledge_pdu(RdpgfxServerContext* context,
1475 WINPR_ASSERT(context);
1477 if (!checkCapsAreExchanged(context))
1478 return CHANNEL_RC_NOT_INITIALIZED;
1481 UINT error = CHANNEL_RC_OK;
1483 WINPR_ASSERT(context->priv);
1485 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 12))
1486 return ERROR_INVALID_DATA;
1488 Stream_Read_UINT32(s, pdu.frameId);
1489 Stream_Read_UINT32(s, pdu.timestamp);
1490 Stream_Read_UINT16(s, pdu.timeDiffSE);
1491 Stream_Read_UINT16(s, pdu.timeDiffEDR);
1493 IFCALLRET(context->QoeFrameAcknowledge, error, context, &pdu);
1496 WLog_Print(context->priv->log, WLOG_ERROR,
1497 "context->QoeFrameAcknowledge failed with error %" PRIu32
"", error);
1502WINPR_ATTR_NODISCARD
static UINT
1503rdpgfx_send_map_surface_to_scaled_output_pdu(RdpgfxServerContext* context,
1506 if (!checkCapsAreExchanged(context))
1507 return CHANNEL_RC_NOT_INITIALIZED;
1510 WINPR_ASSERT(context->priv);
1511 wStream* s = rdpgfx_server_single_packet_new(context->priv->log,
1512 RDPGFX_CMDID_MAPSURFACETOSCALEDOUTPUT, 20);
1516 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1517 return CHANNEL_RC_NO_MEMORY;
1520 Stream_Write_UINT16(s, pdu->surfaceId);
1521 Stream_Write_UINT16(s, 0);
1522 Stream_Write_UINT32(s, pdu->outputOriginX);
1523 Stream_Write_UINT32(s, pdu->outputOriginY);
1524 Stream_Write_UINT32(s, pdu->targetWidth);
1525 Stream_Write_UINT32(s, pdu->targetHeight);
1526 return rdpgfx_server_single_packet_send(context, s);
1534WINPR_ATTR_NODISCARD
static UINT rdpgfx_server_receive_pdu(RdpgfxServerContext* context,
wStream* s)
1538 UINT error = CHANNEL_RC_OK;
1539 size_t beg = Stream_GetPosition(s);
1541 WINPR_ASSERT(context);
1542 WINPR_ASSERT(context->priv);
1544 if ((error = rdpgfx_read_header(context->priv->log, s, &header)))
1546 WLog_Print(context->priv->log, WLOG_ERROR,
1547 "rdpgfx_read_header failed with error %" PRIu32
"!", error);
1551 WLog_Print(context->priv->log, WLOG_TRACE,
1552 "cmdId: %s (0x%04" PRIX16
") flags: 0x%04" PRIX16
" pduLength: %" PRIu32
"",
1553 rdpgfx_get_cmd_id_string(header.cmdId), header.cmdId, header.flags,
1556 switch (header.cmdId)
1558 case RDPGFX_CMDID_FRAMEACKNOWLEDGE:
1559 if ((error = rdpgfx_recv_frame_acknowledge_pdu(context, s)))
1560 WLog_Print(context->priv->log, WLOG_ERROR,
1561 "rdpgfx_recv_frame_acknowledge_pdu "
1562 "failed with error %" PRIu32
"!",
1567 case RDPGFX_CMDID_CACHEIMPORTOFFER:
1568 if ((error = rdpgfx_recv_cache_import_offer_pdu(context, s)))
1569 WLog_Print(context->priv->log, WLOG_ERROR,
1570 "rdpgfx_recv_cache_import_offer_pdu "
1571 "failed with error %" PRIu32
"!",
1576 case RDPGFX_CMDID_CAPSADVERTISE:
1577 if ((error = rdpgfx_recv_caps_advertise_pdu(context, s)))
1578 WLog_Print(context->priv->log, WLOG_ERROR,
1579 "rdpgfx_recv_caps_advertise_pdu "
1580 "failed with error %" PRIu32
"!",
1585 case RDPGFX_CMDID_QOEFRAMEACKNOWLEDGE:
1586 if ((error = rdpgfx_recv_qoe_frame_acknowledge_pdu(context, s)))
1587 WLog_Print(context->priv->log, WLOG_ERROR,
1588 "rdpgfx_recv_qoe_frame_acknowledge_pdu "
1589 "failed with error %" PRIu32
"!",
1595 error = CHANNEL_RC_BAD_PROC;
1601 WLog_Print(context->priv->log, WLOG_ERROR,
1602 "Error while parsing GFX cmdId: %s (0x%04" PRIX16
")",
1603 rdpgfx_get_cmd_id_string(header.cmdId), header.cmdId);
1607 end = Stream_GetPosition(s);
1609 if (end != (beg + header.pduLength))
1611 WLog_Print(context->priv->log, WLOG_ERROR,
1612 "Unexpected gfx pdu end: Actual: %" PRIuz
", Expected: %" PRIuz
"", end,
1613 (beg + header.pduLength));
1614 if (!Stream_SetPosition(s, (beg + header.pduLength)))
1615 return ERROR_INVALID_DATA;
1621WINPR_ATTR_NODISCARD
static DWORD WINAPI rdpgfx_server_thread_func(LPVOID arg)
1623 RdpgfxServerContext* context = (RdpgfxServerContext*)arg;
1624 WINPR_ASSERT(context);
1626 RdpgfxServerPrivate* priv = context->priv;
1629 HANDLE events[8] = WINPR_C_ARRAY_INIT;
1630 UINT error = CHANNEL_RC_OK;
1634 if (priv->ownThread)
1636 WINPR_ASSERT(priv->stopEvent);
1637 events[nCount++] = priv->stopEvent;
1640 WINPR_ASSERT(priv->channelEvent);
1641 events[nCount++] = priv->channelEvent;
1646 status = WaitForMultipleObjects(nCount, events, FALSE, INFINITE);
1648 if (status == WAIT_FAILED)
1650 error = GetLastError();
1651 WLog_Print(context->priv->log, WLOG_ERROR,
1652 "WaitForMultipleObjects failed with error %" PRIu32
"", error);
1657 if (status == WAIT_OBJECT_0)
1660 if ((error = rdpgfx_server_handle_messages(context)))
1662 WLog_Print(context->priv->log, WLOG_ERROR,
1663 "rdpgfx_server_handle_messages failed with error %" PRIu32
"", error);
1668 if (error && context->rdpcontext)
1669 setChannelError(context->rdpcontext, error,
"rdpgfx_server_thread_func reported an error");
1675WINPR_ATTR_NODISCARD
static BOOL rdpgfx_server_open(RdpgfxServerContext* context)
1677 WINPR_ASSERT(context);
1678 RdpgfxServerPrivate* priv = context->priv;
1679 void* buffer =
nullptr;
1683 if (!priv->isOpened)
1685 PULONG pSessionId =
nullptr;
1686 DWORD BytesReturned = 0;
1687 priv->SessionId = WTS_CURRENT_SESSION;
1688 UINT32 channelId = 0;
1691 if (WTSQuerySessionInformationA(context->vcm, WTS_CURRENT_SESSION, WTSSessionId,
1692 (LPSTR*)&pSessionId, &BytesReturned) == FALSE)
1694 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSQuerySessionInformationA failed!");
1698 priv->SessionId = (DWORD)*pSessionId;
1699 WTSFreeMemory(pSessionId);
1700 priv->rdpgfx_channel = WTSVirtualChannelOpenEx(priv->SessionId, RDPGFX_DVC_CHANNEL_NAME,
1701 WTS_CHANNEL_OPTION_DYNAMIC);
1703 if (!priv->rdpgfx_channel)
1705 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelOpenEx failed!");
1709 channelId = WTSChannelGetIdByHandle(priv->rdpgfx_channel);
1711 IFCALLRET(context->ChannelIdAssigned, status, context, channelId);
1714 WLog_Print(context->priv->log, WLOG_ERROR,
"context->ChannelIdAssigned failed!");
1719 if (!WTSVirtualChannelQuery(priv->rdpgfx_channel, WTSVirtualEventHandle, &buffer,
1721 (BytesReturned !=
sizeof(HANDLE)))
1723 WLog_Print(context->priv->log, WLOG_ERROR,
1724 "WTSVirtualChannelQuery failed "
1725 "or invalid returned size(%" PRIu32
")",
1729 WTSFreeMemory(buffer);
1734 priv->channelEvent = *(HANDLE*)buffer;
1735 WTSFreeMemory(buffer);
1737 if (!(priv->zgfx = zgfx_context_new(TRUE)))
1739 WLog_Print(context->priv->log, WLOG_ERROR,
"Create zgfx context failed!");
1743 priv->isReady = FALSE;
1745 priv->activeCapSet = empty;
1746 if (priv->ownThread)
1748 if (!(priv->stopEvent = CreateEvent(
nullptr, TRUE, FALSE,
nullptr)))
1750 WLog_Print(context->priv->log, WLOG_ERROR,
"CreateEvent failed!");
1754 if (!(priv->thread = CreateThread(
nullptr, 0, rdpgfx_server_thread_func, (
void*)context,
1757 WLog_Print(context->priv->log, WLOG_ERROR,
"CreateThread failed!");
1762 priv->isOpened = TRUE;
1766 WLog_Print(context->priv->log, WLOG_ERROR,
"RDPGFX channel is already opened!");
1769 (void)rdpgfx_server_close(context);
1773BOOL rdpgfx_server_close(RdpgfxServerContext* context)
1775 WINPR_ASSERT(context);
1777 RdpgfxServerPrivate* priv = context->priv;
1780 if (priv->ownThread && priv->thread)
1782 (void)SetEvent(priv->stopEvent);
1784 if (WaitForSingleObject(priv->thread, INFINITE) == WAIT_FAILED)
1786 WLog_Print(context->priv->log, WLOG_ERROR,
1787 "WaitForSingleObject failed with error %" PRIu32
"", GetLastError());
1791 (void)CloseHandle(priv->thread);
1792 (void)CloseHandle(priv->stopEvent);
1793 priv->thread =
nullptr;
1794 priv->stopEvent =
nullptr;
1797 zgfx_context_free(priv->zgfx);
1798 priv->zgfx =
nullptr;
1800 if (priv->rdpgfx_channel)
1802 (void)WTSVirtualChannelClose(priv->rdpgfx_channel);
1803 priv->rdpgfx_channel =
nullptr;
1806 priv->channelEvent =
nullptr;
1807 priv->isOpened = FALSE;
1808 priv->isReady = FALSE;
1810 priv->activeCapSet = empty;
1814WINPR_ATTR_NODISCARD
static BOOL rdpgfx_server_initialize(RdpgfxServerContext* context,
1815 BOOL externalThread)
1817 WINPR_ASSERT(context);
1818 WINPR_ASSERT(context->priv);
1820 if (context->priv->isOpened)
1822 WLog_Print(context->priv->log, WLOG_WARN,
1823 "Application error: RDPEGFX channel already initialized, "
1824 "calling in this state is not possible!");
1828 context->priv->ownThread = !externalThread;
1832RdpgfxServerContext* rdpgfx_server_context_new(HANDLE vcm)
1834 RdpgfxServerContext* context = (RdpgfxServerContext*)calloc(1,
sizeof(RdpgfxServerContext));
1840 context->Initialize = rdpgfx_server_initialize;
1841 context->Open = rdpgfx_server_open;
1842 context->Close = rdpgfx_server_close;
1843 context->ResetGraphics = rdpgfx_send_reset_graphics_pdu;
1844 context->StartFrame = rdpgfx_send_start_frame_pdu;
1845 context->EndFrame = rdpgfx_send_end_frame_pdu;
1846 context->SurfaceCommand = rdpgfx_send_surface_command;
1847 context->SurfaceFrameCommand = rdpgfx_send_surface_frame_command;
1848 context->DeleteEncodingContext = rdpgfx_send_delete_encoding_context_pdu;
1849 context->CreateSurface = rdpgfx_send_create_surface_pdu;
1850 context->DeleteSurface = rdpgfx_send_delete_surface_pdu;
1851 context->SolidFill = rdpgfx_send_solid_fill_pdu;
1852 context->SurfaceToSurface = rdpgfx_send_surface_to_surface_pdu;
1853 context->SurfaceToCache = rdpgfx_send_surface_to_cache_pdu;
1854 context->CacheToSurface = rdpgfx_send_cache_to_surface_pdu;
1855 context->CacheImportOffer = rdpgfx_process_cache_import_offer_pdu;
1856 context->CacheImportReply = rdpgfx_send_cache_import_reply_pdu;
1857 context->EvictCacheEntry = rdpgfx_send_evict_cache_entry_pdu;
1858 context->MapSurfaceToOutput = rdpgfx_send_map_surface_to_output_pdu;
1859 context->MapSurfaceToWindow = rdpgfx_send_map_surface_to_window_pdu;
1860 context->MapSurfaceToScaledOutput = rdpgfx_send_map_surface_to_scaled_output_pdu;
1861 context->MapSurfaceToScaledWindow = rdpgfx_send_map_surface_to_scaled_window_pdu;
1862 context->CapsAdvertise =
nullptr;
1863 context->CapsConfirm = rdpgfx_send_caps_confirm_pdu;
1864 context->FrameAcknowledge =
nullptr;
1865 context->QoeFrameAcknowledge =
nullptr;
1866 RdpgfxServerPrivate* priv = context->priv =
1867 (RdpgfxServerPrivate*)calloc(1,
sizeof(RdpgfxServerPrivate));
1872 priv->log = WLog_Get(TAG);
1877 priv->input_stream = Stream_New(
nullptr, 4);
1879 if (!priv->input_stream)
1881 WLog_Print(context->priv->log, WLOG_ERROR,
"Stream_New failed!");
1885 priv->isOpened = FALSE;
1886 priv->isReady = FALSE;
1887 priv->ownThread = TRUE;
1891 priv->activeCapSet = empty;
1896 WINPR_PRAGMA_DIAG_PUSH
1897 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
1898 rdpgfx_server_context_free(context);
1899 WINPR_PRAGMA_DIAG_POP
1903void rdpgfx_server_context_free(RdpgfxServerContext* context)
1908 (void)rdpgfx_server_close(context);
1911 Stream_Free(context->priv->input_stream, TRUE);
1913 free(context->priv);
1917HANDLE rdpgfx_server_get_event_handle(RdpgfxServerContext* context)
1923 return context->priv->channelEvent;
1935UINT rdpgfx_server_handle_messages(RdpgfxServerContext* context)
1937 DWORD BytesReturned = 0;
1938 void* buffer =
nullptr;
1939 UINT ret = CHANNEL_RC_OK;
1941 WINPR_ASSERT(context);
1942 WINPR_ASSERT(context->priv);
1944 RdpgfxServerPrivate* priv = context->priv;
1945 wStream* s = priv->input_stream;
1950 if (WTSVirtualChannelQuery(priv->rdpgfx_channel, WTSVirtualChannelReady, &buffer,
1951 &BytesReturned) == FALSE)
1953 if (GetLastError() == ERROR_NO_DATA)
1954 return ERROR_NO_DATA;
1956 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelQuery failed");
1957 return ERROR_INTERNAL_ERROR;
1960 priv->isReady = *((BOOL*)buffer);
1961 WTSFreeMemory(buffer);
1967 Stream_ResetPosition(s);
1969 if (!WTSVirtualChannelRead(priv->rdpgfx_channel, 0,
nullptr, 0, &BytesReturned))
1971 if (GetLastError() == ERROR_NO_DATA)
1972 return ERROR_NO_DATA;
1974 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelRead failed!");
1975 return ERROR_INTERNAL_ERROR;
1978 if (BytesReturned < 1)
1979 return CHANNEL_RC_OK;
1981 if (!Stream_EnsureRemainingCapacity(s, BytesReturned))
1983 WLog_Print(context->priv->log, WLOG_ERROR,
"Stream_EnsureRemainingCapacity failed!");
1984 return CHANNEL_RC_NO_MEMORY;
1987 const size_t len = Stream_Capacity(s);
1988 if (len > UINT32_MAX)
1989 return ERROR_INTERNAL_ERROR;
1990 if (WTSVirtualChannelRead(priv->rdpgfx_channel, 0, Stream_BufferAs(s,
char), (UINT32)len,
1991 &BytesReturned) == FALSE)
1993 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelRead failed!");
1994 return ERROR_INTERNAL_ERROR;
1997 if (!Stream_SetLength(s, BytesReturned))
1998 return ERROR_INTERNAL_ERROR;
2000 Stream_ResetPosition(s);
2002 while (Stream_GetPosition(s) < Stream_Length(s))
2004 if ((ret = rdpgfx_server_receive_pdu(context, s)))
2006 WLog_Print(context->priv->log, WLOG_ERROR,
2007 "rdpgfx_server_receive_pdu "
2008 "failed with error %" PRIu32
"!",