21#include <freerdp/config.h>
24#include <winpr/assert.h>
25#include <winpr/cast.h>
26#include <winpr/file.h>
27#include <winpr/path.h>
28#include <winpr/synch.h>
29#include <winpr/thread.h>
30#include <winpr/sysinfo.h>
31#include <winpr/interlocked.h>
33#include <freerdp/log.h>
34#include <freerdp/utils/gfx.h>
35#include <freerdp/channels/drdynvc.h>
39#define TAG CLIENT_TAG("shadow")
44 BOOL gfxSurfaceCreated;
48static BOOL shadow_avc420_enabled(
const rdpShadowClient* client);
50static BOOL shadow_avc444_enabled(
const rdpShadowClient* client);
59#define BitmapUpdateProxy(client, bitmap) \
60 BitmapUpdateProxyEx((client), (bitmap), __FILE__, __LINE__, __func__)
62static BOOL BitmapUpdateProxyEx(rdpShadowClient* client,
const BITMAP_UPDATE* bitmap,
63 const char* file,
size_t line,
const char* fkt)
68 rdpShadowServer* server = client->server;
71 rdpContext* context = (rdpContext*)client;
73 rdpUpdate* update = context->update;
76 if (server->SupportMultiRectBitmapUpdates)
78 const BOOL rc = IFCALLRESULT(FALSE, update->BitmapUpdate, context, bitmap);
81 const DWORD log_level = WLOG_ERROR;
82 wLog* log = WLog_Get(TAG);
83 if (WLog_IsLevelActive(log, log_level))
85 WLog_PrintTextMessage(log, log_level, line, file, fkt,
86 "BitmapUpdate[count %" PRIu32
"] failed", bitmap->number);
93 for (UINT32 x = 0; x < bitmap->number; x++)
99 cur.skipCompression = bitmap->skipCompression;
100 const BOOL rc = IFCALLRESULT(FALSE, update->BitmapUpdate, context, &cur);
103 const DWORD log_level = WLOG_ERROR;
104 wLog* log = WLog_Get(TAG);
105 if (WLog_IsLevelActive(log, log_level))
107 WLog_PrintTextMessage(log, log_level, line, file, fkt,
108 "BitmapUpdate[count 1, at %" PRIu32
"] failed", x);
119static inline BOOL shadow_client_rdpgfx_new_surface(rdpShadowClient* client)
121 UINT error = CHANNEL_RC_OK;
124 RdpgfxServerContext* context =
nullptr;
125 rdpSettings* settings =
nullptr;
127 WINPR_ASSERT(client);
128 context = client->rdpgfx;
129 WINPR_ASSERT(context);
130 settings = ((rdpContext*)client)->settings;
131 WINPR_ASSERT(settings);
137 createSurface.pixelFormat = GFX_PIXEL_FORMAT_XRGB_8888;
138 createSurface.surfaceId = client->surfaceId;
139 surfaceToOutput.outputOriginX = 0;
140 surfaceToOutput.outputOriginY = 0;
141 surfaceToOutput.surfaceId = client->surfaceId;
142 surfaceToOutput.reserved = 0;
143 IFCALLRET(context->CreateSurface, error, context, &createSurface);
147 WLog_ERR(TAG,
"CreateSurface failed with error %" PRIu32
"", error);
151 IFCALLRET(context->MapSurfaceToOutput, error, context, &surfaceToOutput);
155 WLog_ERR(TAG,
"MapSurfaceToOutput failed with error %" PRIu32
"", error);
163static inline BOOL shadow_client_rdpgfx_release_surface(rdpShadowClient* client)
165 UINT error = CHANNEL_RC_OK;
167 RdpgfxServerContext* context =
nullptr;
169 WINPR_ASSERT(client);
171 context = client->rdpgfx;
172 WINPR_ASSERT(context);
174 pdu.surfaceId = client->surfaceId++;
175 IFCALLRET(context->DeleteSurface, error, context, &pdu);
179 WLog_ERR(TAG,
"DeleteSurface failed with error %" PRIu32
"", error);
187static inline BOOL shadow_client_rdpgfx_reset_graphic(rdpShadowClient* client)
189 UINT error = CHANNEL_RC_OK;
191 RdpgfxServerContext* context =
nullptr;
192 rdpSettings* settings =
nullptr;
194 WINPR_ASSERT(client);
195 WINPR_ASSERT(client->rdpgfx);
197 context = client->rdpgfx;
198 WINPR_ASSERT(context);
200 settings = client->context.settings;
201 WINPR_ASSERT(settings);
205 pdu.monitorCount = client->subsystem->numMonitors;
206 pdu.monitorDefArray = client->subsystem->monitors;
207 IFCALLRET(context->ResetGraphics, error, context, &pdu);
211 WLog_ERR(TAG,
"ResetGraphics failed with error %" PRIu32
"", error);
215 client->first_frame = TRUE;
219static inline void shadow_client_free_queued_message(
void* obj)
221 wMessage* message = (wMessage*)obj;
223 WINPR_ASSERT(message);
226 message->Free(message);
227 message->Free =
nullptr;
231static void shadow_client_context_free(freerdp_peer* peer, rdpContext* context)
233 rdpShadowClient* client = (rdpShadowClient*)context;
234 rdpShadowServer* server =
nullptr;
240 server = client->server;
241 if (server && server->clients)
242 ArrayList_Remove(server->clients, (
void*)client);
244 shadow_encoder_free(client->encoder);
247 MessageQueue_Free(client->MsgQueue);
248 WTSCloseServer(client->vcm);
249 region16_uninit(&(client->invalidRegion));
250 DeleteCriticalSection(&(client->lock));
252 client->MsgQueue =
nullptr;
253 client->encoder =
nullptr;
254 client->vcm =
nullptr;
258static BOOL shadow_client_context_new(freerdp_peer* peer, rdpContext* context)
261 const char bind_address[] =
"bind-address,";
262 rdpShadowClient* client = (rdpShadowClient*)context;
263 rdpSettings* settings =
nullptr;
264 const rdpSettings* srvSettings =
nullptr;
265 rdpShadowServer* server =
nullptr;
266 const wObject cb = {
nullptr,
nullptr,
nullptr, shadow_client_free_queued_message,
nullptr };
268 WINPR_ASSERT(client);
270 WINPR_ASSERT(peer->context);
272 server = (rdpShadowServer*)peer->ContextExtra;
273 WINPR_ASSERT(server);
275 srvSettings = server->settings;
276 WINPR_ASSERT(srvSettings);
278 client->surfaceId = 1;
279 client->server = server;
280 client->subsystem = server->subsystem;
281 WINPR_ASSERT(client->subsystem);
283 settings = peer->context->settings;
284 WINPR_ASSERT(settings);
296 settings, FreeRDP_RemoteFxRlgrMode,
306 settings, FreeRDP_SupportGraphicsPipeline,
321 if (server->ipcSocket && (strncmp(bind_address, server->ipcSocket,
322 strnlen(bind_address,
sizeof(bind_address))) != 0))
330 client->inLobby = TRUE;
331 client->mayView = server->mayView;
332 client->mayInteract = server->mayInteract;
334 if (!InitializeCriticalSectionAndSpinCount(&(client->lock), 4000))
337 region16_init(&(client->invalidRegion));
338 client->vcm = WTSOpenServerA((LPSTR)peer->context);
340 if (!client->vcm || client->vcm == INVALID_HANDLE_VALUE)
343 if (!(client->MsgQueue = MessageQueue_New(&cb)))
346 if (!(client->encoder = shadow_encoder_new(client)))
349 if (!ArrayList_Append(server->clients, (
void*)client))
355 shadow_client_context_free(peer, context);
359static inline void shadow_client_mark_invalid(rdpShadowClient* client, UINT32 numRects,
362 WINPR_ASSERT(client);
363 WINPR_ASSERT(rects || (numRects == 0));
365 rdpSettings* settings = client->context.settings;
366 WINPR_ASSERT(settings);
368 EnterCriticalSection(&(client->lock));
373 for (UINT32 index = 0; index < numRects; index++)
375 if (!region16_union_rect(&(client->invalidRegion), &(client->invalidRegion),
387 if (!region16_union_rect(&(client->invalidRegion), &(client->invalidRegion), &screenRegion))
391 LeaveCriticalSection(&(client->lock));
401static inline BOOL shadow_client_recalc_desktop_size(rdpShadowClient* client)
405 rdpShadowServer* server =
nullptr;
406 rdpSettings* settings =
nullptr;
409 WINPR_ASSERT(client);
410 server = client->server;
411 settings = client->context.settings;
413 WINPR_ASSERT(server);
414 WINPR_ASSERT(server->surface);
415 WINPR_ASSERT(settings);
417 WINPR_ASSERT(server->surface->width <= UINT16_MAX);
418 WINPR_ASSERT(server->surface->height <= UINT16_MAX);
419 viewport.right = (UINT16)server->surface->width;
420 viewport.bottom = (UINT16)server->surface->height;
422 if (server->shareSubRect)
424 if (!rectangles_intersection(&viewport, &(server->subRect), &viewport))
428 width = viewport.right - viewport.left;
429 height = viewport.bottom - viewport.top;
431 WINPR_ASSERT(width >= 0);
432 WINPR_ASSERT(width <= UINT16_MAX);
433 WINPR_ASSERT(height >= 0);
434 WINPR_ASSERT(height <= UINT16_MAX);
440static BOOL shadow_client_capabilities(freerdp_peer* peer)
442 rdpShadowSubsystem* subsystem =
nullptr;
443 rdpShadowClient* client =
nullptr;
448 client = (rdpShadowClient*)peer->context;
449 WINPR_ASSERT(client);
450 WINPR_ASSERT(client->server);
452 subsystem = client->server->subsystem;
453 WINPR_ASSERT(subsystem);
455 IFCALLRET(subsystem->ClientCapabilities, ret, subsystem, client);
458 WLog_WARN(TAG,
"subsystem->ClientCapabilities failed");
463static void shadow_reset_desktop_resize(rdpShadowClient* client)
465 WINPR_ASSERT(client);
466 client->resizeRequested = FALSE;
470static BOOL shadow_send_desktop_resize(rdpShadowClient* client)
473 rdpUpdate* update =
nullptr;
474 rdpSettings* settings =
nullptr;
475 const freerdp_peer* peer =
nullptr;
477 WINPR_ASSERT(client);
479 settings = client->context.settings;
480 peer = client->context.peer;
482 WINPR_ASSERT(client->server);
483 WINPR_ASSERT(client->server->surface);
485 const UINT32 resizeWidth = client->server->surface->width;
486 const UINT32 resizeHeight = client->server->surface->height;
488 if (client->resizeRequested)
490 if ((resizeWidth == client->resizeWidth) && (resizeHeight == client->resizeHeight))
495 "detected previous resize request for resolution %" PRIu32
"x%" PRIu32
496 ", still have %" PRIu32
"x%" PRIu32
", disconnecting peer",
497 resizeWidth, resizeHeight, w, h);
502 update = client->context.update;
503 WINPR_ASSERT(update);
504 WINPR_ASSERT(update->DesktopResize);
512 rc = update->DesktopResize(update->context);
513 WLog_INFO(TAG,
"Client %s resize requested (%" PRIu32
"x%" PRIu32
"@%" PRIu32
")",
514 peer->hostname, resizeWidth, resizeHeight,
516 client->resizeRequested = TRUE;
517 client->resizeWidth = resizeWidth;
518 client->resizeHeight = resizeHeight;
524static BOOL shadow_client_post_connect(freerdp_peer* peer)
527 rdpSettings* settings =
nullptr;
528 rdpShadowClient* client =
nullptr;
529 rdpShadowServer* server =
nullptr;
530 rdpShadowSubsystem* subsystem =
nullptr;
534 client = (rdpShadowClient*)peer->context;
535 WINPR_ASSERT(client);
537 settings = peer->context->settings;
538 WINPR_ASSERT(settings);
540 server = client->server;
541 WINPR_ASSERT(server);
543 subsystem = server->subsystem;
544 WINPR_ASSERT(subsystem);
552 const UINT32 MultifragMaxRequestSize =
554 if (MultifragMaxRequestSize < 0x3F0000)
557 settings, FreeRDP_NSCodec,
563 WLog_INFO(TAG,
"Client from %s is activated (%" PRIu32
"x%" PRIu32
"@%" PRIu32
")",
568 if (shadow_client_channels_post_connect(client) != CHANNEL_RC_OK)
571 shadow_client_mark_invalid(client, 0,
nullptr);
578 if (Username && Password)
586 if (subsystem->Authenticate)
588 authStatus = subsystem->Authenticate(subsystem, client, Username, Domain, Password);
593 WLog_ERR(TAG,
"client authentication failure: %d", authStatus);
598 if (subsystem->ClientConnect)
600 return subsystem->ClientConnect(subsystem, client);
607static inline void shadow_client_convert_rects(rdpShadowClient* client,
RECTANGLE_16* dst,
610 WINPR_ASSERT(client);
611 WINPR_ASSERT(client->server);
613 WINPR_ASSERT(src || (numRects == 0));
615 if (client->server->shareSubRect)
617 UINT16 offsetX = client->server->subRect.left;
618 UINT16 offsetY = client->server->subRect.top;
620 for (UINT32 i = 0; i < numRects; i++)
625 d->left = s->left + offsetX;
626 d->right = s->right + offsetX;
627 d->top = s->top + offsetY;
628 d->bottom = s->bottom + offsetY;
641static BOOL shadow_client_refresh_request(rdpShadowClient* client)
643 wMessage message = WINPR_C_ARRAY_INIT;
646 WINPR_ASSERT(client);
647 WINPR_ASSERT(client->subsystem);
649 MsgPipe = client->subsystem->MsgPipe;
650 WINPR_ASSERT(MsgPipe);
652 message.id = SHADOW_MSG_IN_REFRESH_REQUEST_ID;
653 message.wParam =
nullptr;
654 message.lParam =
nullptr;
655 message.context = (
void*)client;
656 message.Free =
nullptr;
657 return MessageQueue_Dispatch(MsgPipe->In, &message);
661static BOOL shadow_client_refresh_rect(rdpContext* context, BYTE count,
const RECTANGLE_16* areas)
663 rdpShadowClient* client = (rdpShadowClient*)context;
679 shadow_client_convert_rects(client, rects, areas, count);
680 shadow_client_mark_invalid(client, count, rects);
685 shadow_client_mark_invalid(client, 0,
nullptr);
688 return shadow_client_refresh_request(client);
692static BOOL shadow_client_suppress_output(rdpContext* context, BYTE allow,
const RECTANGLE_16* area)
694 rdpShadowClient* client = (rdpShadowClient*)context;
697 WINPR_ASSERT(client);
699 client->suppressOutput = !(allow);
705 shadow_client_convert_rects(client, ®ion, area, 1);
706 shadow_client_mark_invalid(client, 1, ®ion);
710 shadow_client_mark_invalid(client, 0,
nullptr);
714 return shadow_client_refresh_request(client);
718static BOOL shadow_client_activate(freerdp_peer* peer)
722 rdpShadowClient* client = (rdpShadowClient*)peer->context;
723 WINPR_ASSERT(client);
726 if (shadow_client_recalc_desktop_size(client))
727 return shadow_send_desktop_resize(client);
729 shadow_reset_desktop_resize(client);
730 client->activated = TRUE;
731 client->inLobby = !(client->mayView);
733 if (shadow_encoder_reset(client->encoder) < 0)
735 WLog_ERR(TAG,
"Failed to reset encoder");
740 return shadow_client_refresh_rect(&client->context, 0,
nullptr);
744static BOOL shadow_client_logon(freerdp_peer* peer,
const SEC_WINNT_AUTH_IDENTITY* identity,
748 char* user =
nullptr;
749 char* domain =
nullptr;
750 char* password =
nullptr;
751 rdpSettings* settings =
nullptr;
753 WINPR_UNUSED(automatic);
756 WINPR_ASSERT(identity);
758 WINPR_ASSERT(peer->context);
760 settings = peer->context->settings;
761 WINPR_ASSERT(settings);
763 if (identity->Flags & SEC_WINNT_AUTH_IDENTITY_UNICODE)
766 user = ConvertWCharNToUtf8Alloc(identity->User, identity->UserLength,
nullptr);
768 if (identity->Domain)
769 domain = ConvertWCharNToUtf8Alloc(identity->Domain, identity->DomainLength,
nullptr);
771 if (identity->Password)
773 ConvertWCharNToUtf8Alloc(identity->Password, identity->PasswordLength,
nullptr);
778 user = _strdup((
char*)identity->User);
780 if (identity->Domain)
781 domain = _strdup((
char*)identity->Domain);
783 if (identity->Password)
784 password = _strdup((
char*)identity->Password);
787 if ((identity->User && !user) || (identity->Domain && !domain) ||
788 (identity->Password && !password))
815static inline void shadow_client_common_frame_acknowledge(rdpShadowClient* client, UINT32 frameId)
825 WINPR_ASSERT(client);
826 WINPR_ASSERT(client->encoder);
827 client->encoder->lastAckframeId = frameId;
831static BOOL shadow_client_surface_frame_acknowledge(rdpContext* context, UINT32 frameId)
833 rdpShadowClient* client = (rdpShadowClient*)context;
834 shadow_client_common_frame_acknowledge(client, frameId);
838 WINPR_ASSERT(client);
839 WINPR_ASSERT(client->encoder);
840 client->encoder->queueDepth = QUEUE_DEPTH_UNAVAILABLE;
846shadow_client_rdpgfx_frame_acknowledge(RdpgfxServerContext* context,
849 rdpShadowClient* client =
nullptr;
851 WINPR_ASSERT(context);
852 WINPR_ASSERT(frameAcknowledge);
854 client = (rdpShadowClient*)context->custom;
855 shadow_client_common_frame_acknowledge(client, frameAcknowledge->frameId);
857 WINPR_ASSERT(client);
858 WINPR_ASSERT(client->encoder);
859 client->encoder->queueDepth = frameAcknowledge->queueDepth;
860 return CHANNEL_RC_OK;
864static BOOL shadow_are_caps_filtered(
const rdpSettings* settings, UINT32 caps)
866 const UINT32 capList[] = {
867#if defined(WITH_GFX_AV1)
868 RDPGFX_CAPVERSION_FRDP_1,
871 RDPGFX_CAPVERSION_81,
872 RDPGFX_CAPVERSION_10,
873 RDPGFX_CAPVERSION_101,
874 RDPGFX_CAPVERSION_102,
875 RDPGFX_CAPVERSION_103,
876 RDPGFX_CAPVERSION_104,
877 RDPGFX_CAPVERSION_105,
878 RDPGFX_CAPVERSION_106,
879 RDPGFX_CAPVERSION_106_ERR,
880 RDPGFX_CAPVERSION_107
881#if defined(WITH_GFX_AZURE)
883 RDPGFX_CAPVERSION_111,
884 RDPGFX_CAPVERSION_112,
885 RDPGFX_CAPVERSION_113
889 WINPR_ASSERT(settings);
892 for (UINT32 x = 0; x < ARRAYSIZE(capList); x++)
894 if (caps == capList[x])
895 return (filter & (1u << x)) != 0;
902static UINT shadow_client_send_caps_confirm(RdpgfxServerContext* context, rdpShadowClient* client,
905 WINPR_ASSERT(context);
906 WINPR_ASSERT(client);
909 WINPR_ASSERT(context->CapsConfirm);
910 UINT rc = context->CapsConfirm(context, pdu);
911 client->areGfxCapsReady = (rc == CHANNEL_RC_OK);
912 client->confirmedCaps = *pdu->capsSet;
914 rdpSettings* clientSettings = client->context.settings;
915 WINPR_ASSERT(clientSettings);
917#if defined(WITH_GFX_AV1)
918 if (pdu->capsSet->version == RDPGFX_CAPVERSION_FRDP_1)
920 UINT32 flags = FREERDP_CODEC_AV1_I420;
921 if ((pdu->capsSet->flags & RDPGFX_CAPS_FLAG_AV1_I444_DISABLED) == 0)
922 flags = FREERDP_CODEC_AV1_I444;
924 if (shadow_encoder_prepare(client->encoder, flags) < 0)
925 return ERROR_INVALID_PARAMETER;
929 if (shadow_avc444_enabled(client) || shadow_avc420_enabled(client))
932 const BOOL h264 = (shadow_encoder_prepare(client->encoder, FREERDP_CODEC_AVC420 |
933 FREERDP_CODEC_AVC444) >= 0);
937 clientSettings, FreeRDP_GfxH264,
938 (pdu->capsSet->flags & RDPGFX_CAPS_FLAG_AVC420_ENABLED) != 0))
960static BOOL shadow_client_caps_test_version(RdpgfxServerContext* context, rdpShadowClient* client,
962 UINT32 capsSetCount, UINT32 capsVersion, UINT* rc)
964 WINPR_ASSERT(context);
965 WINPR_ASSERT(client);
966 WINPR_ASSERT(capsSets || (capsSetCount == 0));
969 WINPR_ASSERT(context->rdpcontext);
970 const rdpSettings* srvSettings = context->rdpcontext->settings;
971 WINPR_ASSERT(srvSettings);
973 rdpSettings* clientSettings = client->context.settings;
974 WINPR_ASSERT(clientSettings);
976 if (shadow_are_caps_filtered(srvSettings, capsVersion))
979 for (UINT32 index = 0; index < capsSetCount; index++)
983 WLog_INFO(TAG,
"testing %s vs %s[0x%08" PRIx32
"]",
984 rdpgfx_caps_version_str(currentCaps->version),
985 rdpgfx_caps_version_str(capsVersion), capsVersion);
986 if (currentCaps->version == capsVersion)
992 const UINT32 flags = pdu.capsSet->flags;
995 (flags & RDPGFX_CAPS_FLAG_SMALL_CACHE) != 0))
999 BOOL avc444 = !(flags & RDPGFX_CAPS_FLAG_AVC_DISABLED);
1000 BOOL avc444v2 = avc444;
1017 const BOOL progressivev2 =
1029#if defined(WITH_GFX_AV1)
1033 const UINT32 av1Profile =
1040 if (!h264 || (!avc444v2 && !avc444 && !avc420))
1041 pdu.capsSet->flags |= RDPGFX_CAPS_FLAG_AVC_DISABLED;
1043#if defined(WITH_GFX_AV1)
1044 if (currentCaps->version == RDPGFX_CAPVERSION_FRDP_1)
1049 pdu.capsSet->flags &=
1050 ~(RDPGFX_CAPS_FLAG_AV1_I444_SUPPORTED | RDPGFX_CAPS_FLAG_AV1_I444_DISABLED);
1052 if ((currentCaps->flags & RDPGFX_CAPS_FLAG_AV1_I444_SUPPORTED) != 0)
1054 if (av1Profile == 0)
1058 if ((av1Profile == 0) ||
1059 ((currentCaps->flags & RDPGFX_CAPS_FLAG_AV1_I444_SUPPORTED) == 0))
1060 pdu.capsSet->flags |= RDPGFX_CAPS_FLAG_AV1_I444_DISABLED;
1061 if ((currentCaps->flags & RDPGFX_CAPS_FLAG_AV1_I444_DISABLED) != 0)
1062 pdu.capsSet->flags |= RDPGFX_CAPS_FLAG_AV1_I444_DISABLED;
1065 *rc = shadow_client_send_caps_confirm(context, client, &pdu);
1079static UINT shadow_client_rdpgfx_caps_advertise(RdpgfxServerContext* context,
1082 UINT rc = ERROR_INTERNAL_ERROR;
1086 WINPR_ASSERT(context);
1087 WINPR_ASSERT(capsAdvertise);
1089 rdpShadowClient* client = (rdpShadowClient*)context->custom;
1090 WINPR_ASSERT(client);
1091 WINPR_ASSERT(context->rdpcontext);
1093 const rdpSettings* srvSettings = context->rdpcontext->settings;
1094 WINPR_ASSERT(srvSettings);
1096 rdpSettings* clientSettings = client->context.settings;
1097 WINPR_ASSERT(clientSettings);
1100 if (!shadow_client_refresh_rect(&client->context, 0,
nullptr))
1103 const UINT32 capsVersions[] = {
1104#if defined(WITH_GFX_AV1)
1105 RDPGFX_CAPVERSION_FRDP_1,
1107#if defined(WITH_GFX_AZURE)
1108 RDPGFX_CAPVERSION_113, RDPGFX_CAPVERSION_112, RDPGFX_CAPVERSION_111,
1110 RDPGFX_CAPVERSION_107, RDPGFX_CAPVERSION_106, RDPGFX_CAPVERSION_106_ERR,
1111 RDPGFX_CAPVERSION_105, RDPGFX_CAPVERSION_104, RDPGFX_CAPVERSION_103,
1112 RDPGFX_CAPVERSION_102, RDPGFX_CAPVERSION_101, RDPGFX_CAPVERSION_10,
1116 shadow_encoder_prepare(client->encoder, FREERDP_CODEC_AVC420 | FREERDP_CODEC_AVC444) >= 0;
1118#if defined(WITH_GFX_AV1)
1122 const UINT32 profile =
1127 codec = FREERDP_CODEC_AV1_I420;
1130 codec = FREERDP_CODEC_AV1_I444;
1133 return ERROR_BAD_PROFILE;
1135 const BOOL av1 = shadow_encoder_prepare(client->encoder, codec) >= 0;
1139 return ERROR_INTERNAL_ERROR;
1144 for (
size_t x = 0; x < ARRAYSIZE(capsVersions); x++)
1146 const UINT32 cur = capsVersions[x];
1147 if (shadow_client_caps_test_version(context, client, h264, capsAdvertise->capsSets,
1148 capsAdvertise->capsSetCount, cur, &rc))
1152 if (!shadow_are_caps_filtered(srvSettings, RDPGFX_CAPVERSION_81))
1154 for (UINT32 index = 0; index < capsAdvertise->capsSetCount; index++)
1156 const RDPGFX_CAPSET* currentCaps = &capsAdvertise->capsSets[index];
1158 if (currentCaps->version == RDPGFX_CAPVERSION_81)
1162 pdu.capsSet = ∩︀
1164 flags = pdu.capsSet->flags;
1172 (flags & RDPGFX_CAPS_FLAG_THINCLIENT) != 0))
1175 (flags & RDPGFX_CAPS_FLAG_SMALL_CACHE) != 0))
1178#ifndef WITH_GFX_H264
1182 pdu.capsSet->flags &= ~RDPGFX_CAPS_FLAG_AVC420_ENABLED;
1187 return shadow_client_send_caps_confirm(context, client, &pdu);
1192 if (!shadow_are_caps_filtered(srvSettings, RDPGFX_CAPVERSION_8))
1194 for (UINT32 index = 0; index < capsAdvertise->capsSetCount; index++)
1196 const RDPGFX_CAPSET* currentCaps = &capsAdvertise->capsSets[index];
1198 if (currentCaps->version == RDPGFX_CAPVERSION_8)
1202 pdu.capsSet = ∩︀
1203 flags = pdu.capsSet->flags;
1213 (flags & RDPGFX_CAPS_FLAG_THINCLIENT) != 0))
1216 (flags & RDPGFX_CAPS_FLAG_SMALL_CACHE) != 0))
1219 return shadow_client_send_caps_confirm(context, client, &pdu);
1224 return CHANNEL_RC_UNSUPPORTED_VERSION;
1231 WINPR_ASSERT(havc420);
1232 return sizeof(UINT32)
1234 * havc420->meta.numRegionRects +
1238#if defined(WITH_GFX_AV1)
1240static BOOL shadow_client_send_av1(rdpShadowClient* client,
const BYTE* pSrcData, UINT32 nSrcStep,
1241 UINT32 SrcFormat, UINT16 nWidth, UINT16 nHeight,
1246 WINPR_ASSERT(client);
1248 rdpShadowEncoder* encoder = client->encoder;
1249 WINPR_ASSERT(encoder);
1251 UINT error = CHANNEL_RC_OK;
1256 UINT32 flags = FREERDP_CODEC_AV1_I444;
1257 if ((client->confirmedCaps.flags & RDPGFX_CAPS_FLAG_AV1_I444_DISABLED) != 0)
1258 flags = FREERDP_CODEC_AV1_I420;
1260 if (shadow_encoder_prepare(encoder, flags) < 0)
1262 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_AV1");
1266 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1267 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1268 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1269 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1270 regionRect.left = (UINT16)cmd->left;
1271 regionRect.top = (UINT16)cmd->top;
1272 regionRect.right = (UINT16)cmd->right;
1273 regionRect.bottom = (UINT16)cmd->bottom;
1274 rc = freerdp_av1_compress(encoder->av1, pSrcData, SrcFormat, nSrcStep, nWidth, nHeight,
1275 ®ionRect, &avc420.data, &avc420.length, &avc420.meta);
1278 WLog_ERR(TAG,
"freerdp_av1_compress failed");
1285 cmd->codecId = RDPGFX_CODECID_AV1;
1286 cmd->extra = (
void*)&avc420;
1288 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1290 cmd->extra =
nullptr;
1292 free_h264_metablock(&avc420.meta);
1296 WLog_ERR(TAG,
"SurfaceFrameCommand failed with error %" PRIu32
"", error);
1304static BOOL shadow_client_send_avc444(rdpShadowClient* client,
const BYTE* pSrcData,
1305 UINT32 nSrcStep, UINT32 SrcFormat, UINT16 nWidth,
1310 WINPR_ASSERT(client);
1312 rdpShadowEncoder* encoder = client->encoder;
1313 WINPR_ASSERT(encoder);
1315 UINT error = CHANNEL_RC_OK;
1319 const BOOL GfxAVC444v2 = (cmd->codecId == RDPGFX_CODECID_AVC444v2);
1320 BYTE version = GfxAVC444v2 ? 2 : 1;
1322 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_AVC444) < 0)
1324 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_AVC444");
1328 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1329 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1330 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1331 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1332 regionRect.left = (UINT16)cmd->left;
1333 regionRect.top = (UINT16)cmd->top;
1334 regionRect.right = (UINT16)cmd->right;
1335 regionRect.bottom = (UINT16)cmd->bottom;
1336 rc = avc444_compress(encoder->h264, pSrcData, SrcFormat, nSrcStep, nWidth, nHeight, version,
1337 ®ionRect, &avc444.LC, &avc444.bitstream[0].data,
1338 &avc444.bitstream[0].length, &avc444.bitstream[1].data,
1339 &avc444.bitstream[1].length, &avc444.bitstream[0].meta,
1340 &avc444.bitstream[1].meta);
1343 WLog_ERR(TAG,
"avc420_compress failed for avc444");
1350 avc444.cbAvc420EncodedBitstream1 = rdpgfx_estimate_h264_avc420(&avc444.bitstream[0]);
1351 cmd->extra = (
void*)&avc444;
1352 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1354 cmd->extra =
nullptr;
1357 free_h264_metablock(&avc444.bitstream[0].meta);
1358 free_h264_metablock(&avc444.bitstream[1].meta);
1361 WLog_ERR(TAG,
"SurfaceFrameCommand failed with error %" PRIu32
"", error);
1368static BOOL shadow_client_send_avc420(rdpShadowClient* client,
const BYTE* pSrcData,
1369 UINT32 nSrcStep, UINT32 SrcFormat, UINT16 nWidth,
1374 WINPR_ASSERT(client);
1376 rdpShadowEncoder* encoder = client->encoder;
1377 WINPR_ASSERT(encoder);
1379 UINT error = CHANNEL_RC_OK;
1384 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_AVC420) < 0)
1386 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_AVC420");
1390 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1391 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1392 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1393 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1394 regionRect.left = (UINT16)cmd->left;
1395 regionRect.top = (UINT16)cmd->top;
1396 regionRect.right = (UINT16)cmd->right;
1397 regionRect.bottom = (UINT16)cmd->bottom;
1398 rc = avc420_compress(encoder->h264, pSrcData, SrcFormat, nSrcStep, nWidth, nHeight, ®ionRect,
1399 &avc420.data, &avc420.length, &avc420.meta);
1402 WLog_ERR(TAG,
"avc420_compress failed");
1409 cmd->codecId = RDPGFX_CODECID_AVC420;
1410 cmd->extra = (
void*)&avc420;
1411 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1413 cmd->extra =
nullptr;
1415 free_h264_metablock(&avc420.meta);
1419 WLog_ERR(TAG,
"SurfaceFrameCommand failed with error %" PRIu32
"", error);
1426static BOOL shadow_client_send_rfx(rdpShadowClient* client,
const BYTE* pSrcData, UINT32 nSrcStep,
1427 UINT32 SrcFormat, UINT16 nWidth, UINT16 nHeight,
1432 WINPR_ASSERT(client);
1434 rdpShadowEncoder* encoder = client->encoder;
1435 WINPR_ASSERT(encoder);
1437 UINT error = CHANNEL_RC_OK;
1439 RFX_RECT rect = WINPR_C_ARRAY_INIT;
1441 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_REMOTEFX) < 0)
1443 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_REMOTEFX");
1447 const UINT32 rfxFormat = rfx_context_get_pixel_format(encoder->rfx);
1448 if (rfxFormat != SrcFormat)
1450 WLog_ERR(TAG,
"RFX context pixel format mismatch %s, expected %s",
1451 FreeRDPGetColorFormatName(rfxFormat), FreeRDPGetColorFormatName(SrcFormat));
1455 wStream* s = Stream_New(
nullptr, 1024);
1458 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1459 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1460 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1461 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1462 rect.x = (UINT16)cmd->left;
1463 rect.y = (UINT16)cmd->top;
1464 rect.width = WINPR_ASSERTING_INT_CAST(UINT16, cmd->right - cmd->left);
1465 rect.height = WINPR_ASSERTING_INT_CAST(UINT16, cmd->bottom - cmd->top);
1467 rc = rfx_compose_message(encoder->rfx, s, &rect, 1, pSrcData, nWidth, nHeight, nSrcStep);
1471 WLog_ERR(TAG,
"rfx_compose_message failed");
1472 Stream_Free(s, TRUE);
1479 const size_t pos = Stream_GetPosition(s);
1480 WINPR_ASSERT(pos <= UINT32_MAX);
1482 cmd->codecId = RDPGFX_CODECID_CAVIDEO;
1483 cmd->data = Stream_Buffer(s);
1484 cmd->length = (UINT32)pos;
1486 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1488 cmd->data =
nullptr;
1491 Stream_Free(s, TRUE);
1494 WLog_ERR(TAG,
"SurfaceFrameCommand failed with error %" PRIu32
"", error);
1501static BOOL shadow_client_send_progressive(rdpShadowClient* client,
const BYTE* pSrcData,
1502 UINT32 nSrcStep, UINT32 SrcFormat, UINT16 nWidth,
1507 WINPR_ASSERT(client);
1509 rdpShadowEncoder* encoder = client->encoder;
1510 WINPR_ASSERT(encoder);
1512 UINT error = CHANNEL_RC_OK;
1517 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_PROGRESSIVE) < 0)
1519 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_PROGRESSIVE");
1523 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1524 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1525 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1526 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1527 regionRect.left = (UINT16)cmd->left;
1528 regionRect.top = (UINT16)cmd->top;
1529 regionRect.right = (UINT16)cmd->right;
1530 regionRect.bottom = (UINT16)cmd->bottom;
1531 region16_init(®ion);
1532 if (!region16_union_rect(®ion, ®ion, ®ionRect))
1534 region16_uninit(®ion);
1537 rc = progressive_compress(encoder->progressive, pSrcData, nSrcStep * nHeight, SrcFormat, nWidth,
1538 nHeight, nSrcStep, ®ion, &cmd->data, &cmd->length);
1539 region16_uninit(®ion);
1542 WLog_ERR(TAG,
"progressive_compress failed");
1549 cmd->codecId = RDPGFX_CODECID_CAPROGRESSIVE;
1551 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1554 cmd->data =
nullptr;
1558 WLog_ERR(TAG,
"SurfaceFrameCommand failed with error %" PRIu32
"", error);
1565static BOOL shadow_client_send_planar(rdpShadowClient* client,
const BYTE* pSrcData,
1566 UINT32 nSrcStep, UINT32 SrcFormat,
1571 WINPR_ASSERT(client);
1573 rdpShadowEncoder* encoder = client->encoder;
1574 WINPR_ASSERT(encoder);
1576 UINT error = CHANNEL_RC_OK;
1577 const UINT32 w = cmd->right - cmd->left;
1578 const UINT32 h = cmd->bottom - cmd->top;
1580 &pSrcData[cmd->top * nSrcStep + cmd->left * FreeRDPGetBytesPerPixel(SrcFormat)];
1581 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_PLANAR) < 0)
1583 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_PLANAR");
1587 const BOOL rc = freerdp_bitmap_planar_context_reset(encoder->planar, w, h);
1591 freerdp_planar_topdown_image(encoder->planar, TRUE);
1593 cmd->data = freerdp_bitmap_compress_planar(encoder->planar, src, SrcFormat, w, h, nSrcStep,
1594 nullptr, &cmd->length);
1595 WINPR_ASSERT(cmd->data || (cmd->length == 0));
1597 cmd->codecId = RDPGFX_CODECID_PLANAR;
1599 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart, cmdend);
1601 cmd->data =
nullptr;
1605 WLog_ERR(TAG,
"SurfaceFrameCommand failed with error %" PRIu32
"", error);
1612static BOOL shadow_client_send_uncompressed(rdpShadowClient* client,
const BYTE* pSrcData,
1613 UINT32 nSrcStep, UINT32 SrcFormat,
1618 WINPR_ASSERT(client);
1620 UINT error = CHANNEL_RC_OK;
1621 const UINT32 w = cmd->right - cmd->left;
1622 const UINT32 h = cmd->bottom - cmd->top;
1623 const UINT32 length = w * 4 * h;
1625 BYTE* data = malloc(length);
1629 BOOL rc = freerdp_image_copy_no_overlap(data, cmd->format, 0, 0, 0, w, h, pSrcData, SrcFormat,
1630 nSrcStep, cmd->left, cmd->top,
nullptr, 0);
1638 cmd->length = length;
1639 cmd->codecId = RDPGFX_CODECID_UNCOMPRESSED;
1641 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart, cmdend);
1643 cmd->data =
nullptr;
1646 WLog_ERR(TAG,
"SurfaceFrameCommand failed with error %" PRIu32
"", error);
1653static BOOL shadow_avc444_enabled(
const rdpShadowClient* client)
1655 WINPR_ASSERT(client);
1656 switch (client->confirmedCaps.version)
1658 case RDPGFX_CAPVERSION_10:
1659 case RDPGFX_CAPVERSION_101:
1660 case RDPGFX_CAPVERSION_102:
1661 case RDPGFX_CAPVERSION_103:
1662 case RDPGFX_CAPVERSION_104:
1663 case RDPGFX_CAPVERSION_105:
1664 case RDPGFX_CAPVERSION_106:
1665 case RDPGFX_CAPVERSION_107:
1666#if defined(WITH_GFX_AZURE)
1667 case RDPGFX_CAPVERSION_111:
1668 case RDPGFX_CAPVERSION_112:
1669 case RDPGFX_CAPVERSION_113:
1671 return (client->confirmedCaps.flags & RDPGFX_CAPS_FLAG_AVC_DISABLED) == 0;
1679static BOOL shadow_avc420_enabled(
const rdpShadowClient* client)
1681 if (shadow_avc444_enabled(client))
1683 if (client->confirmedCaps.version != RDPGFX_CAPVERSION_81)
1685 return (client->confirmedCaps.flags & RDPGFX_CAPS_FLAG_AVC420_ENABLED) != 0;
1694static BOOL shadow_client_send_surface_gfx(rdpShadowClient* client,
const BYTE* pSrcData,
1695 UINT32 nSrcStep, UINT32 SrcFormat, UINT16 nXSrc,
1696 UINT16 nYSrc, UINT16 nWidth, UINT16 nHeight)
1698 const rdpContext* context = (
const rdpContext*)client;
1704 if (!context || !pSrcData)
1707 const rdpSettings* settings = context->settings;
1708 rdpShadowEncoder* encoder = client->encoder;
1710 if (!settings || !encoder)
1713 if (client->first_frame)
1717 if (!rfx_context_reset(encoder->rfx, nWidth, nHeight))
1720 client->first_frame = FALSE;
1723 cmdstart.frameId = shadow_encoder_create_frame_id(encoder);
1724 GetSystemTime(&sTime);
1725 cmdstart.timestamp = (UINT32)(sTime.wHour << 22U | sTime.wMinute << 16U | sTime.wSecond << 10U |
1726 sTime.wMilliseconds);
1727 cmdend.frameId = cmdstart.frameId;
1728 cmd.surfaceId = client->surfaceId;
1729 cmd.format = PIXEL_FORMAT_BGRX32;
1732 cmd.right = cmd.left + nWidth;
1733 cmd.bottom = cmd.top + nHeight;
1735 cmd.height = nHeight;
1737#if defined(WITH_GFX_AV1)
1740 if (client->confirmedCaps.version == RDPGFX_CAPVERSION_FRDP_1)
1742 return shadow_client_send_av1(client, pSrcData, nSrcStep, SrcFormat, nWidth, nHeight,
1743 &cmd, &cmdstart, &cmdend);
1753 if (GfxAVC444 || GfxAVC444v2)
1755 if (shadow_avc444_enabled(client))
1757 cmd.codecId = GfxAVC444v2 ? RDPGFX_CODECID_AVC444v2 : RDPGFX_CODECID_AVC444;
1758 return shadow_client_send_avc444(client, pSrcData, nSrcStep, SrcFormat, nWidth, nHeight,
1759 &cmd, &cmdstart, &cmdend);
1763 if (GfxH264 && shadow_avc420_enabled(client))
1765 return shadow_client_send_avc420(client, pSrcData, nSrcStep, SrcFormat, nWidth, nHeight,
1766 &cmd, &cmdstart, &cmdend);
1772 return shadow_client_send_rfx(client, pSrcData, nSrcStep, SrcFormat, nWidth, nHeight, &cmd,
1773 &cmdstart, &cmdend);
1778 return shadow_client_send_progressive(client, pSrcData, nSrcStep, SrcFormat, nWidth,
1779 nHeight, &cmd, &cmdstart, &cmdend);
1784 return shadow_client_send_planar(client, pSrcData, nSrcStep, SrcFormat, &cmd, &cmdstart,
1788 return shadow_client_send_uncompressed(client, pSrcData, nSrcStep, SrcFormat, &cmd, &cmdstart,
1793static BOOL stream_surface_bits_supported(
const rdpSettings* settings)
1795 const UINT32 supported =
1797 return ((supported & SURFCMDS_STREAM_SURFACE_BITS) != 0);
1801static BOOL set_surface_bits_supported(
const rdpSettings* settings)
1803 const UINT32 supported =
1805 return ((supported & SURFCMDS_SET_SURFACE_BITS) != 0);
1809static BOOL is_surface_command_supported(
const rdpSettings* settings)
1811 if (stream_surface_bits_supported(settings))
1815 if (supported && (rfxID != 0))
1818 if (set_surface_bits_supported(settings))
1822 if (supported && (nsID != 0))
1834static BOOL shadow_client_send_surface_bits(rdpShadowClient* client, BYTE* pSrcData,
1835 UINT32 nSrcStep, UINT16 nXSrc, UINT16 nYSrc,
1836 UINT16 nWidth, UINT16 nHeight)
1842 size_t numMessages = 0;
1844 rdpUpdate* update =
nullptr;
1845 rdpContext* context = (rdpContext*)client;
1846 rdpSettings* settings =
nullptr;
1847 rdpShadowEncoder* encoder =
nullptr;
1850 if (!context || !pSrcData)
1853 update = context->update;
1854 settings = context->settings;
1855 encoder = client->encoder;
1857 if (!update || !settings || !encoder)
1860 if (encoder->frameAck)
1861 frameId = shadow_encoder_create_frame_id(encoder);
1866 if (stream_surface_bits_supported(settings) &&
1869 RFX_RECT rect = WINPR_C_ARRAY_INIT;
1871 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_REMOTEFX) < 0)
1873 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_REMOTEFX");
1880 rect.width = nWidth;
1881 rect.height = nHeight;
1883 const UINT32 MultifragMaxRequestSize =
1885 RFX_MESSAGE_LIST* messages =
1886 rfx_encode_messages(encoder->rfx, &rect, 1, pSrcData,
1889 nSrcStep, &numMessages, MultifragMaxRequestSize);
1892 WLog_ERR(TAG,
"rfx_encode_messages failed");
1896 cmd.cmdType = CMDTYPE_STREAM_SURFACE_BITS;
1897 WINPR_ASSERT(rfxID <= UINT16_MAX);
1898 cmd.bmp.codecID = (UINT16)rfxID;
1909 cmd.skipCompression = TRUE;
1911 for (
size_t i = 0; i < numMessages; i++)
1913 Stream_ResetPosition(s);
1915 const RFX_MESSAGE* msg = rfx_message_list_get(messages, i);
1916 if (!rfx_write_message(encoder->rfx, s, msg))
1918 WLog_ERR(TAG,
"rfx_write_message failed");
1923 cmd.bmp.bitmapDataLength = WINPR_ASSERTING_INT_CAST(UINT32, Stream_GetPosition(s));
1924 cmd.bmp.bitmapData = Stream_Buffer(s);
1926 last = ((i + 1) == numMessages);
1928 if (!encoder->frameAck)
1929 IFCALLRET(update->SurfaceBits, ret, update->context, &cmd);
1931 IFCALLRET(update->SurfaceFrameBits, ret, update->context, &cmd, first, last,
1936 WLog_ERR(TAG,
"Send surface bits(RemoteFxCodec) failed");
1941 rfx_message_list_free(messages);
1943 else if (set_surface_bits_supported(settings) &&
1946 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_NSCODEC) < 0)
1948 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_NSCODEC");
1953 Stream_ResetPosition(s);
1954 pSrcData = &pSrcData[(nYSrc * nSrcStep) + (nXSrc * 4)];
1955 if (!nsc_compose_message(encoder->nsc, s, pSrcData, nWidth, nHeight, nSrcStep))
1958 cmd.cmdType = CMDTYPE_SET_SURFACE_BITS;
1960 WINPR_ASSERT(nsID <= UINT16_MAX);
1961 cmd.bmp.codecID = (UINT16)nsID;
1962 cmd.destLeft = nXSrc;
1963 cmd.destTop = nYSrc;
1964 cmd.destRight = cmd.destLeft + nWidth;
1965 cmd.destBottom = cmd.destTop + nHeight;
1966 cmd.bmp.width = nWidth;
1967 cmd.bmp.height = nHeight;
1969 cmd.bmp.bitmapDataLength = WINPR_ASSERTING_INT_CAST(UINT32, Stream_GetPosition(s));
1970 cmd.bmp.bitmapData = Stream_Buffer(s);
1974 if (!encoder->frameAck)
1975 IFCALLRET(update->SurfaceBits, ret, update->context, &cmd);
1977 IFCALLRET(update->SurfaceFrameBits, ret, update->context, &cmd, first, last, frameId);
1981 WLog_ERR(TAG,
"Send surface bits(NSCodec) failed");
1994static BOOL shadow_client_send_bitmap_update(rdpShadowClient* client, BYTE* pSrcData,
1995 UINT32 nSrcStep, UINT16 nXSrc, UINT16 nYSrc,
1996 UINT16 nWidth, UINT16 nHeight)
1999 BYTE* data =
nullptr;
2000 BYTE* buffer =
nullptr;
2007 UINT32 SrcFormat = 0;
2009 rdpContext* context = (rdpContext*)client;
2010 UINT32 totalBitmapSize = 0;
2011 UINT32 updateSizeEstimate = 0;
2015 if (!context || !pSrcData)
2018 rdpUpdate* update = context->update;
2019 rdpSettings* settings = context->settings;
2020 rdpShadowEncoder* encoder = client->encoder;
2022 if (!update || !settings || !encoder)
2025 const UINT32 maxUpdateSize =
2029 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_INTERLEAVED) < 0)
2031 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_INTERLEAVED");
2037 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_PLANAR) < 0)
2039 WLog_ERR(TAG,
"Failed to prepare encoder FREERDP_CODEC_PLANAR");
2044 SrcFormat = PIXEL_FORMAT_BGRX32;
2046 if ((nXSrc % 4) != 0)
2048 nWidth += (nXSrc % 4);
2049 nXSrc -= (nXSrc % 4);
2052 if ((nYSrc % 4) != 0)
2054 nHeight += (nYSrc % 4);
2055 nYSrc -= (nYSrc % 4);
2058 rows = (nHeight / 64) + ((nHeight % 64) ? 1 : 0);
2059 cols = (nWidth / 64) + ((nWidth % 64) ? 1 : 0);
2061 totalBitmapSize = 0;
2062 bitmapUpdate.number = rows * cols;
2067 bitmapUpdate.rectangles = bitmapData;
2069 if ((nWidth % 4) != 0)
2071 nWidth += (4 - (nWidth % 4));
2074 if ((nHeight % 4) != 0)
2076 nHeight += (4 - (nHeight % 4));
2079 for (yIdx = 0; yIdx < rows; yIdx++)
2081 for (xIdx = 0; xIdx < cols; xIdx++)
2083 bitmap = &bitmapData[k];
2085 bitmap->height = 64;
2086 bitmap->destLeft = nXSrc + (xIdx * 64);
2087 bitmap->destTop = nYSrc + (yIdx * 64);
2089 if (((INT64)bitmap->destLeft + bitmap->width) > (nXSrc + nWidth))
2090 bitmap->width = (UINT32)(nXSrc + nWidth) - bitmap->destLeft;
2092 if (((INT64)bitmap->destTop + bitmap->height) > (nYSrc + nHeight))
2093 bitmap->height = (UINT32)(nYSrc + nHeight) - bitmap->destTop;
2095 bitmap->destRight = bitmap->destLeft + bitmap->width - 1;
2096 bitmap->destBottom = bitmap->destTop + bitmap->height - 1;
2097 bitmap->compressed = TRUE;
2099 if ((bitmap->width < 4) || (bitmap->height < 4))
2105 UINT32 bytesPerPixel = (bitsPerPixel + 7) / 8;
2106 DstSize = 64 * 64 * 4;
2107 buffer = encoder->grid[k];
2109 ret = interleaved_compress(
2110 encoder->interleaved, buffer, &DstSize, bitmap->width, bitmap->height, pSrcData,
2111 SrcFormat, nSrcStep, bitmap->destLeft, bitmap->destTop,
nullptr, bitsPerPixel);
2114 bitmap->bitmapDataStream = buffer;
2115 bitmap->bitmapLength = DstSize;
2116 bitmap->bitsPerPixel = bitsPerPixel;
2117 bitmap->cbScanWidth = bitmap->width * bytesPerPixel;
2118 bitmap->cbUncompressedSize = bitmap->width * bitmap->height * bytesPerPixel;
2123 buffer = encoder->grid[k];
2124 data = &pSrcData[(bitmap->destTop * nSrcStep) + (bitmap->destLeft * 4)];
2127 freerdp_bitmap_compress_planar(encoder->planar, data, SrcFormat, bitmap->width,
2128 bitmap->height, nSrcStep, buffer, &dstSize);
2129 bitmap->bitmapDataStream = buffer;
2130 bitmap->bitmapLength = dstSize;
2131 bitmap->bitsPerPixel = 32;
2132 bitmap->cbScanWidth = bitmap->width * 4;
2133 bitmap->cbUncompressedSize = bitmap->width * bitmap->height * 4;
2136 bitmap->cbCompFirstRowSize = 0;
2137 bitmap->cbCompMainBodySize = bitmap->bitmapLength;
2138 totalBitmapSize += bitmap->bitmapLength;
2143 bitmapUpdate.number = k;
2144 updateSizeEstimate = totalBitmapSize + (k * bitmapUpdate.number) + 16;
2146 if (updateSizeEstimate > maxUpdateSize)
2150 UINT32 updateSize = 0;
2151 UINT32 newUpdateSize = 0;
2157 if (!fragBitmapData)
2159 WLog_ERR(TAG,
"Failed to allocate memory for fragBitmapData");
2164 bitmapUpdate.rectangles = fragBitmapData;
2170 newUpdateSize = updateSize + (bitmapData[i].bitmapLength + 16);
2172 if (newUpdateSize < maxUpdateSize)
2174 CopyMemory(&fragBitmapData[j++], &bitmapData[i++],
sizeof(
BITMAP_DATA));
2175 updateSize = newUpdateSize;
2178 if ((newUpdateSize >= maxUpdateSize) || (i + 1) >= k)
2180 bitmapUpdate.number = j;
2181 ret = BitmapUpdateProxy(client, &bitmapUpdate);
2191 free(fragBitmapData);
2195 ret = BitmapUpdateProxy(client, &bitmapUpdate);
2209static BOOL shadow_client_send_surface_update(rdpShadowClient* client, SHADOW_GFX_STATUS* pStatus)
2216 rdpContext* context = (rdpContext*)client;
2217 rdpSettings* settings =
nullptr;
2218 rdpShadowServer* server =
nullptr;
2219 rdpShadowSurface* surface =
nullptr;
2223 BYTE* pSrcData =
nullptr;
2224 UINT32 nSrcStep = 0;
2225 UINT32 SrcFormat = 0;
2226 UINT32 numRects = 0;
2229 if (!context || !pStatus)
2232 settings = context->settings;
2233 server = client->server;
2235 if (!settings || !server)
2238 surface = client->inLobby ? server->lobby : server->surface;
2244 EnterCriticalSection(&(client->lock));
2245 region16_init(&invalidRegion);
2247 const BOOL res = region16_copy(&invalidRegion, &(client->invalidRegion));
2248 region16_clear(&(client->invalidRegion));
2249 LeaveCriticalSection(&(client->lock));
2254 EnterCriticalSection(&surface->lock);
2255 rects = region16_rects(&(surface->invalidRegion), &numRects);
2257 for (UINT32 index = 0; index < numRects; index++)
2259 if (!region16_union_rect(&invalidRegion, &invalidRegion, &rects[index]))
2263 surfaceRect.left = 0;
2264 surfaceRect.top = 0;
2265 WINPR_ASSERT(surface->width <= UINT16_MAX);
2266 WINPR_ASSERT(surface->height <= UINT16_MAX);
2267 surfaceRect.right = (UINT16)surface->width;
2268 surfaceRect.bottom = (UINT16)surface->height;
2269 if (!region16_intersect_rect(&invalidRegion, &invalidRegion, &surfaceRect))
2272 if (server->shareSubRect)
2274 if (!region16_intersect_rect(&invalidRegion, &invalidRegion, &(server->subRect)))
2278 if (region16_is_empty(&invalidRegion))
2284 extents = region16_extents(&invalidRegion);
2285 nXSrc = extents->left;
2286 nYSrc = extents->top;
2287 nWidth = extents->right - extents->left;
2288 nHeight = extents->bottom - extents->top;
2289 pSrcData = surface->data;
2290 nSrcStep = surface->scanline;
2291 SrcFormat = surface->format;
2294 if (server->shareSubRect)
2298 subX = server->subRect.left;
2299 subY = server->subRect.top;
2302 WINPR_ASSERT(nXSrc >= 0);
2303 WINPR_ASSERT(nXSrc <= UINT16_MAX);
2304 WINPR_ASSERT(nYSrc >= 0);
2305 WINPR_ASSERT(nYSrc <= UINT16_MAX);
2306 pSrcData = &pSrcData[((UINT16)subY * nSrcStep) + ((UINT16)subX * 4U)];
2315 if (pStatus->gfxOpened && client->areGfxCapsReady)
2322 if (!pStatus->gfxSurfaceCreated)
2325 if (!(ret = shadow_client_rdpgfx_reset_graphic(client)))
2328 if (!(ret = shadow_client_rdpgfx_new_surface(client)))
2331 pStatus->gfxSurfaceCreated = TRUE;
2334 WINPR_ASSERT(nWidth >= 0);
2335 WINPR_ASSERT(nWidth <= UINT16_MAX);
2336 WINPR_ASSERT(nHeight >= 0);
2337 WINPR_ASSERT(nHeight <= UINT16_MAX);
2338 ret = shadow_client_send_surface_gfx(client, pSrcData, nSrcStep, SrcFormat, 0, 0,
2339 (UINT16)nWidth, (UINT16)nHeight);
2346 else if (is_surface_command_supported(settings))
2348 WINPR_ASSERT(nXSrc >= 0);
2349 WINPR_ASSERT(nXSrc <= UINT16_MAX);
2350 WINPR_ASSERT(nYSrc >= 0);
2351 WINPR_ASSERT(nYSrc <= UINT16_MAX);
2352 WINPR_ASSERT(nWidth >= 0);
2353 WINPR_ASSERT(nWidth <= UINT16_MAX);
2354 WINPR_ASSERT(nHeight >= 0);
2355 WINPR_ASSERT(nHeight <= UINT16_MAX);
2356 ret = shadow_client_send_surface_bits(client, pSrcData, nSrcStep, (UINT16)nXSrc,
2357 (UINT16)nYSrc, (UINT16)nWidth, (UINT16)nHeight);
2361 WINPR_ASSERT(nXSrc >= 0);
2362 WINPR_ASSERT(nXSrc <= UINT16_MAX);
2363 WINPR_ASSERT(nYSrc >= 0);
2364 WINPR_ASSERT(nYSrc <= UINT16_MAX);
2365 WINPR_ASSERT(nWidth >= 0);
2366 WINPR_ASSERT(nWidth <= UINT16_MAX);
2367 WINPR_ASSERT(nHeight >= 0);
2368 WINPR_ASSERT(nHeight <= UINT16_MAX);
2369 ret = shadow_client_send_bitmap_update(client, pSrcData, nSrcStep, (UINT16)nXSrc,
2370 (UINT16)nYSrc, (UINT16)nWidth, (UINT16)nHeight);
2374 LeaveCriticalSection(&surface->lock);
2375 region16_uninit(&invalidRegion);
2387static BOOL shadow_client_send_resize(rdpShadowClient* client, SHADOW_GFX_STATUS* pStatus)
2389 rdpContext* context = (rdpContext*)client;
2390 rdpSettings* settings =
nullptr;
2391 freerdp_peer* peer =
nullptr;
2393 if (!context || !pStatus)
2396 peer = context->peer;
2397 settings = context->settings;
2399 if (!peer || !settings)
2407 client->activated = FALSE;
2410 if (pStatus->gfxSurfaceCreated)
2412 if (!shadow_client_rdpgfx_release_surface(client))
2415 pStatus->gfxSurfaceCreated = FALSE;
2419 if (!shadow_send_desktop_resize(client))
2421 shadow_reset_desktop_resize(client);
2424 EnterCriticalSection(&(client->lock));
2425 region16_clear(&(client->invalidRegion));
2426 LeaveCriticalSection(&(client->lock));
2437static BOOL shadow_client_surface_update(rdpShadowClient* client,
REGION16* region)
2439 UINT32 numRects = 0;
2441 rects = region16_rects(region, &numRects);
2442 shadow_client_mark_invalid(client, numRects, rects);
2453static inline BOOL shadow_client_no_surface_update(rdpShadowClient* client,
2454 SHADOW_GFX_STATUS* pStatus)
2456 rdpShadowServer* server =
nullptr;
2457 rdpShadowSurface* surface =
nullptr;
2458 WINPR_UNUSED(pStatus);
2459 WINPR_ASSERT(client);
2460 server = client->server;
2461 WINPR_ASSERT(server);
2462 surface = client->inLobby ? server->lobby : server->surface;
2463 EnterCriticalSection(&surface->lock);
2464 const BOOL rc = shadow_client_surface_update(client, &(surface->invalidRegion));
2465 LeaveCriticalSection(&surface->lock);
2470static int shadow_client_subsystem_process_message(rdpShadowClient* client, wMessage* message)
2472 rdpContext* context = (rdpContext*)client;
2474 WINPR_ASSERT(message);
2475 WINPR_ASSERT(context);
2477 rdpUpdate* update = context->update;
2478 WINPR_ASSERT(update);
2482 switch (message->id)
2484 case SHADOW_MSG_OUT_POINTER_POSITION_UPDATE_ID:
2489 pointerPosition.xPos = msg->xPos;
2490 pointerPosition.yPos = msg->yPos;
2492 WINPR_ASSERT(client->server);
2493 if (client->server->shareSubRect)
2495 pointerPosition.xPos -= client->server->subRect.left;
2496 pointerPosition.yPos -= client->server->subRect.top;
2499 if (client->activated)
2501 if ((msg->xPos != client->pointerX) || (msg->yPos != client->pointerY))
2503 WINPR_ASSERT(update->pointer);
2504 if (!IFCALLRESULT(TRUE, update->pointer->PointerPosition, context,
2507 client->pointerX = msg->xPos;
2508 client->pointerY = msg->yPos;
2515 case SHADOW_MSG_OUT_POINTER_ALPHA_UPDATE_ID:
2524 pointerNew.xorBpp = 24;
2525 pointerColor = &(pointerNew.colorPtrAttr);
2526 pointerColor->cacheIndex = 0;
2527 pointerColor->hotSpotX = WINPR_ASSERTING_INT_CAST(UINT16, msg->xHot);
2528 pointerColor->hotSpotY = WINPR_ASSERTING_INT_CAST(UINT16, msg->yHot);
2529 pointerColor->width = WINPR_ASSERTING_INT_CAST(UINT16, msg->width);
2530 pointerColor->height = WINPR_ASSERTING_INT_CAST(UINT16, msg->height);
2531 pointerColor->lengthAndMask = WINPR_ASSERTING_INT_CAST(UINT16, msg->lengthAndMask);
2532 pointerColor->lengthXorMask = WINPR_ASSERTING_INT_CAST(UINT16, msg->lengthXorMask);
2533 pointerColor->xorMaskData = msg->xorMaskData;
2534 pointerColor->andMaskData = msg->andMaskData;
2535 pointerCached.cacheIndex = pointerColor->cacheIndex;
2537 if (client->activated)
2539 if (client->server->ShowMouseCursor)
2541 if (!IFCALLRESULT(TRUE, update->pointer->PointerNew, context, &pointerNew))
2543 if (!IFCALLRESULT(TRUE, update->pointer->PointerCached, context,
2550 pointer_system.type = SYSPTR_DEFAULT;
2551 if (!IFCALLRESULT(TRUE, update->pointer->PointerSystem, context,
2560 case SHADOW_MSG_OUT_AUDIO_OUT_SAMPLES_ID:
2567 if (client->activated && client->rdpsnd && client->rdpsnd->Activated)
2569 client->rdpsnd->src_format = msg->audio_format;
2572 IFCALLRESULT(CHANNEL_RC_OK, client->rdpsnd->SendSamples, client->rdpsnd,
2573 msg->buf, msg->nFrames, msg->wTimestamp);
2574 if (CHANNEL_RC_OK != error)
2581 case SHADOW_MSG_OUT_AUDIO_OUT_VOLUME_ID:
2586 if (client->activated && client->rdpsnd && client->rdpsnd->Activated)
2588 const UINT error = IFCALLRESULT(CHANNEL_RC_OK, client->rdpsnd->SetVolume,
2589 client->rdpsnd, msg->left, msg->right);
2590 if (CHANNEL_RC_OK != error)
2598 WLog_ERR(TAG,
"Unknown message id: %" PRIu32
"", message->id);
2602 shadow_client_free_queued_message(message);
2607static DWORD WINAPI shadow_client_thread(LPVOID arg)
2609 rdpShadowClient* client = (rdpShadowClient*)arg;
2612 wMessage message = WINPR_C_ARRAY_INIT;
2613 wMessage pointerPositionMsg = WINPR_C_ARRAY_INIT;
2614 wMessage pointerAlphaMsg = WINPR_C_ARRAY_INIT;
2615 wMessage audioVolumeMsg = WINPR_C_ARRAY_INIT;
2616 HANDLE ChannelEvent =
nullptr;
2617 void* UpdateSubscriber =
nullptr;
2618 HANDLE UpdateEvent =
nullptr;
2619 freerdp_peer* peer =
nullptr;
2620 rdpContext* context =
nullptr;
2621 rdpSettings* settings =
nullptr;
2622 rdpShadowServer* server =
nullptr;
2623 rdpShadowSubsystem* subsystem =
nullptr;
2624 wMessageQueue* MsgQueue =
nullptr;
2626 SHADOW_GFX_STATUS gfxstatus = WINPR_C_ARRAY_INIT;
2627 rdpUpdate* update =
nullptr;
2629 WINPR_ASSERT(client);
2631 MsgQueue = client->MsgQueue;
2632 WINPR_ASSERT(MsgQueue);
2634 server = client->server;
2635 WINPR_ASSERT(server);
2636 subsystem = server->subsystem;
2637 context = (rdpContext*)client;
2638 peer = context->peer;
2640 WINPR_ASSERT(peer->context);
2642 settings = peer->context->settings;
2643 WINPR_ASSERT(settings);
2645 peer->Capabilities = shadow_client_capabilities;
2646 peer->PostConnect = shadow_client_post_connect;
2647 peer->Activate = shadow_client_activate;
2648 peer->Logon = shadow_client_logon;
2649 shadow_input_register_callbacks(peer->context->input);
2651 rc = peer->Initialize(peer);
2655 update = peer->context->update;
2656 WINPR_ASSERT(update);
2658 update->RefreshRect = shadow_client_refresh_rect;
2659 update->SuppressOutput = shadow_client_suppress_output;
2660 update->SurfaceFrameAcknowledge = shadow_client_surface_frame_acknowledge;
2662 if ((!client->vcm) || (!subsystem->updateEvent))
2665 UpdateSubscriber = shadow_multiclient_get_subscriber(subsystem->updateEvent);
2667 if (!UpdateSubscriber)
2670 UpdateEvent = shadow_multiclient_getevent(UpdateSubscriber);
2671 WINPR_ASSERT(UpdateEvent);
2673 ChannelEvent = WTSVirtualChannelManagerGetEventHandle(client->vcm);
2674 WINPR_ASSERT(ChannelEvent);
2686 HANDLE events[MAXIMUM_WAIT_OBJECTS] = WINPR_C_ARRAY_INIT;
2688 events[nCount++] = UpdateEvent;
2690 DWORD tmp = peer->GetEventHandles(peer, &events[nCount], 64 - nCount);
2694 WLog_ERR(TAG,
"Failed to get FreeRDP transport event handles");
2700 events[nCount++] = ChannelEvent;
2701 events[nCount++] = MessageQueue_Event(MsgQueue);
2703#if defined(CHANNEL_RDPGFX_SERVER)
2704 HANDLE gfxevent = rdpgfx_server_get_event_handle(client->rdpgfx);
2707 events[nCount++] = gfxevent;
2710 status = WaitForMultipleObjects(nCount, events, FALSE, INFINITE);
2712 if (status == WAIT_FAILED)
2715 if (WaitForSingleObject(UpdateEvent, 0) == WAIT_OBJECT_0)
2724 if (client->activated && !client->suppressOutput)
2729 if (shadow_client_recalc_desktop_size(client))
2732 if (!shadow_client_send_resize(client, &gfxstatus))
2734 WLog_ERR(TAG,
"Failed to send resize message");
2741 if (!shadow_client_send_surface_update(client, &gfxstatus))
2743 WLog_ERR(TAG,
"Failed to send surface update");
2751 if (!shadow_client_no_surface_update(client, &gfxstatus))
2753 WLog_ERR(TAG,
"Failed to handle surface update");
2762 (void)shadow_multiclient_consume(UpdateSubscriber);
2765 WINPR_ASSERT(peer->CheckFileDescriptor);
2766 if (!peer->CheckFileDescriptor(peer))
2768 WLog_ERR(TAG,
"Failed to check FreeRDP file descriptor");
2772 if (client->activated &&
2773 WTSVirtualChannelManagerIsChannelJoined(client->vcm, DRDYNVC_SVC_CHANNEL_NAME))
2775 switch (WTSVirtualChannelManagerGetDrdynvcState(client->vcm))
2778 case DRDYNVC_STATE_NONE:
2781 if (!WTSVirtualChannelManagerCheckFileDescriptor(client->vcm))
2783 WLog_ERR(TAG,
"Failed to initialize drdynvc channel");
2789 case DRDYNVC_STATE_READY:
2790#if defined(CHANNEL_AUDIN_SERVER)
2791 if (client->audin && !IFCALLRESULT(TRUE, client->audin->IsOpen, client->audin))
2793 if (!IFCALLRESULT(FALSE, client->audin->Open, client->audin))
2795 WLog_ERR(TAG,
"Failed to initialize audin channel");
2803 client->rdpgfx && !gfxstatus.gfxOpened)
2805 client->rdpgfx->FrameAcknowledge = shadow_client_rdpgfx_frame_acknowledge;
2806 client->rdpgfx->CapsAdvertise = shadow_client_rdpgfx_caps_advertise;
2808 if (!client->rdpgfx->Open(client->rdpgfx))
2810 WLog_WARN(TAG,
"Failed to open GraphicsPipeline");
2812 FreeRDP_SupportGraphicsPipeline, FALSE))
2817 gfxstatus.gfxOpened = TRUE;
2818 WLog_INFO(TAG,
"Gfx Pipeline Opened");
2829 if (WaitForSingleObject(ChannelEvent, 0) == WAIT_OBJECT_0)
2831 if (!WTSVirtualChannelManagerCheckFileDescriptor(client->vcm))
2833 WLog_ERR(TAG,
"WTSVirtualChannelManagerCheckFileDescriptor failure");
2838#if defined(CHANNEL_RDPGFX_SERVER)
2841 if (WaitForSingleObject(gfxevent, 0) == WAIT_OBJECT_0)
2843 const UINT error = rdpgfx_server_handle_messages(client->rdpgfx);
2844 if (error != CHANNEL_RC_OK)
2850 if (WaitForSingleObject(MessageQueue_Event(MsgQueue), 0) == WAIT_OBJECT_0)
2853 pointerPositionMsg.id = 0;
2854 pointerPositionMsg.Free =
nullptr;
2855 pointerAlphaMsg.id = 0;
2856 pointerAlphaMsg.Free =
nullptr;
2857 audioVolumeMsg.id = 0;
2858 audioVolumeMsg.Free =
nullptr;
2860 while (MessageQueue_Peek(MsgQueue, &message, TRUE))
2862 if (message.id == WMQ_QUIT)
2869 case SHADOW_MSG_OUT_POINTER_POSITION_UPDATE_ID:
2871 shadow_client_free_queued_message(&pointerPositionMsg);
2872 pointerPositionMsg = message;
2875 case SHADOW_MSG_OUT_POINTER_ALPHA_UPDATE_ID:
2877 shadow_client_free_queued_message(&pointerAlphaMsg);
2878 pointerAlphaMsg = message;
2881 case SHADOW_MSG_OUT_AUDIO_OUT_VOLUME_ID:
2883 shadow_client_free_queued_message(&audioVolumeMsg);
2884 audioVolumeMsg = message;
2888 if (!shadow_client_subsystem_process_message(client, &message))
2894 if (message.id == WMQ_QUIT)
2897 shadow_client_free_queued_message(&pointerPositionMsg);
2898 shadow_client_free_queued_message(&pointerAlphaMsg);
2899 shadow_client_free_queued_message(&audioVolumeMsg);
2905 if (pointerPositionMsg.id)
2907 if (!shadow_client_subsystem_process_message(client, &pointerPositionMsg))
2911 if (pointerAlphaMsg.id)
2913 if (!shadow_client_subsystem_process_message(client, &pointerAlphaMsg))
2917 if (audioVolumeMsg.id)
2919 if (!shadow_client_subsystem_process_message(client, &audioVolumeMsg))
2929#if defined(CHANNEL_AUDIN_SERVER)
2930 if (client->audin && !IFCALLRESULT(TRUE, client->audin->IsOpen, client->audin))
2932 if (!IFCALLRESULT(FALSE, client->audin->Close, client->audin))
2934 WLog_WARN(TAG,
"AUDIN shutdown failure!");
2939 if (gfxstatus.gfxOpened)
2941 if (gfxstatus.gfxSurfaceCreated)
2943 if (!shadow_client_rdpgfx_release_surface(client))
2944 WLog_WARN(TAG,
"GFX release surface failure!");
2947 WINPR_ASSERT(client->rdpgfx);
2948 WINPR_ASSERT(client->rdpgfx->Close);
2949 rc = client->rdpgfx->Close(client->rdpgfx);
2953 shadow_client_channels_free(client);
2955 if (UpdateSubscriber)
2957 shadow_multiclient_release_subscriber(UpdateSubscriber);
2958 UpdateSubscriber =
nullptr;
2961 if (peer->connected && subsystem->ClientDisconnect)
2963 subsystem->ClientDisconnect(subsystem, client);
2967 WINPR_ASSERT(peer->Disconnect);
2968 peer->Disconnect(peer);
2969 freerdp_peer_context_free(peer);
2970 freerdp_peer_free(peer);
2975BOOL shadow_client_accepted(freerdp_listener* listener, freerdp_peer* peer)
2977 rdpShadowClient* client =
nullptr;
2978 rdpShadowServer* server =
nullptr;
2980 if (!listener || !peer)
2983 server = (rdpShadowServer*)listener->info;
2984 WINPR_ASSERT(server);
2986 peer->ContextExtra = (
void*)server;
2987 peer->ContextSize =
sizeof(rdpShadowClient);
2988 peer->ContextNew = shadow_client_context_new;
2989 peer->ContextFree = shadow_client_context_free;
2991 if (!freerdp_peer_context_new_ex(peer, server->settings))
2994 client = (rdpShadowClient*)peer->context;
2995 WINPR_ASSERT(client);
2997 if (!(client->thread = CreateThread(
nullptr, 0, shadow_client_thread, client, 0,
nullptr)))
2999 freerdp_peer_context_free(peer);
3005 (void)CloseHandle(client->thread);
3006 client->thread =
nullptr;
3012static void shadow_msg_out_addref(wMessage* message)
3014 SHADOW_MSG_OUT* msg =
nullptr;
3016 WINPR_ASSERT(message);
3017 msg = (SHADOW_MSG_OUT*)message->wParam;
3020 InterlockedIncrement(&(msg->refCount));
3023static void shadow_msg_out_release(wMessage* message)
3025 SHADOW_MSG_OUT* msg =
nullptr;
3027 WINPR_ASSERT(message);
3028 msg = (SHADOW_MSG_OUT*)message->wParam;
3031 if (InterlockedDecrement(&(msg->refCount)) <= 0)
3033 IFCALL(msg->Free, message->id, msg);
3038static BOOL shadow_client_dispatch_msg(rdpShadowClient* client, wMessage* message)
3040 if (!client || !message)
3044 shadow_msg_out_addref(message);
3046 WINPR_ASSERT(client->MsgQueue);
3047 if (MessageQueue_Dispatch(client->MsgQueue, message))
3052 shadow_msg_out_release(message);
3057BOOL shadow_client_post_msg(rdpShadowClient* client,
void* context, UINT32 type,
3058 SHADOW_MSG_OUT* msg,
void* lParam)
3060 wMessage message = WINPR_C_ARRAY_INIT;
3061 message.context = context;
3063 message.wParam = (
void*)msg;
3064 message.lParam = lParam;
3065 message.Free = shadow_msg_out_release;
3066 return shadow_client_dispatch_msg(client, &message);
3069int shadow_client_boardcast_msg(rdpShadowServer* server,
void* context, UINT32 type,
3070 SHADOW_MSG_OUT* msg,
void* lParam)
3072 wMessage message = WINPR_C_ARRAY_INIT;
3075 WINPR_ASSERT(server);
3078 message.context = context;
3080 message.wParam = (
void*)msg;
3081 message.lParam = lParam;
3082 message.Free = shadow_msg_out_release;
3085 shadow_msg_out_addref(&message);
3087 WINPR_ASSERT(server->clients);
3088 ArrayList_Lock(server->clients);
3090 for (
size_t index = 0; index < ArrayList_Count(server->clients); index++)
3092 rdpShadowClient* client = (rdpShadowClient*)ArrayList_GetItem(server->clients, index);
3094 if (shadow_client_dispatch_msg(client, &message))
3100 ArrayList_Unlock(server->clients);
3102 shadow_msg_out_release(&message);
3106int shadow_client_boardcast_quit(rdpShadowServer* server,
int nExitCode)
3108 wMessageQueue* queue =
nullptr;
3111 WINPR_ASSERT(server);
3112 WINPR_ASSERT(server->clients);
3114 ArrayList_Lock(server->clients);
3116 for (
size_t index = 0; index < ArrayList_Count(server->clients); index++)
3118 queue = ((rdpShadowClient*)ArrayList_GetItem(server->clients, index))->MsgQueue;
3120 if (MessageQueue_PostQuit(queue, nExitCode))
3126 ArrayList_Unlock(server->clients);
WINPR_ATTR_NODISCARD FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_bool(rdpSettings *settings, FreeRDP_Settings_Keys_Bool id, BOOL val)
Sets a BOOL settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_uint32(rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id, UINT32 val)
Sets a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_set_string(rdpSettings *settings, FreeRDP_Settings_Keys_String id, const char *val)
Sets a string settings value. The param is copied.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.
This struct contains function pointer to initialize/free objects.