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 switch (cmd->codecId)
783#if defined(WITH_GFX_AV1)
784 case RDPGFX_CODECID_AV1:
786 case RDPGFX_CODECID_AVC420:
789 error = rdpgfx_write_h264_avc420(log, s, havc420);
791 if (error != CHANNEL_RC_OK)
793 WLog_Print(log, WLOG_ERROR,
"rdpgfx_write_h264_avc420 failed!");
799 case RDPGFX_CODECID_AVC444:
800 case RDPGFX_CODECID_AVC444v2:
803 havc420 = &(havc444->bitstream[0]);
804 if (!Stream_EnsureRemainingCapacity(s, 4))
805 return ERROR_INTERNAL_ERROR;
806 Stream_Write_UINT32(s, havc444->cbAvc420EncodedBitstream1 |
807 ((uint32_t)havc444->LC << 30UL));
809 error = rdpgfx_write_h264_avc420(log, s, havc420);
811 if (error != CHANNEL_RC_OK)
813 WLog_Print(log, WLOG_ERROR,
"rdpgfx_write_h264_avc420 failed!");
818 if (havc444->LC == 0)
820 havc420 = &(havc444->bitstream[1]);
821 error = rdpgfx_write_h264_avc420(log, s, havc420);
823 if (error != CHANNEL_RC_OK)
825 WLog_Print(log, WLOG_ERROR,
"rdpgfx_write_h264_avc420 failed!");
832 if (!Stream_EnsureRemainingCapacity(s, cmd->length))
833 return ERROR_INTERNAL_ERROR;
834 Stream_Write(s, cmd->data, cmd->length);
839 const size_t bitmapDataLength = Stream_GetPosition(s) - bitmapDataStart;
840 if (bitmapDataLength > UINT32_MAX)
841 return ERROR_INTERNAL_ERROR;
843 if (!Stream_SetPosition(s, bitmapDataStart -
sizeof(UINT32)))
844 return ERROR_INVALID_DATA;
845 if (!Stream_EnsureRemainingCapacity(s, 4))
846 return ERROR_INTERNAL_ERROR;
847 Stream_Write_UINT32(s, (UINT32)bitmapDataLength);
848 if (!Stream_SafeSeek(s, bitmapDataLength))
849 return ERROR_INTERNAL_ERROR;
862WINPR_ATTR_NODISCARD
static UINT rdpgfx_send_surface_command(RdpgfxServerContext* context,
865 if (!checkCapsAreExchanged(context))
866 return CHANNEL_RC_NOT_INITIALIZED;
867 UINT error = CHANNEL_RC_OK;
869 WINPR_ASSERT(context);
870 WINPR_ASSERT(context->priv);
874 rdpgfx_server_single_packet_new(context->priv->log, rdpgfx_surface_command_cmdid(cmd),
875 rdpgfx_estimate_surface_command(cmd));
879 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
880 return CHANNEL_RC_NO_MEMORY;
883 error = rdpgfx_write_surface_command(context->priv->log, s, cmd);
885 if (error != CHANNEL_RC_OK)
887 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_write_surface_command failed!");
891 return rdpgfx_server_single_packet_send(context, s);
893 Stream_Free(s, TRUE);
905WINPR_ATTR_NODISCARD
static UINT
911 if (!checkCapsAreExchanged(context))
912 return CHANNEL_RC_NOT_INITIALIZED;
914 WINPR_ASSERT(context->priv);
916 WINPR_ASSERT(startFrame);
917 WINPR_ASSERT(endFrame);
919 UINT error = CHANNEL_RC_OK;
920 size_t size = rdpgfx_pdu_length(rdpgfx_estimate_surface_command(cmd));
924 size += rdpgfx_pdu_length(RDPGFX_START_FRAME_PDU_SIZE);
929 size += rdpgfx_pdu_length(RDPGFX_END_FRAME_PDU_SIZE);
932 wStream* s = Stream_New(
nullptr, size);
936 WLog_Print(context->priv->log, WLOG_ERROR,
"Stream_New failed!");
937 return CHANNEL_RC_NO_MEMORY;
943 const size_t position = Stream_GetPosition(s);
944 error = rdpgfx_server_packet_init_header(s, RDPGFX_CMDID_STARTFRAME, 0);
946 if (error != CHANNEL_RC_OK)
948 WLog_Print(context->priv->log, WLOG_ERROR,
949 "Failed to init header with error %" PRIu32
"!", error);
953 if (!rdpgfx_write_start_frame_pdu(s, startFrame) ||
954 !rdpgfx_server_packet_complete_header(s, position))
960 const size_t pos = Stream_GetPosition(s);
961 error = rdpgfx_server_packet_init_header(s, rdpgfx_surface_command_cmdid(cmd),
964 if (error != CHANNEL_RC_OK)
966 WLog_Print(context->priv->log, WLOG_ERROR,
967 "Failed to init header with error %" PRIu32
"!", error);
971 error = rdpgfx_write_surface_command(context->priv->log, s, cmd);
973 if (error != CHANNEL_RC_OK)
975 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_write_surface_command failed!");
979 if (!rdpgfx_server_packet_complete_header(s, pos))
986 const size_t position = Stream_GetPosition(s);
987 error = rdpgfx_server_packet_init_header(s, RDPGFX_CMDID_ENDFRAME, 0);
989 if (error != CHANNEL_RC_OK)
991 WLog_Print(context->priv->log, WLOG_ERROR,
992 "Failed to init header with error %" PRIu32
"!", error);
996 if (!rdpgfx_write_end_frame_pdu(s, endFrame) ||
997 !rdpgfx_server_packet_complete_header(s, position))
1001 return rdpgfx_server_packet_send(context, s);
1003 Stream_Free(s, TRUE);
1012WINPR_ATTR_NODISCARD
static UINT
1013rdpgfx_send_delete_encoding_context_pdu(RdpgfxServerContext* context,
1016 if (!checkCapsAreExchanged(context))
1017 return CHANNEL_RC_NOT_INITIALIZED;
1019 WINPR_ASSERT(context->priv);
1022 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_DELETEENCODINGCONTEXT, 6);
1026 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1027 return CHANNEL_RC_NO_MEMORY;
1030 Stream_Write_UINT16(s, pdu->surfaceId);
1031 Stream_Write_UINT32(s, pdu->codecContextId);
1032 return rdpgfx_server_single_packet_send(context, s);
1040WINPR_ATTR_NODISCARD
static UINT rdpgfx_send_solid_fill_pdu(RdpgfxServerContext* context,
1043 if (!checkCapsAreExchanged(context))
1044 return CHANNEL_RC_NOT_INITIALIZED;
1046 WINPR_ASSERT(context->priv);
1049 UINT error = CHANNEL_RC_OK;
1051 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_SOLIDFILL,
1052 8 + 8 * pdu->fillRectCount);
1056 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1057 return CHANNEL_RC_NO_MEMORY;
1060 Stream_Write_UINT16(s, pdu->surfaceId);
1063 if ((error = rdpgfx_write_color32(s, &(pdu->fillPixel))))
1065 WLog_Print(context->priv->log, WLOG_ERROR,
1066 "rdpgfx_write_color32 failed with error %" PRIu32
"!", error);
1070 Stream_Write_UINT16(s, pdu->fillRectCount);
1072 for (UINT16 index = 0; index < pdu->fillRectCount; index++)
1074 fillRect = &(pdu->fillRects[index]);
1076 if ((error = rdpgfx_write_rect16(s, fillRect)))
1078 WLog_Print(context->priv->log, WLOG_ERROR,
1079 "rdpgfx_write_rect16 failed with error %" PRIu32
"!", error);
1084 return rdpgfx_server_single_packet_send(context, s);
1086 Stream_Free(s, TRUE);
1095WINPR_ATTR_NODISCARD
static UINT
1096rdpgfx_send_surface_to_surface_pdu(RdpgfxServerContext* context,
1099 if (!checkCapsAreExchanged(context))
1100 return CHANNEL_RC_NOT_INITIALIZED;
1103 WINPR_ASSERT(context->priv);
1105 UINT error = CHANNEL_RC_OK;
1107 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_SURFACETOSURFACE,
1108 14 + 4 * pdu->destPtsCount);
1112 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1113 return CHANNEL_RC_NO_MEMORY;
1116 Stream_Write_UINT16(s, pdu->surfaceIdSrc);
1117 Stream_Write_UINT16(s, pdu->surfaceIdDest);
1120 if ((error = rdpgfx_write_rect16(s, &(pdu->rectSrc))))
1122 WLog_Print(context->priv->log, WLOG_ERROR,
1123 "rdpgfx_write_rect16 failed with error %" PRIu32
"!", error);
1127 Stream_Write_UINT16(s, pdu->destPtsCount);
1129 for (UINT16 index = 0; index < pdu->destPtsCount; index++)
1131 destPt = &(pdu->destPts[index]);
1133 if ((error = rdpgfx_write_point16(s, destPt)))
1135 WLog_Print(context->priv->log, WLOG_ERROR,
1136 "rdpgfx_write_point16 failed with error %" PRIu32
"!", error);
1141 return rdpgfx_server_single_packet_send(context, s);
1143 Stream_Free(s, TRUE);
1152WINPR_ATTR_NODISCARD
static UINT
1153rdpgfx_send_surface_to_cache_pdu(RdpgfxServerContext* context,
1156 if (!checkCapsAreExchanged(context))
1157 return CHANNEL_RC_NOT_INITIALIZED;
1160 WINPR_ASSERT(context->priv);
1162 UINT error = CHANNEL_RC_OK;
1164 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_SURFACETOCACHE, 20);
1168 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1169 return CHANNEL_RC_NO_MEMORY;
1172 Stream_Write_UINT16(s, pdu->surfaceId);
1173 Stream_Write_UINT64(s, pdu->cacheKey);
1174 Stream_Write_UINT16(s, pdu->cacheSlot);
1177 if ((error = rdpgfx_write_rect16(s, &(pdu->rectSrc))))
1179 WLog_Print(context->priv->log, WLOG_ERROR,
1180 "rdpgfx_write_rect16 failed with error %" PRIu32
"!", error);
1184 return rdpgfx_server_single_packet_send(context, s);
1186 Stream_Free(s, TRUE);
1195WINPR_ATTR_NODISCARD
static UINT
1196rdpgfx_send_cache_to_surface_pdu(RdpgfxServerContext* context,
1199 if (!checkCapsAreExchanged(context))
1200 return CHANNEL_RC_NOT_INITIALIZED;
1203 WINPR_ASSERT(context->priv);
1205 UINT error = CHANNEL_RC_OK;
1207 wStream* s = rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_CACHETOSURFACE,
1208 6 + 4 * pdu->destPtsCount);
1212 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1213 return CHANNEL_RC_NO_MEMORY;
1216 Stream_Write_UINT16(s, pdu->cacheSlot);
1217 Stream_Write_UINT16(s, pdu->surfaceId);
1218 Stream_Write_UINT16(s, pdu->destPtsCount);
1220 for (UINT16 index = 0; index < pdu->destPtsCount; index++)
1222 destPt = &(pdu->destPts[index]);
1224 if ((error = rdpgfx_write_point16(s, destPt)))
1226 WLog_Print(context->priv->log, WLOG_ERROR,
1227 "rdpgfx_write_point16 failed with error %" PRIu32
"", error);
1232 return rdpgfx_server_single_packet_send(context, s);
1234 Stream_Free(s, TRUE);
1243WINPR_ATTR_NODISCARD
static UINT
1244rdpgfx_send_map_surface_to_output_pdu(RdpgfxServerContext* context,
1247 if (!checkCapsAreExchanged(context))
1248 return CHANNEL_RC_NOT_INITIALIZED;
1251 WINPR_ASSERT(context->priv);
1253 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_MAPSURFACETOOUTPUT, 12);
1257 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1258 return CHANNEL_RC_NO_MEMORY;
1261 Stream_Write_UINT16(s, pdu->surfaceId);
1262 Stream_Write_UINT16(s, 0);
1263 Stream_Write_UINT32(s, pdu->outputOriginX);
1264 Stream_Write_UINT32(s, pdu->outputOriginY);
1265 return rdpgfx_server_single_packet_send(context, s);
1273WINPR_ATTR_NODISCARD
static UINT
1274rdpgfx_send_map_surface_to_window_pdu(RdpgfxServerContext* context,
1277 if (!checkCapsAreExchanged(context))
1278 return CHANNEL_RC_NOT_INITIALIZED;
1281 WINPR_ASSERT(context->priv);
1284 rdpgfx_server_single_packet_new(context->priv->log, RDPGFX_CMDID_MAPSURFACETOWINDOW, 18);
1288 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1289 return CHANNEL_RC_NO_MEMORY;
1292 Stream_Write_UINT16(s, pdu->surfaceId);
1293 Stream_Write_UINT64(s, pdu->windowId);
1294 Stream_Write_UINT32(s, pdu->mappedWidth);
1295 Stream_Write_UINT32(s, pdu->mappedHeight);
1296 return rdpgfx_server_single_packet_send(context, s);
1299WINPR_ATTR_NODISCARD
static UINT
1300rdpgfx_send_map_surface_to_scaled_window_pdu(RdpgfxServerContext* context,
1303 if (!checkCapsAreExchanged(context))
1304 return CHANNEL_RC_NOT_INITIALIZED;
1307 WINPR_ASSERT(context->priv);
1308 wStream* s = rdpgfx_server_single_packet_new(context->priv->log,
1309 RDPGFX_CMDID_MAPSURFACETOSCALEDWINDOW, 26);
1313 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1314 return CHANNEL_RC_NO_MEMORY;
1317 Stream_Write_UINT16(s, pdu->surfaceId);
1318 Stream_Write_UINT64(s, pdu->windowId);
1319 Stream_Write_UINT32(s, pdu->mappedWidth);
1320 Stream_Write_UINT32(s, pdu->mappedHeight);
1321 Stream_Write_UINT32(s, pdu->targetWidth);
1322 Stream_Write_UINT32(s, pdu->targetHeight);
1323 return rdpgfx_server_single_packet_send(context, s);
1331WINPR_ATTR_NODISCARD
static UINT rdpgfx_recv_frame_acknowledge_pdu(RdpgfxServerContext* context,
1334 WINPR_ASSERT(context);
1336 if (!checkCapsAreExchanged(context))
1337 return CHANNEL_RC_NOT_INITIALIZED;
1339 WINPR_ASSERT(context->priv);
1342 UINT error = CHANNEL_RC_OK;
1344 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 12))
1345 return ERROR_INVALID_DATA;
1347 Stream_Read_UINT32(s, pdu.queueDepth);
1348 Stream_Read_UINT32(s, pdu.frameId);
1349 Stream_Read_UINT32(s, pdu.totalFramesDecoded);
1351 IFCALLRET(context->FrameAcknowledge, error, context, &pdu);
1354 WLog_Print(context->priv->log, WLOG_ERROR,
1355 "context->FrameAcknowledge failed with error %" PRIu32
"", error);
1365WINPR_ATTR_NODISCARD
static UINT rdpgfx_recv_cache_import_offer_pdu(RdpgfxServerContext* context,
1368 WINPR_ASSERT(context);
1369 if (!checkCapsAreExchanged(context))
1370 return CHANNEL_RC_NOT_INITIALIZED;
1372 WINPR_ASSERT(context->priv);
1376 UINT error = CHANNEL_RC_OK;
1378 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 2))
1379 return ERROR_INVALID_DATA;
1382 Stream_Read_UINT16(s, pdu.cacheEntriesCount);
1385 if (pdu.cacheEntriesCount >= 5462)
1387 WLog_Print(context->priv->log, WLOG_ERROR,
"Invalid cacheEntriesCount: %" PRIu16
"",
1388 pdu.cacheEntriesCount);
1389 return ERROR_INVALID_DATA;
1392 if (!Stream_CheckAndLogRequiredLengthOfSizeWLog(context->priv->log, s, pdu.cacheEntriesCount,
1394 return ERROR_INVALID_DATA;
1396 for (UINT16 index = 0; index < pdu.cacheEntriesCount; index++)
1398 cacheEntry = &(pdu.cacheEntries[index]);
1399 Stream_Read_UINT64(s, cacheEntry->cacheKey);
1400 Stream_Read_UINT32(s, cacheEntry->bitmapLength);
1403 IFCALLRET(context->CacheImportOffer, error, context, &pdu);
1406 WLog_Print(context->priv->log, WLOG_ERROR,
1407 "context->CacheImportOffer failed with error %" PRIu32
"", error);
1417WINPR_ATTR_NODISCARD
static UINT rdpgfx_recv_caps_advertise_pdu(RdpgfxServerContext* context,
1422 UINT error = ERROR_INVALID_DATA;
1425 return ERROR_BAD_ARGUMENTS;
1427 WINPR_ASSERT(context->priv);
1428 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 2))
1429 return ERROR_INVALID_DATA;
1431 Stream_Read_UINT16(s, pdu.capsSetCount);
1432 if (pdu.capsSetCount > 0)
1434 capsSets = calloc(pdu.capsSetCount, (RDPGFX_CAPSET_BASE_SIZE + 4));
1436 return ERROR_OUTOFMEMORY;
1439 pdu.capsSets = capsSets;
1441 for (UINT16 index = 0; index < pdu.capsSetCount; index++)
1445 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 8))
1448 Stream_Read_UINT32(s, capsSet->version);
1449 Stream_Read_UINT32(s, capsSet->length);
1451 if (capsSet->length >= 4)
1453 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 4))
1456 Stream_Peek_UINT32(s, capsSet->flags);
1459 if (!Stream_SafeSeek(s, capsSet->length))
1463 error = ERROR_BAD_CONFIGURATION;
1464 IFCALLRET(context->CapsAdvertise, error, context, &pdu);
1467 WLog_Print(context->priv->log, WLOG_ERROR,
1468 "context->CapsAdvertise failed with error %" PRIu32
"", error);
1480WINPR_ATTR_NODISCARD
static UINT rdpgfx_recv_qoe_frame_acknowledge_pdu(RdpgfxServerContext* context,
1483 WINPR_ASSERT(context);
1485 if (!checkCapsAreExchanged(context))
1486 return CHANNEL_RC_NOT_INITIALIZED;
1489 UINT error = CHANNEL_RC_OK;
1491 WINPR_ASSERT(context->priv);
1493 if (!Stream_CheckAndLogRequiredLengthWLog(context->priv->log, s, 12))
1494 return ERROR_INVALID_DATA;
1496 Stream_Read_UINT32(s, pdu.frameId);
1497 Stream_Read_UINT32(s, pdu.timestamp);
1498 Stream_Read_UINT16(s, pdu.timeDiffSE);
1499 Stream_Read_UINT16(s, pdu.timeDiffEDR);
1501 IFCALLRET(context->QoeFrameAcknowledge, error, context, &pdu);
1504 WLog_Print(context->priv->log, WLOG_ERROR,
1505 "context->QoeFrameAcknowledge failed with error %" PRIu32
"", error);
1510WINPR_ATTR_NODISCARD
static UINT
1511rdpgfx_send_map_surface_to_scaled_output_pdu(RdpgfxServerContext* context,
1514 if (!checkCapsAreExchanged(context))
1515 return CHANNEL_RC_NOT_INITIALIZED;
1518 WINPR_ASSERT(context->priv);
1519 wStream* s = rdpgfx_server_single_packet_new(context->priv->log,
1520 RDPGFX_CMDID_MAPSURFACETOSCALEDOUTPUT, 20);
1524 WLog_Print(context->priv->log, WLOG_ERROR,
"rdpgfx_server_single_packet_new failed!");
1525 return CHANNEL_RC_NO_MEMORY;
1528 Stream_Write_UINT16(s, pdu->surfaceId);
1529 Stream_Write_UINT16(s, 0);
1530 Stream_Write_UINT32(s, pdu->outputOriginX);
1531 Stream_Write_UINT32(s, pdu->outputOriginY);
1532 Stream_Write_UINT32(s, pdu->targetWidth);
1533 Stream_Write_UINT32(s, pdu->targetHeight);
1534 return rdpgfx_server_single_packet_send(context, s);
1542WINPR_ATTR_NODISCARD
static UINT rdpgfx_server_receive_pdu(RdpgfxServerContext* context,
wStream* s)
1546 UINT error = CHANNEL_RC_OK;
1547 size_t beg = Stream_GetPosition(s);
1549 WINPR_ASSERT(context);
1550 WINPR_ASSERT(context->priv);
1552 if ((error = rdpgfx_read_header(context->priv->log, s, &header)))
1554 WLog_Print(context->priv->log, WLOG_ERROR,
1555 "rdpgfx_read_header failed with error %" PRIu32
"!", error);
1559 WLog_Print(context->priv->log, WLOG_TRACE,
1560 "cmdId: %s (0x%04" PRIX16
") flags: 0x%04" PRIX16
" pduLength: %" PRIu32
"",
1561 rdpgfx_get_cmd_id_string(header.cmdId), header.cmdId, header.flags,
1564 switch (header.cmdId)
1566 case RDPGFX_CMDID_FRAMEACKNOWLEDGE:
1567 if ((error = rdpgfx_recv_frame_acknowledge_pdu(context, s)))
1568 WLog_Print(context->priv->log, WLOG_ERROR,
1569 "rdpgfx_recv_frame_acknowledge_pdu "
1570 "failed with error %" PRIu32
"!",
1575 case RDPGFX_CMDID_CACHEIMPORTOFFER:
1576 if ((error = rdpgfx_recv_cache_import_offer_pdu(context, s)))
1577 WLog_Print(context->priv->log, WLOG_ERROR,
1578 "rdpgfx_recv_cache_import_offer_pdu "
1579 "failed with error %" PRIu32
"!",
1584 case RDPGFX_CMDID_CAPSADVERTISE:
1585 if ((error = rdpgfx_recv_caps_advertise_pdu(context, s)))
1586 WLog_Print(context->priv->log, WLOG_ERROR,
1587 "rdpgfx_recv_caps_advertise_pdu "
1588 "failed with error %" PRIu32
"!",
1593 case RDPGFX_CMDID_QOEFRAMEACKNOWLEDGE:
1594 if ((error = rdpgfx_recv_qoe_frame_acknowledge_pdu(context, s)))
1595 WLog_Print(context->priv->log, WLOG_ERROR,
1596 "rdpgfx_recv_qoe_frame_acknowledge_pdu "
1597 "failed with error %" PRIu32
"!",
1603 error = CHANNEL_RC_BAD_PROC;
1609 WLog_Print(context->priv->log, WLOG_ERROR,
1610 "Error while parsing GFX cmdId: %s (0x%04" PRIX16
")",
1611 rdpgfx_get_cmd_id_string(header.cmdId), header.cmdId);
1615 end = Stream_GetPosition(s);
1617 if (end != (beg + header.pduLength))
1619 WLog_Print(context->priv->log, WLOG_ERROR,
1620 "Unexpected gfx pdu end: Actual: %" PRIuz
", Expected: %" PRIuz
"", end,
1621 (beg + header.pduLength));
1622 if (!Stream_SetPosition(s, (beg + header.pduLength)))
1623 return ERROR_INVALID_DATA;
1629WINPR_ATTR_NODISCARD
static DWORD WINAPI rdpgfx_server_thread_func(LPVOID arg)
1631 RdpgfxServerContext* context = (RdpgfxServerContext*)arg;
1632 WINPR_ASSERT(context);
1634 RdpgfxServerPrivate* priv = context->priv;
1637 HANDLE events[8] = WINPR_C_ARRAY_INIT;
1638 UINT error = CHANNEL_RC_OK;
1642 if (priv->ownThread)
1644 WINPR_ASSERT(priv->stopEvent);
1645 events[nCount++] = priv->stopEvent;
1648 WINPR_ASSERT(priv->channelEvent);
1649 events[nCount++] = priv->channelEvent;
1654 status = WaitForMultipleObjects(nCount, events, FALSE, INFINITE);
1656 if (status == WAIT_FAILED)
1658 error = GetLastError();
1659 WLog_Print(context->priv->log, WLOG_ERROR,
1660 "WaitForMultipleObjects failed with error %" PRIu32
"", error);
1665 if (status == WAIT_OBJECT_0)
1668 if ((error = rdpgfx_server_handle_messages(context)))
1670 WLog_Print(context->priv->log, WLOG_ERROR,
1671 "rdpgfx_server_handle_messages failed with error %" PRIu32
"", error);
1676 if (error && context->rdpcontext)
1677 setChannelError(context->rdpcontext, error,
"rdpgfx_server_thread_func reported an error");
1683WINPR_ATTR_NODISCARD
static BOOL rdpgfx_server_open(RdpgfxServerContext* context)
1685 WINPR_ASSERT(context);
1686 RdpgfxServerPrivate* priv = context->priv;
1687 void* buffer =
nullptr;
1691 if (!priv->isOpened)
1693 PULONG pSessionId =
nullptr;
1694 DWORD BytesReturned = 0;
1695 priv->SessionId = WTS_CURRENT_SESSION;
1696 UINT32 channelId = 0;
1699 if (WTSQuerySessionInformationA(context->vcm, WTS_CURRENT_SESSION, WTSSessionId,
1700 (LPSTR*)&pSessionId, &BytesReturned) == FALSE)
1702 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSQuerySessionInformationA failed!");
1706 priv->SessionId = (DWORD)*pSessionId;
1707 WTSFreeMemory(pSessionId);
1708 priv->rdpgfx_channel = WTSVirtualChannelOpenEx(priv->SessionId, RDPGFX_DVC_CHANNEL_NAME,
1709 WTS_CHANNEL_OPTION_DYNAMIC);
1711 if (!priv->rdpgfx_channel)
1713 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelOpenEx failed!");
1717 channelId = WTSChannelGetIdByHandle(priv->rdpgfx_channel);
1719 IFCALLRET(context->ChannelIdAssigned, status, context, channelId);
1722 WLog_Print(context->priv->log, WLOG_ERROR,
"context->ChannelIdAssigned failed!");
1727 if (!WTSVirtualChannelQuery(priv->rdpgfx_channel, WTSVirtualEventHandle, &buffer,
1729 (BytesReturned !=
sizeof(HANDLE)))
1731 WLog_Print(context->priv->log, WLOG_ERROR,
1732 "WTSVirtualChannelQuery failed "
1733 "or invalid returned size(%" PRIu32
")",
1737 WTSFreeMemory(buffer);
1742 priv->channelEvent = *(HANDLE*)buffer;
1743 WTSFreeMemory(buffer);
1745 if (!(priv->zgfx = zgfx_context_new(TRUE)))
1747 WLog_Print(context->priv->log, WLOG_ERROR,
"Create zgfx context failed!");
1751 priv->isReady = FALSE;
1753 priv->activeCapSet = empty;
1754 if (priv->ownThread)
1756 if (!(priv->stopEvent = CreateEvent(
nullptr, TRUE, FALSE,
nullptr)))
1758 WLog_Print(context->priv->log, WLOG_ERROR,
"CreateEvent failed!");
1762 if (!(priv->thread = CreateThread(
nullptr, 0, rdpgfx_server_thread_func, (
void*)context,
1765 WLog_Print(context->priv->log, WLOG_ERROR,
"CreateThread failed!");
1770 priv->isOpened = TRUE;
1774 WLog_Print(context->priv->log, WLOG_ERROR,
"RDPGFX channel is already opened!");
1777 (void)rdpgfx_server_close(context);
1781BOOL rdpgfx_server_close(RdpgfxServerContext* context)
1783 WINPR_ASSERT(context);
1785 RdpgfxServerPrivate* priv = context->priv;
1788 if (priv->ownThread && priv->thread)
1790 (void)SetEvent(priv->stopEvent);
1792 if (WaitForSingleObject(priv->thread, INFINITE) == WAIT_FAILED)
1794 WLog_Print(context->priv->log, WLOG_ERROR,
1795 "WaitForSingleObject failed with error %" PRIu32
"", GetLastError());
1799 (void)CloseHandle(priv->thread);
1800 (void)CloseHandle(priv->stopEvent);
1801 priv->thread =
nullptr;
1802 priv->stopEvent =
nullptr;
1805 zgfx_context_free(priv->zgfx);
1806 priv->zgfx =
nullptr;
1808 if (priv->rdpgfx_channel)
1810 (void)WTSVirtualChannelClose(priv->rdpgfx_channel);
1811 priv->rdpgfx_channel =
nullptr;
1814 priv->channelEvent =
nullptr;
1815 priv->isOpened = FALSE;
1816 priv->isReady = FALSE;
1818 priv->activeCapSet = empty;
1822WINPR_ATTR_NODISCARD
static BOOL rdpgfx_server_initialize(RdpgfxServerContext* context,
1823 BOOL externalThread)
1825 WINPR_ASSERT(context);
1826 WINPR_ASSERT(context->priv);
1828 if (context->priv->isOpened)
1830 WLog_Print(context->priv->log, WLOG_WARN,
1831 "Application error: RDPEGFX channel already initialized, "
1832 "calling in this state is not possible!");
1836 context->priv->ownThread = !externalThread;
1840RdpgfxServerContext* rdpgfx_server_context_new(HANDLE vcm)
1842 RdpgfxServerContext* context = (RdpgfxServerContext*)calloc(1,
sizeof(RdpgfxServerContext));
1848 context->Initialize = rdpgfx_server_initialize;
1849 context->Open = rdpgfx_server_open;
1850 context->Close = rdpgfx_server_close;
1851 context->ResetGraphics = rdpgfx_send_reset_graphics_pdu;
1852 context->StartFrame = rdpgfx_send_start_frame_pdu;
1853 context->EndFrame = rdpgfx_send_end_frame_pdu;
1854 context->SurfaceCommand = rdpgfx_send_surface_command;
1855 context->SurfaceFrameCommand = rdpgfx_send_surface_frame_command;
1856 context->DeleteEncodingContext = rdpgfx_send_delete_encoding_context_pdu;
1857 context->CreateSurface = rdpgfx_send_create_surface_pdu;
1858 context->DeleteSurface = rdpgfx_send_delete_surface_pdu;
1859 context->SolidFill = rdpgfx_send_solid_fill_pdu;
1860 context->SurfaceToSurface = rdpgfx_send_surface_to_surface_pdu;
1861 context->SurfaceToCache = rdpgfx_send_surface_to_cache_pdu;
1862 context->CacheToSurface = rdpgfx_send_cache_to_surface_pdu;
1863 context->CacheImportOffer = rdpgfx_process_cache_import_offer_pdu;
1864 context->CacheImportReply = rdpgfx_send_cache_import_reply_pdu;
1865 context->EvictCacheEntry = rdpgfx_send_evict_cache_entry_pdu;
1866 context->MapSurfaceToOutput = rdpgfx_send_map_surface_to_output_pdu;
1867 context->MapSurfaceToWindow = rdpgfx_send_map_surface_to_window_pdu;
1868 context->MapSurfaceToScaledOutput = rdpgfx_send_map_surface_to_scaled_output_pdu;
1869 context->MapSurfaceToScaledWindow = rdpgfx_send_map_surface_to_scaled_window_pdu;
1870 context->CapsAdvertise =
nullptr;
1871 context->CapsConfirm = rdpgfx_send_caps_confirm_pdu;
1872 context->FrameAcknowledge =
nullptr;
1873 context->QoeFrameAcknowledge =
nullptr;
1874 RdpgfxServerPrivate* priv = context->priv =
1875 (RdpgfxServerPrivate*)calloc(1,
sizeof(RdpgfxServerPrivate));
1880 priv->log = WLog_Get(TAG);
1885 priv->input_stream = Stream_New(
nullptr, 4);
1887 if (!priv->input_stream)
1889 WLog_Print(context->priv->log, WLOG_ERROR,
"Stream_New failed!");
1893 priv->isOpened = FALSE;
1894 priv->isReady = FALSE;
1895 priv->ownThread = TRUE;
1899 priv->activeCapSet = empty;
1904 WINPR_PRAGMA_DIAG_PUSH
1905 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
1906 rdpgfx_server_context_free(context);
1907 WINPR_PRAGMA_DIAG_POP
1911void rdpgfx_server_context_free(RdpgfxServerContext* context)
1916 (void)rdpgfx_server_close(context);
1919 Stream_Free(context->priv->input_stream, TRUE);
1921 free(context->priv);
1925HANDLE rdpgfx_server_get_event_handle(RdpgfxServerContext* context)
1931 return context->priv->channelEvent;
1943UINT rdpgfx_server_handle_messages(RdpgfxServerContext* context)
1945 DWORD BytesReturned = 0;
1946 void* buffer =
nullptr;
1947 UINT ret = CHANNEL_RC_OK;
1949 WINPR_ASSERT(context);
1950 WINPR_ASSERT(context->priv);
1952 RdpgfxServerPrivate* priv = context->priv;
1953 wStream* s = priv->input_stream;
1958 if (WTSVirtualChannelQuery(priv->rdpgfx_channel, WTSVirtualChannelReady, &buffer,
1959 &BytesReturned) == FALSE)
1961 if (GetLastError() == ERROR_NO_DATA)
1962 return ERROR_NO_DATA;
1964 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelQuery failed");
1965 return ERROR_INTERNAL_ERROR;
1968 priv->isReady = *((BOOL*)buffer);
1969 WTSFreeMemory(buffer);
1975 Stream_ResetPosition(s);
1977 if (!WTSVirtualChannelRead(priv->rdpgfx_channel, 0,
nullptr, 0, &BytesReturned))
1979 if (GetLastError() == ERROR_NO_DATA)
1980 return ERROR_NO_DATA;
1982 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelRead failed!");
1983 return ERROR_INTERNAL_ERROR;
1986 if (BytesReturned < 1)
1987 return CHANNEL_RC_OK;
1989 if (!Stream_EnsureRemainingCapacity(s, BytesReturned))
1991 WLog_Print(context->priv->log, WLOG_ERROR,
"Stream_EnsureRemainingCapacity failed!");
1992 return CHANNEL_RC_NO_MEMORY;
1995 const size_t len = Stream_Capacity(s);
1996 if (len > UINT32_MAX)
1997 return ERROR_INTERNAL_ERROR;
1998 if (WTSVirtualChannelRead(priv->rdpgfx_channel, 0, Stream_BufferAs(s,
char), (UINT32)len,
1999 &BytesReturned) == FALSE)
2001 WLog_Print(context->priv->log, WLOG_ERROR,
"WTSVirtualChannelRead failed!");
2002 return ERROR_INTERNAL_ERROR;
2005 if (!Stream_SetLength(s, BytesReturned))
2006 return ERROR_INTERNAL_ERROR;
2008 Stream_ResetPosition(s);
2010 while (Stream_GetPosition(s) < Stream_Length(s))
2012 if ((ret = rdpgfx_server_receive_pdu(context, s)))
2014 WLog_Print(context->priv->log, WLOG_ERROR,
2015 "rdpgfx_server_receive_pdu "
2016 "failed with error %" PRIu32
"!",