FreeRDP
Loading...
Searching...
No Matches
shadow_client.c
1
21#include <freerdp/config.h>
22
23#include <winpr/crt.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>
32
33#include <freerdp/log.h>
34#include <freerdp/utils/gfx.h>
35#include <freerdp/channels/drdynvc.h>
36
37#include "shadow.h"
38
39#define TAG CLIENT_TAG("shadow")
40
41typedef struct
42{
43 BOOL gfxOpened;
44 BOOL gfxSurfaceCreated;
45} SHADOW_GFX_STATUS;
46
47WINPR_ATTR_NODISCARD
48static BOOL shadow_avc420_enabled(const rdpShadowClient* client);
49WINPR_ATTR_NODISCARD
50static BOOL shadow_avc444_enabled(const rdpShadowClient* client);
51
52/* See https://github.com/FreeRDP/FreeRDP/issues/10413
53 *
54 * Microsoft ditched support for RFX and multiple rectangles in BitmapUpdate for
55 * windows 11 24H2.
56 *
57 * So send all updates only with a single rectangle.
58 */
59#define BitmapUpdateProxy(client, bitmap) \
60 BitmapUpdateProxyEx((client), (bitmap), __FILE__, __LINE__, __func__)
61WINPR_ATTR_NODISCARD
62static BOOL BitmapUpdateProxyEx(rdpShadowClient* client, const BITMAP_UPDATE* bitmap,
63 const char* file, size_t line, const char* fkt)
64{
65 WINPR_ASSERT(client);
66 WINPR_ASSERT(bitmap);
67
68 rdpShadowServer* server = client->server;
69 WINPR_ASSERT(server);
70
71 rdpContext* context = (rdpContext*)client;
72
73 rdpUpdate* update = context->update;
74 WINPR_ASSERT(update);
75
76 if (server->SupportMultiRectBitmapUpdates)
77 {
78 const BOOL rc = IFCALLRESULT(FALSE, update->BitmapUpdate, context, bitmap);
79 if (!rc)
80 {
81 const DWORD log_level = WLOG_ERROR;
82 wLog* log = WLog_Get(TAG);
83 if (WLog_IsLevelActive(log, log_level))
84 {
85 WLog_PrintTextMessage(log, log_level, line, file, fkt,
86 "BitmapUpdate[count %" PRIu32 "] failed", bitmap->number);
87 }
88 return FALSE;
89 }
90 }
91 else
92 {
93 for (UINT32 x = 0; x < bitmap->number; x++)
94 {
95 BITMAP_UPDATE cur = WINPR_C_ARRAY_INIT;
96 BITMAP_DATA* bmp = &bitmap->rectangles[x];
97 cur.rectangles = bmp;
98 cur.number = 1;
99 cur.skipCompression = bitmap->skipCompression;
100 const BOOL rc = IFCALLRESULT(FALSE, update->BitmapUpdate, context, &cur);
101 if (!rc)
102 {
103 const DWORD log_level = WLOG_ERROR;
104 wLog* log = WLog_Get(TAG);
105 if (WLog_IsLevelActive(log, log_level))
106 {
107 WLog_PrintTextMessage(log, log_level, line, file, fkt,
108 "BitmapUpdate[count 1, at %" PRIu32 "] failed", x);
109 }
110 return FALSE;
111 }
112 }
113 }
114
115 return TRUE;
116}
117
118WINPR_ATTR_NODISCARD
119static inline BOOL shadow_client_rdpgfx_new_surface(rdpShadowClient* client)
120{
121 UINT error = CHANNEL_RC_OK;
122 RDPGFX_CREATE_SURFACE_PDU createSurface;
123 RDPGFX_MAP_SURFACE_TO_OUTPUT_PDU surfaceToOutput;
124 RdpgfxServerContext* context = nullptr;
125 rdpSettings* settings = nullptr;
126
127 WINPR_ASSERT(client);
128 context = client->rdpgfx;
129 WINPR_ASSERT(context);
130 settings = ((rdpContext*)client)->settings;
131 WINPR_ASSERT(settings);
132
133 WINPR_ASSERT(freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth) <= UINT16_MAX);
134 WINPR_ASSERT(freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight) <= UINT16_MAX);
135 createSurface.width = (UINT16)freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
136 createSurface.height = (UINT16)freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
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);
144
145 if (error)
146 {
147 WLog_ERR(TAG, "CreateSurface failed with error %" PRIu32 "", error);
148 return FALSE;
149 }
150
151 IFCALLRET(context->MapSurfaceToOutput, error, context, &surfaceToOutput);
152
153 if (error)
154 {
155 WLog_ERR(TAG, "MapSurfaceToOutput failed with error %" PRIu32 "", error);
156 return FALSE;
157 }
158
159 return TRUE;
160}
161
162WINPR_ATTR_NODISCARD
163static inline BOOL shadow_client_rdpgfx_release_surface(rdpShadowClient* client)
164{
165 UINT error = CHANNEL_RC_OK;
167 RdpgfxServerContext* context = nullptr;
168
169 WINPR_ASSERT(client);
170
171 context = client->rdpgfx;
172 WINPR_ASSERT(context);
173
174 pdu.surfaceId = client->surfaceId++;
175 IFCALLRET(context->DeleteSurface, error, context, &pdu);
176
177 if (error)
178 {
179 WLog_ERR(TAG, "DeleteSurface failed with error %" PRIu32 "", error);
180 return FALSE;
181 }
182
183 return TRUE;
184}
185
186WINPR_ATTR_NODISCARD
187static inline BOOL shadow_client_rdpgfx_reset_graphic(rdpShadowClient* client)
188{
189 UINT error = CHANNEL_RC_OK;
190 RDPGFX_RESET_GRAPHICS_PDU pdu = WINPR_C_ARRAY_INIT;
191 RdpgfxServerContext* context = nullptr;
192 rdpSettings* settings = nullptr;
193
194 WINPR_ASSERT(client);
195 WINPR_ASSERT(client->rdpgfx);
196
197 context = client->rdpgfx;
198 WINPR_ASSERT(context);
199
200 settings = client->context.settings;
201 WINPR_ASSERT(settings);
202
203 pdu.width = freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
204 pdu.height = freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
205 pdu.monitorCount = client->subsystem->numMonitors;
206 pdu.monitorDefArray = client->subsystem->monitors;
207 IFCALLRET(context->ResetGraphics, error, context, &pdu);
208
209 if (error)
210 {
211 WLog_ERR(TAG, "ResetGraphics failed with error %" PRIu32 "", error);
212 return FALSE;
213 }
214
215 client->first_frame = TRUE;
216 return TRUE;
217}
218
219static inline void shadow_client_free_queued_message(void* obj)
220{
221 wMessage* message = (wMessage*)obj;
222
223 WINPR_ASSERT(message);
224 if (message->Free)
225 {
226 message->Free(message);
227 message->Free = nullptr;
228 }
229}
230
231static void shadow_client_context_free(freerdp_peer* peer, rdpContext* context)
232{
233 rdpShadowClient* client = (rdpShadowClient*)context;
234 rdpShadowServer* server = nullptr;
235
236 WINPR_UNUSED(peer);
237 if (!client)
238 return;
239
240 server = client->server;
241 if (server && server->clients)
242 ArrayList_Remove(server->clients, (void*)client);
243
244 shadow_encoder_free(client->encoder);
245
246 /* Clear queued messages and free resource */
247 MessageQueue_Free(client->MsgQueue);
248 WTSCloseServer(client->vcm);
249 region16_uninit(&(client->invalidRegion));
250 DeleteCriticalSection(&(client->lock));
251
252 client->MsgQueue = nullptr;
253 client->encoder = nullptr;
254 client->vcm = nullptr;
255}
256
257WINPR_ATTR_NODISCARD
258static BOOL shadow_client_context_new(freerdp_peer* peer, rdpContext* context)
259{
260 BOOL NSCodec = 0;
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 };
267
268 WINPR_ASSERT(client);
269 WINPR_ASSERT(peer);
270 WINPR_ASSERT(peer->context);
271
272 server = (rdpShadowServer*)peer->ContextExtra;
273 WINPR_ASSERT(server);
274
275 srvSettings = server->settings;
276 WINPR_ASSERT(srvSettings);
277
278 client->surfaceId = 1;
279 client->server = server;
280 client->subsystem = server->subsystem;
281 WINPR_ASSERT(client->subsystem);
282
283 settings = peer->context->settings;
284 WINPR_ASSERT(settings);
285
286 if (!freerdp_settings_set_uint32(settings, FreeRDP_ColorDepth,
287 freerdp_settings_get_uint32(srvSettings, FreeRDP_ColorDepth)))
288 return FALSE;
289 NSCodec = freerdp_settings_get_bool(srvSettings, FreeRDP_NSCodec);
290 if (!freerdp_settings_set_bool(settings, FreeRDP_NSCodec, NSCodec))
291 return FALSE;
292 if (!freerdp_settings_set_bool(settings, FreeRDP_RemoteFxCodec,
293 freerdp_settings_get_bool(srvSettings, FreeRDP_RemoteFxCodec)))
294 return FALSE;
296 settings, FreeRDP_RemoteFxRlgrMode,
297 freerdp_settings_get_uint32(srvSettings, FreeRDP_RemoteFxRlgrMode)))
298 return FALSE;
299 if (!freerdp_settings_set_bool(settings, FreeRDP_BitmapCacheV3Enabled, TRUE))
300 return FALSE;
301 if (!freerdp_settings_set_bool(settings, FreeRDP_FrameMarkerCommandEnabled, TRUE))
302 return FALSE;
303 if (!freerdp_settings_set_bool(settings, FreeRDP_SurfaceFrameMarkerEnabled, TRUE))
304 return FALSE;
306 settings, FreeRDP_SupportGraphicsPipeline,
307 freerdp_settings_get_bool(srvSettings, FreeRDP_SupportGraphicsPipeline)))
308 return FALSE;
309 if (!freerdp_settings_set_bool(settings, FreeRDP_GfxH264,
310 freerdp_settings_get_bool(srvSettings, FreeRDP_GfxH264)))
311 return FALSE;
312 if (!freerdp_settings_set_bool(settings, FreeRDP_DrawAllowSkipAlpha, TRUE))
313 return FALSE;
314 if (!freerdp_settings_set_bool(settings, FreeRDP_DrawAllowColorSubsampling, TRUE))
315 return FALSE;
316 if (!freerdp_settings_set_bool(settings, FreeRDP_DrawAllowDynamicColorFidelity, TRUE))
317 return FALSE;
318 if (!freerdp_settings_set_uint32(settings, FreeRDP_CompressionLevel, PACKET_COMPR_TYPE_RDP8))
319 return FALSE;
320
321 if (server->ipcSocket && (strncmp(bind_address, server->ipcSocket,
322 strnlen(bind_address, sizeof(bind_address))) != 0))
323 {
324 if (!freerdp_settings_set_bool(settings, FreeRDP_LyncRdpMode, TRUE))
325 return FALSE;
326 if (!freerdp_settings_set_bool(settings, FreeRDP_CompressionEnabled, FALSE))
327 return FALSE;
328 }
329
330 client->inLobby = TRUE;
331 client->mayView = server->mayView;
332 client->mayInteract = server->mayInteract;
333
334 if (!InitializeCriticalSectionAndSpinCount(&(client->lock), 4000))
335 goto fail;
336
337 region16_init(&(client->invalidRegion));
338 client->vcm = WTSOpenServerA((LPSTR)peer->context);
339
340 if (!client->vcm || client->vcm == INVALID_HANDLE_VALUE)
341 goto fail;
342
343 if (!(client->MsgQueue = MessageQueue_New(&cb)))
344 goto fail;
345
346 if (!(client->encoder = shadow_encoder_new(client)))
347 goto fail;
348
349 if (!ArrayList_Append(server->clients, (void*)client))
350 goto fail;
351
352 return TRUE;
353
354fail:
355 shadow_client_context_free(peer, context);
356 return FALSE;
357}
358
359static inline void shadow_client_mark_invalid(rdpShadowClient* client, UINT32 numRects,
360 const RECTANGLE_16* rects)
361{
362 WINPR_ASSERT(client);
363 WINPR_ASSERT(rects || (numRects == 0));
364
365 rdpSettings* settings = client->context.settings;
366 WINPR_ASSERT(settings);
367
368 EnterCriticalSection(&(client->lock));
369
370 /* Mark client invalid region. No rectangle means full screen */
371 if (numRects > 0)
372 {
373 for (UINT32 index = 0; index < numRects; index++)
374 {
375 if (!region16_union_rect(&(client->invalidRegion), &(client->invalidRegion),
376 &rects[index]))
377 goto fail;
378 }
379 }
380 else
381 {
382 RECTANGLE_16 screenRegion = WINPR_C_ARRAY_INIT;
383 WINPR_ASSERT(freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth) <= UINT16_MAX);
384 WINPR_ASSERT(freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight) <= UINT16_MAX);
385 screenRegion.right = (UINT16)freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
386 screenRegion.bottom = (UINT16)freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
387 if (!region16_union_rect(&(client->invalidRegion), &(client->invalidRegion), &screenRegion))
388 goto fail;
389 }
390fail:
391 LeaveCriticalSection(&(client->lock));
392}
393
400WINPR_ATTR_NODISCARD
401static inline BOOL shadow_client_recalc_desktop_size(rdpShadowClient* client)
402{
403 INT32 width = 0;
404 INT32 height = 0;
405 rdpShadowServer* server = nullptr;
406 rdpSettings* settings = nullptr;
407 RECTANGLE_16 viewport = WINPR_C_ARRAY_INIT;
408
409 WINPR_ASSERT(client);
410 server = client->server;
411 settings = client->context.settings;
412
413 WINPR_ASSERT(server);
414 WINPR_ASSERT(server->surface);
415 WINPR_ASSERT(settings);
416
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;
421
422 if (server->shareSubRect)
423 {
424 if (!rectangles_intersection(&viewport, &(server->subRect), &viewport))
425 return FALSE;
426 }
427
428 width = viewport.right - viewport.left;
429 height = viewport.bottom - viewport.top;
430
431 WINPR_ASSERT(width >= 0);
432 WINPR_ASSERT(width <= UINT16_MAX);
433 WINPR_ASSERT(height >= 0);
434 WINPR_ASSERT(height <= UINT16_MAX);
435 return (freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth) != (UINT32)width ||
436 freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight) != (UINT32)height);
437}
438
439WINPR_ATTR_NODISCARD
440static BOOL shadow_client_capabilities(freerdp_peer* peer)
441{
442 rdpShadowSubsystem* subsystem = nullptr;
443 rdpShadowClient* client = nullptr;
444 BOOL ret = TRUE;
445
446 WINPR_ASSERT(peer);
447
448 client = (rdpShadowClient*)peer->context;
449 WINPR_ASSERT(client);
450 WINPR_ASSERT(client->server);
451
452 subsystem = client->server->subsystem;
453 WINPR_ASSERT(subsystem);
454
455 IFCALLRET(subsystem->ClientCapabilities, ret, subsystem, client);
456
457 if (!ret)
458 WLog_WARN(TAG, "subsystem->ClientCapabilities failed");
459
460 return ret;
461}
462
463static void shadow_reset_desktop_resize(rdpShadowClient* client)
464{
465 WINPR_ASSERT(client);
466 client->resizeRequested = FALSE;
467}
468
469WINPR_ATTR_NODISCARD
470static BOOL shadow_send_desktop_resize(rdpShadowClient* client)
471{
472 BOOL rc = 0;
473 rdpUpdate* update = nullptr;
474 rdpSettings* settings = nullptr;
475 const freerdp_peer* peer = nullptr;
476
477 WINPR_ASSERT(client);
478
479 settings = client->context.settings;
480 peer = client->context.peer;
481 WINPR_ASSERT(peer);
482 WINPR_ASSERT(client->server);
483 WINPR_ASSERT(client->server->surface);
484
485 const UINT32 resizeWidth = client->server->surface->width;
486 const UINT32 resizeHeight = client->server->surface->height;
487
488 if (client->resizeRequested)
489 {
490 if ((resizeWidth == client->resizeWidth) && (resizeHeight == client->resizeHeight))
491 {
492 const UINT32 w = freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
493 const UINT32 h = freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
494 WLog_WARN(TAG,
495 "detected previous resize request for resolution %" PRIu32 "x%" PRIu32
496 ", still have %" PRIu32 "x%" PRIu32 ", disconnecting peer",
497 resizeWidth, resizeHeight, w, h);
498 return FALSE;
499 }
500 }
501
502 update = client->context.update;
503 WINPR_ASSERT(update);
504 WINPR_ASSERT(update->DesktopResize);
505
506 // Update peer resolution, required so that during disconnect/reconnect the correct resolution
507 // is sent to the client.
508 if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopWidth, resizeWidth))
509 return FALSE;
510 if (!freerdp_settings_set_uint32(settings, FreeRDP_DesktopHeight, resizeHeight))
511 return FALSE;
512 rc = update->DesktopResize(update->context);
513 WLog_INFO(TAG, "Client %s resize requested (%" PRIu32 "x%" PRIu32 "@%" PRIu32 ")",
514 peer->hostname, resizeWidth, resizeHeight,
515 freerdp_settings_get_uint32(settings, FreeRDP_ColorDepth));
516 client->resizeRequested = TRUE;
517 client->resizeWidth = resizeWidth;
518 client->resizeHeight = resizeHeight;
519
520 return rc;
521}
522
523WINPR_ATTR_NODISCARD
524static BOOL shadow_client_post_connect(freerdp_peer* peer)
525{
526 int authStatus = 0;
527 rdpSettings* settings = nullptr;
528 rdpShadowClient* client = nullptr;
529 rdpShadowServer* server = nullptr;
530 rdpShadowSubsystem* subsystem = nullptr;
531
532 WINPR_ASSERT(peer);
533
534 client = (rdpShadowClient*)peer->context;
535 WINPR_ASSERT(client);
536
537 settings = peer->context->settings;
538 WINPR_ASSERT(settings);
539
540 server = client->server;
541 WINPR_ASSERT(server);
542
543 subsystem = server->subsystem;
544 WINPR_ASSERT(subsystem);
545
546 if (freerdp_settings_get_uint32(settings, FreeRDP_ColorDepth) == 24)
547 {
548 if (!freerdp_settings_set_uint32(settings, FreeRDP_ColorDepth, 16)) /* disable 24bpp */
549 return FALSE;
550 }
551
552 const UINT32 MultifragMaxRequestSize =
553 freerdp_settings_get_uint32(settings, FreeRDP_MultifragMaxRequestSize);
554 if (MultifragMaxRequestSize < 0x3F0000)
555 {
557 settings, FreeRDP_NSCodec,
558 FALSE); /* NSCodec compressor does not support fragmentation yet */
559 if (!rc)
560 return FALSE;
561 }
562
563 WLog_INFO(TAG, "Client from %s is activated (%" PRIu32 "x%" PRIu32 "@%" PRIu32 ")",
564 peer->hostname, freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth),
565 freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight),
566 freerdp_settings_get_uint32(settings, FreeRDP_ColorDepth));
567
568 if (shadow_client_channels_post_connect(client) != CHANNEL_RC_OK)
569 return FALSE;
570
571 shadow_client_mark_invalid(client, 0, nullptr);
572 authStatus = -1;
573
574 const char* Username = freerdp_settings_get_string(settings, FreeRDP_Username);
575 const char* Domain = freerdp_settings_get_string(settings, FreeRDP_Domain);
576 const char* Password = freerdp_settings_get_string(settings, FreeRDP_Password);
577
578 if (Username && Password)
579 {
580 if (!freerdp_settings_set_bool(settings, FreeRDP_AutoLogonEnabled, TRUE))
581 return FALSE;
582 }
583
584 if (server->authentication && !freerdp_settings_get_bool(settings, FreeRDP_NlaSecurity))
585 {
586 if (subsystem->Authenticate)
587 {
588 authStatus = subsystem->Authenticate(subsystem, client, Username, Domain, Password);
589 }
590
591 if (authStatus < 0)
592 {
593 WLog_ERR(TAG, "client authentication failure: %d", authStatus);
594 return FALSE;
595 }
596 }
597
598 if (subsystem->ClientConnect)
599 {
600 return subsystem->ClientConnect(subsystem, client);
601 }
602
603 return TRUE;
604}
605
606/* Convert rects in sub rect coordinate to client/surface coordinate */
607static inline void shadow_client_convert_rects(rdpShadowClient* client, RECTANGLE_16* dst,
608 const RECTANGLE_16* src, UINT32 numRects)
609{
610 WINPR_ASSERT(client);
611 WINPR_ASSERT(client->server);
612 WINPR_ASSERT(dst);
613 WINPR_ASSERT(src || (numRects == 0));
614
615 if (client->server->shareSubRect)
616 {
617 UINT16 offsetX = client->server->subRect.left;
618 UINT16 offsetY = client->server->subRect.top;
619
620 for (UINT32 i = 0; i < numRects; i++)
621 {
622 const RECTANGLE_16* s = &src[i];
623 RECTANGLE_16* d = &dst[i];
624
625 d->left = s->left + offsetX;
626 d->right = s->right + offsetX;
627 d->top = s->top + offsetY;
628 d->bottom = s->bottom + offsetY;
629 }
630 }
631 else
632 {
633 if (src != dst)
634 {
635 CopyMemory(dst, src, numRects * sizeof(RECTANGLE_16));
636 }
637 }
638}
639
640WINPR_ATTR_NODISCARD
641static BOOL shadow_client_refresh_request(rdpShadowClient* client)
642{
643 wMessage message = WINPR_C_ARRAY_INIT;
644 wMessagePipe* MsgPipe = nullptr;
645
646 WINPR_ASSERT(client);
647 WINPR_ASSERT(client->subsystem);
648
649 MsgPipe = client->subsystem->MsgPipe;
650 WINPR_ASSERT(MsgPipe);
651
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);
658}
659
660WINPR_ATTR_NODISCARD
661static BOOL shadow_client_refresh_rect(rdpContext* context, BYTE count, const RECTANGLE_16* areas)
662{
663 rdpShadowClient* client = (rdpShadowClient*)context;
664 RECTANGLE_16* rects = nullptr;
665
666 /* It is invalid if we have area count but no actual area */
667 if (count && !areas)
668 return FALSE;
669
670 if (count)
671 {
672 rects = (RECTANGLE_16*)calloc(count, sizeof(RECTANGLE_16));
673
674 if (!rects)
675 {
676 return FALSE;
677 }
678
679 shadow_client_convert_rects(client, rects, areas, count);
680 shadow_client_mark_invalid(client, count, rects);
681 free(rects);
682 }
683 else
684 {
685 shadow_client_mark_invalid(client, 0, nullptr);
686 }
687
688 return shadow_client_refresh_request(client);
689}
690
691WINPR_ATTR_NODISCARD
692static BOOL shadow_client_suppress_output(rdpContext* context, BYTE allow, const RECTANGLE_16* area)
693{
694 rdpShadowClient* client = (rdpShadowClient*)context;
695 RECTANGLE_16 region;
696
697 WINPR_ASSERT(client);
698
699 client->suppressOutput = !(allow);
700
701 if (allow)
702 {
703 if (area)
704 {
705 shadow_client_convert_rects(client, &region, area, 1);
706 shadow_client_mark_invalid(client, 1, &region);
707 }
708 else
709 {
710 shadow_client_mark_invalid(client, 0, nullptr);
711 }
712 }
713
714 return shadow_client_refresh_request(client);
715}
716
717WINPR_ATTR_NODISCARD
718static BOOL shadow_client_activate(freerdp_peer* peer)
719{
720 WINPR_ASSERT(peer);
721
722 rdpShadowClient* client = (rdpShadowClient*)peer->context;
723 WINPR_ASSERT(client);
724
725 /* Resize client if necessary */
726 if (shadow_client_recalc_desktop_size(client))
727 return shadow_send_desktop_resize(client);
728
729 shadow_reset_desktop_resize(client);
730 client->activated = TRUE;
731 client->inLobby = !(client->mayView);
732
733 if (shadow_encoder_reset(client->encoder) < 0)
734 {
735 WLog_ERR(TAG, "Failed to reset encoder");
736 return FALSE;
737 }
738
739 /* Update full screen in next update */
740 return shadow_client_refresh_rect(&client->context, 0, nullptr);
741}
742
743WINPR_ATTR_NODISCARD
744static BOOL shadow_client_logon(freerdp_peer* peer, const SEC_WINNT_AUTH_IDENTITY* identity,
745 BOOL automatic)
746{
747 BOOL rc = FALSE;
748 char* user = nullptr;
749 char* domain = nullptr;
750 char* password = nullptr;
751 rdpSettings* settings = nullptr;
752
753 WINPR_UNUSED(automatic);
754
755 WINPR_ASSERT(peer);
756 WINPR_ASSERT(identity);
757
758 WINPR_ASSERT(peer->context);
759
760 settings = peer->context->settings;
761 WINPR_ASSERT(settings);
762
763 if (identity->Flags & SEC_WINNT_AUTH_IDENTITY_UNICODE)
764 {
765 if (identity->User)
766 user = ConvertWCharNToUtf8Alloc(identity->User, identity->UserLength, nullptr);
767
768 if (identity->Domain)
769 domain = ConvertWCharNToUtf8Alloc(identity->Domain, identity->DomainLength, nullptr);
770
771 if (identity->Password)
772 password =
773 ConvertWCharNToUtf8Alloc(identity->Password, identity->PasswordLength, nullptr);
774 }
775 else
776 {
777 if (identity->User)
778 user = _strdup((char*)identity->User);
779
780 if (identity->Domain)
781 domain = _strdup((char*)identity->Domain);
782
783 if (identity->Password)
784 password = _strdup((char*)identity->Password);
785 }
786
787 if ((identity->User && !user) || (identity->Domain && !domain) ||
788 (identity->Password && !password))
789 goto fail;
790
791 if (user)
792 {
793 if (!freerdp_settings_set_string(settings, FreeRDP_Username, user))
794 goto fail;
795 }
796
797 if (domain)
798 {
799 if (!freerdp_settings_set_string(settings, FreeRDP_Domain, domain))
800 goto fail;
801 }
802 if (password)
803 {
804 if (!freerdp_settings_set_string(settings, FreeRDP_Password, password))
805 goto fail;
806 }
807 rc = TRUE;
808fail:
809 free(user);
810 free(domain);
811 free(password);
812 return rc;
813}
814
815static inline void shadow_client_common_frame_acknowledge(rdpShadowClient* client, UINT32 frameId)
816{
817 /*
818 * Record the last client acknowledged frame id to
819 * calculate how much frames are in progress.
820 * Some rdp clients (win7 mstsc) skips frame ACK if it is
821 * inactive, we should not expect ACK for each frame.
822 * So it is OK to calculate in-flight frame count according to
823 * a latest acknowledged frame id.
824 */
825 WINPR_ASSERT(client);
826 WINPR_ASSERT(client->encoder);
827 client->encoder->lastAckframeId = frameId;
828}
829
830WINPR_ATTR_NODISCARD
831static BOOL shadow_client_surface_frame_acknowledge(rdpContext* context, UINT32 frameId)
832{
833 rdpShadowClient* client = (rdpShadowClient*)context;
834 shadow_client_common_frame_acknowledge(client, frameId);
835 /*
836 * Reset queueDepth for legacy none RDPGFX acknowledge
837 */
838 WINPR_ASSERT(client);
839 WINPR_ASSERT(client->encoder);
840 client->encoder->queueDepth = QUEUE_DEPTH_UNAVAILABLE;
841 return TRUE;
842}
843
844WINPR_ATTR_NODISCARD
845static UINT
846shadow_client_rdpgfx_frame_acknowledge(RdpgfxServerContext* context,
847 const RDPGFX_FRAME_ACKNOWLEDGE_PDU* frameAcknowledge)
848{
849 rdpShadowClient* client = nullptr;
850
851 WINPR_ASSERT(context);
852 WINPR_ASSERT(frameAcknowledge);
853
854 client = (rdpShadowClient*)context->custom;
855 shadow_client_common_frame_acknowledge(client, frameAcknowledge->frameId);
856
857 WINPR_ASSERT(client);
858 WINPR_ASSERT(client->encoder);
859 client->encoder->queueDepth = frameAcknowledge->queueDepth;
860 return CHANNEL_RC_OK;
861}
862
863WINPR_ATTR_NODISCARD
864static BOOL shadow_are_caps_filtered(const rdpSettings* settings, UINT32 caps)
865{
866 const UINT32 capList[] = {
867#if defined(WITH_GFX_AV1)
868 RDPGFX_CAPVERSION_FRDP_1,
869#endif
870 RDPGFX_CAPVERSION_8, RDPGFX_CAPVERSION_81, RDPGFX_CAPVERSION_10,
871 RDPGFX_CAPVERSION_101, RDPGFX_CAPVERSION_102, RDPGFX_CAPVERSION_103,
872 RDPGFX_CAPVERSION_104, RDPGFX_CAPVERSION_105, RDPGFX_CAPVERSION_106,
873 RDPGFX_CAPVERSION_106_ERR, RDPGFX_CAPVERSION_107
874 };
875
876 WINPR_ASSERT(settings);
877 const UINT32 filter = freerdp_settings_get_uint32(settings, FreeRDP_GfxCapsFilter);
878
879 for (UINT32 x = 0; x < ARRAYSIZE(capList); x++)
880 {
881 if (caps == capList[x])
882 return (filter & (1u << x)) != 0;
883 }
884
885 return TRUE;
886}
887
888WINPR_ATTR_NODISCARD
889static UINT shadow_client_send_caps_confirm(RdpgfxServerContext* context, rdpShadowClient* client,
890 const RDPGFX_CAPS_CONFIRM_PDU* pdu)
891{
892 WINPR_ASSERT(context);
893 WINPR_ASSERT(client);
894 WINPR_ASSERT(pdu);
895
896 WINPR_ASSERT(context->CapsConfirm);
897 UINT rc = context->CapsConfirm(context, pdu);
898 client->areGfxCapsReady = (rc == CHANNEL_RC_OK);
899 client->confirmedCaps = *pdu->capsSet;
900
901 rdpSettings* clientSettings = client->context.settings;
902 WINPR_ASSERT(clientSettings);
903
904#if defined(WITH_GFX_AV1)
905 if (pdu->capsSet->version == RDPGFX_CAPVERSION_FRDP_1)
906 {
907 UINT32 flags = FREERDP_CODEC_AV1_I420;
908 if ((pdu->capsSet->flags & RDPGFX_CAPS_FLAG_AV1_I444_DISABLED) == 0)
909 flags = FREERDP_CODEC_AV1_I444;
910
911 if (shadow_encoder_prepare(client->encoder, flags) < 0)
912 return ERROR_INVALID_PARAMETER;
913 }
914 else
915#endif
916 if (shadow_avc444_enabled(client) || shadow_avc420_enabled(client))
917 {
918#ifdef WITH_GFX_H264
919 const BOOL h264 = (shadow_encoder_prepare(client->encoder, FREERDP_CODEC_AVC420 |
920 FREERDP_CODEC_AVC444) >= 0);
921 if (h264)
922 {
924 clientSettings, FreeRDP_GfxH264,
925 (pdu->capsSet->flags & RDPGFX_CAPS_FLAG_AVC420_ENABLED) != 0))
926 return rc;
927 }
928 else
929 {
930 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxH264, FALSE))
931 return rc;
932 }
933#else
934 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxAVC444v2, FALSE))
935 return rc;
936 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxAVC444, FALSE))
937 return rc;
938 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxH264, FALSE))
939 return rc;
940#endif
941 }
942
943 return rc;
944}
945
946WINPR_ATTR_NODISCARD
947static BOOL shadow_client_caps_test_version(RdpgfxServerContext* context, rdpShadowClient* client,
948 BOOL h264, const RDPGFX_CAPSET* capsSets,
949 UINT32 capsSetCount, UINT32 capsVersion, UINT* rc)
950{
951 WINPR_ASSERT(context);
952 WINPR_ASSERT(client);
953 WINPR_ASSERT(capsSets || (capsSetCount == 0));
954 WINPR_ASSERT(rc);
955
956 WINPR_ASSERT(context->rdpcontext);
957 const rdpSettings* srvSettings = context->rdpcontext->settings;
958 WINPR_ASSERT(srvSettings);
959
960 rdpSettings* clientSettings = client->context.settings;
961 WINPR_ASSERT(clientSettings);
962
963 if (shadow_are_caps_filtered(srvSettings, capsVersion))
964 return FALSE;
965
966 for (UINT32 index = 0; index < capsSetCount; index++)
967 {
968 const RDPGFX_CAPSET* currentCaps = &capsSets[index];
969
970 WLog_INFO(TAG, "testing %s vs %s[0x%08" PRIx32 "]",
971 rdpgfx_caps_version_str(currentCaps->version),
972 rdpgfx_caps_version_str(capsVersion), capsVersion);
973 if (currentCaps->version == capsVersion)
974 {
975 RDPGFX_CAPSET caps = *currentCaps;
976 RDPGFX_CAPS_CONFIRM_PDU pdu = WINPR_C_ARRAY_INIT;
977 pdu.capsSet = &caps;
978
979 const UINT32 flags = pdu.capsSet->flags;
980
981 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxSmallCache,
982 (flags & RDPGFX_CAPS_FLAG_SMALL_CACHE) != 0))
983 return FALSE;
984
985 BOOL avc420 = TRUE;
986 BOOL avc444 = !(flags & RDPGFX_CAPS_FLAG_AVC_DISABLED);
987 BOOL avc444v2 = avc444;
988 if (!freerdp_settings_get_bool(srvSettings, FreeRDP_GfxAVC444v2) || !h264)
989 avc444v2 = FALSE;
990 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxAVC444v2, avc444v2))
991 return FALSE;
992 if (!freerdp_settings_get_bool(srvSettings, FreeRDP_GfxAVC444) || !h264)
993 avc444 = FALSE;
994 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxAVC444, avc444))
995 return FALSE;
996 if (!freerdp_settings_get_bool(srvSettings, FreeRDP_GfxH264) || !h264)
997 avc420 = FALSE;
998 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxH264, avc420))
999 return FALSE;
1000
1001 const BOOL progressive = freerdp_settings_get_bool(srvSettings, FreeRDP_GfxProgressive);
1002 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxProgressive, progressive))
1003 return FALSE;
1004 const BOOL progressivev2 =
1005 freerdp_settings_get_bool(srvSettings, FreeRDP_GfxProgressiveV2);
1006 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxProgressiveV2, progressivev2))
1007 return FALSE;
1008
1009 const BOOL rfx = freerdp_settings_get_bool(srvSettings, FreeRDP_RemoteFxCodec);
1010 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_RemoteFxCodec, rfx))
1011 return FALSE;
1012
1013 const BOOL planar = freerdp_settings_get_bool(srvSettings, FreeRDP_GfxPlanar);
1014 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxPlanar, planar))
1015 return FALSE;
1016#if defined(WITH_GFX_AV1)
1017 const BOOL av1 = freerdp_settings_get_bool(srvSettings, FreeRDP_GfxCodecAV1);
1018 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxCodecAV1, av1))
1019 return FALSE;
1020 const UINT32 av1Profile =
1021 freerdp_settings_get_uint32(srvSettings, FreeRDP_GfxCodecAV1Profile);
1022 if (!freerdp_settings_set_uint32(clientSettings, FreeRDP_GfxCodecAV1Profile,
1023 av1Profile))
1024 return FALSE;
1025#endif
1026
1027 if (!h264 || (!avc444v2 && !avc444 && !avc420))
1028 pdu.capsSet->flags |= RDPGFX_CAPS_FLAG_AVC_DISABLED;
1029
1030#if defined(WITH_GFX_AV1)
1031 if (currentCaps->version == RDPGFX_CAPVERSION_FRDP_1)
1032 {
1033 if (!av1)
1034 return FALSE;
1035
1036 pdu.capsSet->flags &=
1037 ~(RDPGFX_CAPS_FLAG_AV1_I444_SUPPORTED | RDPGFX_CAPS_FLAG_AV1_I444_DISABLED);
1038
1039 if ((currentCaps->flags & RDPGFX_CAPS_FLAG_AV1_I444_SUPPORTED) != 0)
1040 {
1041 if (av1Profile == 0)
1042 return FALSE;
1043 }
1044
1045 if ((av1Profile == 0) ||
1046 ((currentCaps->flags & RDPGFX_CAPS_FLAG_AV1_I444_SUPPORTED) == 0))
1047 pdu.capsSet->flags |= RDPGFX_CAPS_FLAG_AV1_I444_DISABLED;
1048 if ((currentCaps->flags & RDPGFX_CAPS_FLAG_AV1_I444_DISABLED) != 0)
1049 pdu.capsSet->flags |= RDPGFX_CAPS_FLAG_AV1_I444_DISABLED;
1050 }
1051#endif
1052 *rc = shadow_client_send_caps_confirm(context, client, &pdu);
1053 return TRUE;
1054 }
1055 }
1056
1057 return FALSE;
1058}
1059
1065WINPR_ATTR_NODISCARD
1066static UINT shadow_client_rdpgfx_caps_advertise(RdpgfxServerContext* context,
1067 const RDPGFX_CAPS_ADVERTISE_PDU* capsAdvertise)
1068{
1069 UINT rc = ERROR_INTERNAL_ERROR;
1070
1071 UINT32 flags = 0;
1072
1073 WINPR_ASSERT(context);
1074 WINPR_ASSERT(capsAdvertise);
1075
1076 rdpShadowClient* client = (rdpShadowClient*)context->custom;
1077 WINPR_ASSERT(client);
1078 WINPR_ASSERT(context->rdpcontext);
1079
1080 const rdpSettings* srvSettings = context->rdpcontext->settings;
1081 WINPR_ASSERT(srvSettings);
1082
1083 rdpSettings* clientSettings = client->context.settings;
1084 WINPR_ASSERT(clientSettings);
1085
1086 /* Request full screen update for new gfx channel */
1087 if (!shadow_client_refresh_rect(&client->context, 0, nullptr))
1088 return rc;
1089
1090 const UINT32 capsVersions[] = {
1091#if defined(WITH_GFX_AV1)
1092 RDPGFX_CAPVERSION_FRDP_1,
1093#endif
1094 RDPGFX_CAPVERSION_107, RDPGFX_CAPVERSION_106, RDPGFX_CAPVERSION_106_ERR,
1095 RDPGFX_CAPVERSION_105, RDPGFX_CAPVERSION_104, RDPGFX_CAPVERSION_103,
1096 RDPGFX_CAPVERSION_102, RDPGFX_CAPVERSION_101, RDPGFX_CAPVERSION_10,
1097 };
1098
1099 const BOOL h264 =
1100 shadow_encoder_prepare(client->encoder, FREERDP_CODEC_AVC420 | FREERDP_CODEC_AVC444) >= 0;
1101
1102#if defined(WITH_GFX_AV1)
1103 if (freerdp_settings_get_bool(clientSettings, FreeRDP_GfxCodecAV1))
1104 {
1105 UINT32 codec = 0;
1106 const UINT32 profile =
1107 freerdp_settings_get_uint32(clientSettings, FreeRDP_GfxCodecAV1Profile);
1108 switch (profile)
1109 {
1110 case 0:
1111 codec = FREERDP_CODEC_AV1_I420;
1112 break;
1113 case 1:
1114 codec = FREERDP_CODEC_AV1_I444;
1115 break;
1116 default:
1117 return ERROR_BAD_PROFILE;
1118 }
1119 const BOOL av1 = shadow_encoder_prepare(client->encoder, codec) >= 0;
1120 if (!av1)
1121 {
1122 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxCodecAV1, FALSE))
1123 return ERROR_INTERNAL_ERROR;
1124 }
1125 }
1126#endif
1127
1128 for (size_t x = 0; x < ARRAYSIZE(capsVersions); x++)
1129 {
1130 const UINT32 cur = capsVersions[x];
1131 if (shadow_client_caps_test_version(context, client, h264, capsAdvertise->capsSets,
1132 capsAdvertise->capsSetCount, cur, &rc))
1133 return rc;
1134 }
1135
1136 if (!shadow_are_caps_filtered(srvSettings, RDPGFX_CAPVERSION_81))
1137 {
1138 for (UINT32 index = 0; index < capsAdvertise->capsSetCount; index++)
1139 {
1140 const RDPGFX_CAPSET* currentCaps = &capsAdvertise->capsSets[index];
1141
1142 if (currentCaps->version == RDPGFX_CAPVERSION_81)
1143 {
1144 RDPGFX_CAPSET caps = *currentCaps;
1146 pdu.capsSet = &caps;
1147
1148 flags = pdu.capsSet->flags;
1149
1150 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxAVC444v2, FALSE))
1151 return rc;
1152 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxAVC444, FALSE))
1153 return rc;
1154
1155 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxThinClient,
1156 (flags & RDPGFX_CAPS_FLAG_THINCLIENT) != 0))
1157 return rc;
1158 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxSmallCache,
1159 (flags & RDPGFX_CAPS_FLAG_SMALL_CACHE) != 0))
1160 return rc;
1161
1162#ifndef WITH_GFX_H264
1163 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxH264, FALSE))
1164 return rc;
1165 if (h264)
1166 pdu.capsSet->flags &= ~RDPGFX_CAPS_FLAG_AVC420_ENABLED;
1167#else
1168
1169#endif
1170
1171 return shadow_client_send_caps_confirm(context, client, &pdu);
1172 }
1173 }
1174 }
1175
1176 if (!shadow_are_caps_filtered(srvSettings, RDPGFX_CAPVERSION_8))
1177 {
1178 for (UINT32 index = 0; index < capsAdvertise->capsSetCount; index++)
1179 {
1180 const RDPGFX_CAPSET* currentCaps = &capsAdvertise->capsSets[index];
1181
1182 if (currentCaps->version == RDPGFX_CAPVERSION_8)
1183 {
1184 RDPGFX_CAPSET caps = *currentCaps;
1186 pdu.capsSet = &caps;
1187 flags = pdu.capsSet->flags;
1188
1189 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxAVC444v2, FALSE))
1190 return rc;
1191 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxAVC444, FALSE))
1192 return rc;
1193 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxH264, FALSE))
1194 return rc;
1195
1196 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxThinClient,
1197 (flags & RDPGFX_CAPS_FLAG_THINCLIENT) != 0))
1198 return rc;
1199 if (!freerdp_settings_set_bool(clientSettings, FreeRDP_GfxSmallCache,
1200 (flags & RDPGFX_CAPS_FLAG_SMALL_CACHE) != 0))
1201 return rc;
1202
1203 return shadow_client_send_caps_confirm(context, client, &pdu);
1204 }
1205 }
1206 }
1207
1208 return CHANNEL_RC_UNSUPPORTED_VERSION;
1209}
1210
1211WINPR_ATTR_NODISCARD
1212static inline UINT32 rdpgfx_estimate_h264_avc420(RDPGFX_AVC420_BITMAP_STREAM* havc420)
1213{
1214 /* H264 metadata + H264 stream. See rdpgfx_write_h264_avc420 */
1215 WINPR_ASSERT(havc420);
1216 return sizeof(UINT32) /* numRegionRects */
1217 + 10ULL /* regionRects + quantQualityVals */
1218 * havc420->meta.numRegionRects +
1219 havc420->length;
1220}
1221
1222#if defined(WITH_GFX_AV1)
1223WINPR_ATTR_NODISCARD
1224static BOOL shadow_client_send_av1(rdpShadowClient* client, const BYTE* pSrcData, UINT32 nSrcStep,
1225 UINT32 SrcFormat, UINT16 nWidth, UINT16 nHeight,
1227 const RDPGFX_START_FRAME_PDU* cmdstart,
1228 const RDPGFX_END_FRAME_PDU* cmdend)
1229{
1230 WINPR_ASSERT(client);
1231
1232 rdpShadowEncoder* encoder = client->encoder;
1233 WINPR_ASSERT(encoder);
1234
1235 UINT error = CHANNEL_RC_OK;
1236 INT32 rc = 0;
1237 RDPGFX_AVC420_BITMAP_STREAM avc420 = WINPR_C_ARRAY_INIT;
1238 RECTANGLE_16 regionRect = WINPR_C_ARRAY_INIT;
1239
1240 UINT32 flags = FREERDP_CODEC_AV1_I444;
1241 if ((client->confirmedCaps.flags & RDPGFX_CAPS_FLAG_AV1_I444_DISABLED) != 0)
1242 flags = FREERDP_CODEC_AV1_I420;
1243
1244 if (shadow_encoder_prepare(encoder, flags) < 0)
1245 {
1246 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_AV1");
1247 return FALSE;
1248 }
1249
1250 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1251 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1252 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1253 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1254 regionRect.left = (UINT16)cmd->left;
1255 regionRect.top = (UINT16)cmd->top;
1256 regionRect.right = (UINT16)cmd->right;
1257 regionRect.bottom = (UINT16)cmd->bottom;
1258 rc = freerdp_av1_compress(encoder->av1, pSrcData, SrcFormat, nSrcStep, nWidth, nHeight,
1259 &regionRect, &avc420.data, &avc420.length, &avc420.meta);
1260 if (rc < 0)
1261 {
1262 WLog_ERR(TAG, "freerdp_av1_compress failed");
1263 return FALSE;
1264 }
1265
1266 /* rc > 0 means new data */
1267 if (rc > 0)
1268 {
1269 cmd->codecId = RDPGFX_CODECID_AV1;
1270 cmd->extra = (void*)&avc420;
1271
1272 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1273 cmdend);
1274 cmd->extra = nullptr;
1275 }
1276 free_h264_metablock(&avc420.meta);
1277
1278 if (error)
1279 {
1280 WLog_ERR(TAG, "SurfaceFrameCommand failed with error %" PRIu32 "", error);
1281 return FALSE;
1282 }
1283 return TRUE;
1284}
1285#endif
1286
1287WINPR_ATTR_NODISCARD
1288static BOOL shadow_client_send_avc444(rdpShadowClient* client, const BYTE* pSrcData,
1289 UINT32 nSrcStep, UINT32 SrcFormat, UINT16 nWidth,
1290 UINT16 nHeight, RDPGFX_SURFACE_COMMAND* cmd,
1291 const RDPGFX_START_FRAME_PDU* cmdstart,
1292 const RDPGFX_END_FRAME_PDU* cmdend)
1293{
1294 WINPR_ASSERT(client);
1295
1296 rdpShadowEncoder* encoder = client->encoder;
1297 WINPR_ASSERT(encoder);
1298
1299 UINT error = CHANNEL_RC_OK;
1300 INT32 rc = 0;
1301 RDPGFX_AVC444_BITMAP_STREAM avc444 = WINPR_C_ARRAY_INIT;
1302 RECTANGLE_16 regionRect = WINPR_C_ARRAY_INIT;
1303 const BOOL GfxAVC444v2 = (cmd->codecId == RDPGFX_CODECID_AVC444v2);
1304 BYTE version = GfxAVC444v2 ? 2 : 1;
1305
1306 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_AVC444) < 0)
1307 {
1308 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_AVC444");
1309 return FALSE;
1310 }
1311
1312 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1313 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1314 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1315 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1316 regionRect.left = (UINT16)cmd->left;
1317 regionRect.top = (UINT16)cmd->top;
1318 regionRect.right = (UINT16)cmd->right;
1319 regionRect.bottom = (UINT16)cmd->bottom;
1320 rc = avc444_compress(encoder->h264, pSrcData, SrcFormat, nSrcStep, nWidth, nHeight, version,
1321 &regionRect, &avc444.LC, &avc444.bitstream[0].data,
1322 &avc444.bitstream[0].length, &avc444.bitstream[1].data,
1323 &avc444.bitstream[1].length, &avc444.bitstream[0].meta,
1324 &avc444.bitstream[1].meta);
1325 if (rc < 0)
1326 {
1327 WLog_ERR(TAG, "avc420_compress failed for avc444");
1328 return FALSE;
1329 }
1330
1331 /* rc > 0 means new data */
1332 if (rc > 0)
1333 {
1334 avc444.cbAvc420EncodedBitstream1 = rdpgfx_estimate_h264_avc420(&avc444.bitstream[0]);
1335 cmd->extra = (void*)&avc444;
1336 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1337 cmdend);
1338 cmd->extra = nullptr;
1339 }
1340
1341 free_h264_metablock(&avc444.bitstream[0].meta);
1342 free_h264_metablock(&avc444.bitstream[1].meta);
1343 if (error)
1344 {
1345 WLog_ERR(TAG, "SurfaceFrameCommand failed with error %" PRIu32 "", error);
1346 return FALSE;
1347 }
1348 return TRUE;
1349}
1350
1351WINPR_ATTR_NODISCARD
1352static BOOL shadow_client_send_avc420(rdpShadowClient* client, const BYTE* pSrcData,
1353 UINT32 nSrcStep, UINT32 SrcFormat, UINT16 nWidth,
1354 UINT16 nHeight, RDPGFX_SURFACE_COMMAND* cmd,
1355 const RDPGFX_START_FRAME_PDU* cmdstart,
1356 const RDPGFX_END_FRAME_PDU* cmdend)
1357{
1358 WINPR_ASSERT(client);
1359
1360 rdpShadowEncoder* encoder = client->encoder;
1361 WINPR_ASSERT(encoder);
1362
1363 UINT error = CHANNEL_RC_OK;
1364 INT32 rc = 0;
1365 RDPGFX_AVC420_BITMAP_STREAM avc420 = WINPR_C_ARRAY_INIT;
1366 RECTANGLE_16 regionRect;
1367
1368 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_AVC420) < 0)
1369 {
1370 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_AVC420");
1371 return FALSE;
1372 }
1373
1374 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1375 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1376 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1377 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1378 regionRect.left = (UINT16)cmd->left;
1379 regionRect.top = (UINT16)cmd->top;
1380 regionRect.right = (UINT16)cmd->right;
1381 regionRect.bottom = (UINT16)cmd->bottom;
1382 rc = avc420_compress(encoder->h264, pSrcData, SrcFormat, nSrcStep, nWidth, nHeight, &regionRect,
1383 &avc420.data, &avc420.length, &avc420.meta);
1384 if (rc < 0)
1385 {
1386 WLog_ERR(TAG, "avc420_compress failed");
1387 return FALSE;
1388 }
1389
1390 /* rc > 0 means new data */
1391 if (rc > 0)
1392 {
1393 cmd->codecId = RDPGFX_CODECID_AVC420;
1394 cmd->extra = (void*)&avc420;
1395 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1396 cmdend);
1397 cmd->extra = nullptr;
1398 }
1399 free_h264_metablock(&avc420.meta);
1400
1401 if (error)
1402 {
1403 WLog_ERR(TAG, "SurfaceFrameCommand failed with error %" PRIu32 "", error);
1404 return FALSE;
1405 }
1406 return TRUE;
1407}
1408
1409WINPR_ATTR_NODISCARD
1410static BOOL shadow_client_send_rfx(rdpShadowClient* client, const BYTE* pSrcData, UINT32 nSrcStep,
1411 UINT32 SrcFormat, UINT16 nWidth, UINT16 nHeight,
1413 const RDPGFX_START_FRAME_PDU* cmdstart,
1414 const RDPGFX_END_FRAME_PDU* cmdend)
1415{
1416 WINPR_ASSERT(client);
1417
1418 rdpShadowEncoder* encoder = client->encoder;
1419 WINPR_ASSERT(encoder);
1420
1421 UINT error = CHANNEL_RC_OK;
1422 BOOL rc = 0;
1423 RFX_RECT rect = { 0 };
1424
1425 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_REMOTEFX) < 0)
1426 {
1427 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_REMOTEFX");
1428 return FALSE;
1429 }
1430
1431 const UINT32 rfxFormat = rfx_context_get_pixel_format(encoder->rfx);
1432 if (rfxFormat != SrcFormat)
1433 {
1434 WLog_ERR(TAG, "RFX context pixel format mismatch %s, expected %s",
1435 FreeRDPGetColorFormatName(rfxFormat), FreeRDPGetColorFormatName(SrcFormat));
1436 return FALSE;
1437 }
1438
1439 wStream* s = Stream_New(nullptr, 1024);
1440 WINPR_ASSERT(s);
1441
1442 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1443 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1444 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1445 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1446 rect.x = (UINT16)cmd->left;
1447 rect.y = (UINT16)cmd->top;
1448 rect.width = WINPR_ASSERTING_INT_CAST(UINT16, cmd->right - cmd->left);
1449 rect.height = WINPR_ASSERTING_INT_CAST(UINT16, cmd->bottom - cmd->top);
1450
1451 rc = rfx_compose_message(encoder->rfx, s, &rect, 1, pSrcData, nWidth, nHeight, nSrcStep);
1452
1453 if (!rc)
1454 {
1455 WLog_ERR(TAG, "rfx_compose_message failed");
1456 Stream_Free(s, TRUE);
1457 return FALSE;
1458 }
1459
1460 /* rc > 0 means new data */
1461 if (rc > 0)
1462 {
1463 const size_t pos = Stream_GetPosition(s);
1464 WINPR_ASSERT(pos <= UINT32_MAX);
1465
1466 cmd->codecId = RDPGFX_CODECID_CAVIDEO;
1467 cmd->data = Stream_Buffer(s);
1468 cmd->length = (UINT32)pos;
1469
1470 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1471 cmdend);
1472 cmd->data = nullptr;
1473 }
1474
1475 Stream_Free(s, TRUE);
1476 if (error)
1477 {
1478 WLog_ERR(TAG, "SurfaceFrameCommand failed with error %" PRIu32 "", error);
1479 return FALSE;
1480 }
1481 return TRUE;
1482}
1483
1484WINPR_ATTR_NODISCARD
1485static BOOL shadow_client_send_progressive(rdpShadowClient* client, const BYTE* pSrcData,
1486 UINT32 nSrcStep, UINT32 SrcFormat, UINT16 nWidth,
1487 UINT16 nHeight, RDPGFX_SURFACE_COMMAND* cmd,
1488 const RDPGFX_START_FRAME_PDU* cmdstart,
1489 const RDPGFX_END_FRAME_PDU* cmdend)
1490{
1491 WINPR_ASSERT(client);
1492
1493 rdpShadowEncoder* encoder = client->encoder;
1494 WINPR_ASSERT(encoder);
1495
1496 UINT error = CHANNEL_RC_OK;
1497 INT32 rc = 0;
1498 REGION16 region;
1499 RECTANGLE_16 regionRect;
1500
1501 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_PROGRESSIVE) < 0)
1502 {
1503 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_PROGRESSIVE");
1504 return FALSE;
1505 }
1506
1507 WINPR_ASSERT(cmd->left <= UINT16_MAX);
1508 WINPR_ASSERT(cmd->top <= UINT16_MAX);
1509 WINPR_ASSERT(cmd->right <= UINT16_MAX);
1510 WINPR_ASSERT(cmd->bottom <= UINT16_MAX);
1511 regionRect.left = (UINT16)cmd->left;
1512 regionRect.top = (UINT16)cmd->top;
1513 regionRect.right = (UINT16)cmd->right;
1514 regionRect.bottom = (UINT16)cmd->bottom;
1515 region16_init(&region);
1516 if (!region16_union_rect(&region, &region, &regionRect))
1517 {
1518 region16_uninit(&region);
1519 return FALSE;
1520 }
1521 rc = progressive_compress(encoder->progressive, pSrcData, nSrcStep * nHeight, SrcFormat, nWidth,
1522 nHeight, nSrcStep, &region, &cmd->data, &cmd->length);
1523 region16_uninit(&region);
1524 if (rc < 0)
1525 {
1526 WLog_ERR(TAG, "progressive_compress failed");
1527 return FALSE;
1528 }
1529
1530 /* rc > 0 means new data */
1531 if (rc > 0)
1532 {
1533 cmd->codecId = RDPGFX_CODECID_CAPROGRESSIVE;
1534
1535 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart,
1536 cmdend);
1537 }
1538 cmd->data = nullptr;
1539
1540 if (error)
1541 {
1542 WLog_ERR(TAG, "SurfaceFrameCommand failed with error %" PRIu32 "", error);
1543 return FALSE;
1544 }
1545 return TRUE;
1546}
1547
1548WINPR_ATTR_NODISCARD
1549static BOOL shadow_client_send_planar(rdpShadowClient* client, const BYTE* pSrcData,
1550 UINT32 nSrcStep, UINT32 SrcFormat,
1552 const RDPGFX_START_FRAME_PDU* cmdstart,
1553 const RDPGFX_END_FRAME_PDU* cmdend)
1554{
1555 WINPR_ASSERT(client);
1556
1557 rdpShadowEncoder* encoder = client->encoder;
1558 WINPR_ASSERT(encoder);
1559
1560 UINT error = CHANNEL_RC_OK;
1561 const UINT32 w = cmd->right - cmd->left;
1562 const UINT32 h = cmd->bottom - cmd->top;
1563 const BYTE* src =
1564 &pSrcData[cmd->top * nSrcStep + cmd->left * FreeRDPGetBytesPerPixel(SrcFormat)];
1565 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_PLANAR) < 0)
1566 {
1567 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_PLANAR");
1568 return FALSE;
1569 }
1570
1571 const BOOL rc = freerdp_bitmap_planar_context_reset(encoder->planar, w, h);
1572 if (!rc)
1573 return FALSE;
1574
1575 freerdp_planar_topdown_image(encoder->planar, TRUE);
1576
1577 cmd->data = freerdp_bitmap_compress_planar(encoder->planar, src, SrcFormat, w, h, nSrcStep,
1578 nullptr, &cmd->length);
1579 WINPR_ASSERT(cmd->data || (cmd->length == 0));
1580
1581 cmd->codecId = RDPGFX_CODECID_PLANAR;
1582
1583 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart, cmdend);
1584 free(cmd->data);
1585 cmd->data = nullptr;
1586
1587 if (error)
1588 {
1589 WLog_ERR(TAG, "SurfaceFrameCommand failed with error %" PRIu32 "", error);
1590 return FALSE;
1591 }
1592 return TRUE;
1593}
1594
1595WINPR_ATTR_NODISCARD
1596static BOOL shadow_client_send_uncompressed(rdpShadowClient* client, const BYTE* pSrcData,
1597 UINT32 nSrcStep, UINT32 SrcFormat,
1599 const RDPGFX_START_FRAME_PDU* cmdstart,
1600 const RDPGFX_END_FRAME_PDU* cmdend)
1601{
1602 WINPR_ASSERT(client);
1603
1604 UINT error = CHANNEL_RC_OK;
1605 const UINT32 w = cmd->right - cmd->left;
1606 const UINT32 h = cmd->bottom - cmd->top;
1607 const UINT32 length = w * 4 * h;
1608
1609 BYTE* data = malloc(length);
1610 if (!data)
1611 return FALSE;
1612
1613 BOOL rc = freerdp_image_copy_no_overlap(data, cmd->format, 0, 0, 0, w, h, pSrcData, SrcFormat,
1614 nSrcStep, cmd->left, cmd->top, nullptr, 0);
1615 if (!rc)
1616 {
1617 free(data);
1618 return FALSE;
1619 }
1620
1621 cmd->data = data;
1622 cmd->length = length;
1623 cmd->codecId = RDPGFX_CODECID_UNCOMPRESSED;
1624
1625 IFCALLRET(client->rdpgfx->SurfaceFrameCommand, error, client->rdpgfx, cmd, cmdstart, cmdend);
1626 free(data);
1627 cmd->data = nullptr;
1628 if (error)
1629 {
1630 WLog_ERR(TAG, "SurfaceFrameCommand failed with error %" PRIu32 "", error);
1631 return FALSE;
1632 }
1633 return TRUE;
1634}
1635
1636WINPR_ATTR_NODISCARD
1637static BOOL shadow_avc444_enabled(const rdpShadowClient* client)
1638{
1639 WINPR_ASSERT(client);
1640 switch (client->confirmedCaps.version)
1641 {
1642 case RDPGFX_CAPVERSION_10:
1643 case RDPGFX_CAPVERSION_101:
1644 case RDPGFX_CAPVERSION_102:
1645 case RDPGFX_CAPVERSION_103:
1646 case RDPGFX_CAPVERSION_104:
1647 case RDPGFX_CAPVERSION_105:
1648 case RDPGFX_CAPVERSION_106:
1649 case RDPGFX_CAPVERSION_107:
1650 return (client->confirmedCaps.flags & RDPGFX_CAPS_FLAG_AVC_DISABLED) == 0;
1651
1652 default:
1653 return FALSE;
1654 }
1655}
1656
1657WINPR_ATTR_NODISCARD
1658static BOOL shadow_avc420_enabled(const rdpShadowClient* client)
1659{
1660 if (shadow_avc444_enabled(client))
1661 return TRUE;
1662 if (client->confirmedCaps.version != RDPGFX_CAPVERSION_81)
1663 return FALSE;
1664 return (client->confirmedCaps.flags & RDPGFX_CAPS_FLAG_AVC420_ENABLED) != 0;
1665}
1666
1672WINPR_ATTR_NODISCARD
1673static BOOL shadow_client_send_surface_gfx(rdpShadowClient* client, const BYTE* pSrcData,
1674 UINT32 nSrcStep, UINT32 SrcFormat, UINT16 nXSrc,
1675 UINT16 nYSrc, UINT16 nWidth, UINT16 nHeight)
1676{
1677 const rdpContext* context = (const rdpContext*)client;
1678 RDPGFX_SURFACE_COMMAND cmd = WINPR_C_ARRAY_INIT;
1679 RDPGFX_START_FRAME_PDU cmdstart = WINPR_C_ARRAY_INIT;
1680 RDPGFX_END_FRAME_PDU cmdend = WINPR_C_ARRAY_INIT;
1681 SYSTEMTIME sTime = WINPR_C_ARRAY_INIT;
1682
1683 if (!context || !pSrcData)
1684 return FALSE;
1685
1686 const rdpSettings* settings = context->settings;
1687 rdpShadowEncoder* encoder = client->encoder;
1688
1689 if (!settings || !encoder)
1690 return FALSE;
1691
1692 if (client->first_frame)
1693 {
1694 if (encoder->rfx)
1695 {
1696 if (!rfx_context_reset(encoder->rfx, nWidth, nHeight))
1697 return FALSE;
1698 }
1699 client->first_frame = FALSE;
1700 }
1701
1702 cmdstart.frameId = shadow_encoder_create_frame_id(encoder);
1703 GetSystemTime(&sTime);
1704 cmdstart.timestamp = (UINT32)(sTime.wHour << 22U | sTime.wMinute << 16U | sTime.wSecond << 10U |
1705 sTime.wMilliseconds);
1706 cmdend.frameId = cmdstart.frameId;
1707 cmd.surfaceId = client->surfaceId;
1708 cmd.format = PIXEL_FORMAT_BGRX32;
1709 cmd.left = nXSrc;
1710 cmd.top = nYSrc;
1711 cmd.right = cmd.left + nWidth;
1712 cmd.bottom = cmd.top + nHeight;
1713 cmd.width = nWidth;
1714 cmd.height = nHeight;
1715
1716#if defined(WITH_GFX_AV1)
1717 if (freerdp_settings_get_bool(settings, FreeRDP_GfxCodecAV1))
1718 {
1719 if (client->confirmedCaps.version == RDPGFX_CAPVERSION_FRDP_1)
1720 {
1721 return shadow_client_send_av1(client, pSrcData, nSrcStep, SrcFormat, nWidth, nHeight,
1722 &cmd, &cmdstart, &cmdend);
1723 }
1724 }
1725#endif
1726
1727 const UINT32 id = freerdp_settings_get_uint32(settings, FreeRDP_RemoteFxCodecId);
1728#ifdef WITH_GFX_H264
1729 const BOOL GfxH264 = freerdp_settings_get_bool(settings, FreeRDP_GfxH264);
1730 const BOOL GfxAVC444 = freerdp_settings_get_bool(settings, FreeRDP_GfxAVC444);
1731 const BOOL GfxAVC444v2 = freerdp_settings_get_bool(settings, FreeRDP_GfxAVC444v2);
1732 if (GfxAVC444 || GfxAVC444v2)
1733 {
1734 if (shadow_avc444_enabled(client))
1735 {
1736 cmd.codecId = GfxAVC444v2 ? RDPGFX_CODECID_AVC444v2 : RDPGFX_CODECID_AVC444;
1737 return shadow_client_send_avc444(client, pSrcData, nSrcStep, SrcFormat, nWidth, nHeight,
1738 &cmd, &cmdstart, &cmdend);
1739 }
1740 }
1741
1742 if (GfxH264 && shadow_avc420_enabled(client))
1743 {
1744 return shadow_client_send_avc420(client, pSrcData, nSrcStep, SrcFormat, nWidth, nHeight,
1745 &cmd, &cmdstart, &cmdend);
1746 }
1747
1748#endif
1749 if (freerdp_settings_get_bool(settings, FreeRDP_RemoteFxCodec) && (id != 0))
1750 {
1751 return shadow_client_send_rfx(client, pSrcData, nSrcStep, SrcFormat, nWidth, nHeight, &cmd,
1752 &cmdstart, &cmdend);
1753 }
1754
1755 if (freerdp_settings_get_bool(settings, FreeRDP_GfxProgressive))
1756 {
1757 return shadow_client_send_progressive(client, pSrcData, nSrcStep, SrcFormat, nWidth,
1758 nHeight, &cmd, &cmdstart, &cmdend);
1759 }
1760
1761 if (freerdp_settings_get_bool(settings, FreeRDP_GfxPlanar))
1762 {
1763 return shadow_client_send_planar(client, pSrcData, nSrcStep, SrcFormat, &cmd, &cmdstart,
1764 &cmdend);
1765 }
1766
1767 return shadow_client_send_uncompressed(client, pSrcData, nSrcStep, SrcFormat, &cmd, &cmdstart,
1768 &cmdend);
1769}
1770
1771WINPR_ATTR_NODISCARD
1772static BOOL stream_surface_bits_supported(const rdpSettings* settings)
1773{
1774 const UINT32 supported =
1775 freerdp_settings_get_uint32(settings, FreeRDP_SurfaceCommandsSupported);
1776 return ((supported & SURFCMDS_STREAM_SURFACE_BITS) != 0);
1777}
1778
1779WINPR_ATTR_NODISCARD
1780static BOOL set_surface_bits_supported(const rdpSettings* settings)
1781{
1782 const UINT32 supported =
1783 freerdp_settings_get_uint32(settings, FreeRDP_SurfaceCommandsSupported);
1784 return ((supported & SURFCMDS_SET_SURFACE_BITS) != 0);
1785}
1786
1787WINPR_ATTR_NODISCARD
1788static BOOL is_surface_command_supported(const rdpSettings* settings)
1789{
1790 if (stream_surface_bits_supported(settings))
1791 {
1792 const UINT32 rfxID = freerdp_settings_get_uint32(settings, FreeRDP_RemoteFxCodecId);
1793 const BOOL supported = freerdp_settings_get_bool(settings, FreeRDP_RemoteFxCodec);
1794 if (supported && (rfxID != 0))
1795 return TRUE;
1796 }
1797 if (set_surface_bits_supported(settings))
1798 {
1799 const UINT32 nsID = freerdp_settings_get_uint32(settings, FreeRDP_NSCodecId);
1800 const BOOL supported = freerdp_settings_get_bool(settings, FreeRDP_NSCodec);
1801 if (supported && (nsID != 0))
1802 return TRUE;
1803 }
1804 return FALSE;
1805}
1806
1812WINPR_ATTR_NODISCARD
1813static BOOL shadow_client_send_surface_bits(rdpShadowClient* client, BYTE* pSrcData,
1814 UINT32 nSrcStep, UINT16 nXSrc, UINT16 nYSrc,
1815 UINT16 nWidth, UINT16 nHeight)
1816{
1817 BOOL ret = TRUE;
1818 BOOL first = 0;
1819 BOOL last = 0;
1820 wStream* s = nullptr;
1821 size_t numMessages = 0;
1822 UINT32 frameId = 0;
1823 rdpUpdate* update = nullptr;
1824 rdpContext* context = (rdpContext*)client;
1825 rdpSettings* settings = nullptr;
1826 rdpShadowEncoder* encoder = nullptr;
1827 SURFACE_BITS_COMMAND cmd = WINPR_C_ARRAY_INIT;
1828
1829 if (!context || !pSrcData)
1830 return FALSE;
1831
1832 update = context->update;
1833 settings = context->settings;
1834 encoder = client->encoder;
1835
1836 if (!update || !settings || !encoder)
1837 return FALSE;
1838
1839 if (encoder->frameAck)
1840 frameId = shadow_encoder_create_frame_id(encoder);
1841
1842 // TODO: Check FreeRDP_RemoteFxCodecMode if we should send RFX IMAGE or VIDEO data
1843 const UINT32 nsID = freerdp_settings_get_uint32(settings, FreeRDP_NSCodecId);
1844 const UINT32 rfxID = freerdp_settings_get_uint32(settings, FreeRDP_RemoteFxCodecId);
1845 if (stream_surface_bits_supported(settings) &&
1846 freerdp_settings_get_bool(settings, FreeRDP_RemoteFxCodec) && (rfxID != 0))
1847 {
1848 RFX_RECT rect = WINPR_C_ARRAY_INIT;
1849
1850 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_REMOTEFX) < 0)
1851 {
1852 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_REMOTEFX");
1853 return FALSE;
1854 }
1855
1856 s = encoder->bs;
1857 rect.x = nXSrc;
1858 rect.y = nYSrc;
1859 rect.width = nWidth;
1860 rect.height = nHeight;
1861
1862 const UINT32 MultifragMaxRequestSize =
1863 freerdp_settings_get_uint32(settings, FreeRDP_MultifragMaxRequestSize);
1864 RFX_MESSAGE_LIST* messages =
1865 rfx_encode_messages(encoder->rfx, &rect, 1, pSrcData,
1866 freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth),
1867 freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight),
1868 nSrcStep, &numMessages, MultifragMaxRequestSize);
1869 if (!messages)
1870 {
1871 WLog_ERR(TAG, "rfx_encode_messages failed");
1872 return FALSE;
1873 }
1874
1875 cmd.cmdType = CMDTYPE_STREAM_SURFACE_BITS;
1876 WINPR_ASSERT(rfxID <= UINT16_MAX);
1877 cmd.bmp.codecID = (UINT16)rfxID;
1878 cmd.destLeft = 0;
1879 cmd.destTop = 0;
1880 cmd.destRight = freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
1881 cmd.destBottom = freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
1882 cmd.bmp.bpp = 32;
1883 cmd.bmp.flags = 0;
1884 WINPR_ASSERT(freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth) <= UINT16_MAX);
1885 WINPR_ASSERT(freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight) <= UINT16_MAX);
1886 cmd.bmp.width = (UINT16)freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
1887 cmd.bmp.height = (UINT16)freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
1888 cmd.skipCompression = TRUE;
1889
1890 for (size_t i = 0; i < numMessages; i++)
1891 {
1892 Stream_ResetPosition(s);
1893
1894 const RFX_MESSAGE* msg = rfx_message_list_get(messages, i);
1895 if (!rfx_write_message(encoder->rfx, s, msg))
1896 {
1897 WLog_ERR(TAG, "rfx_write_message failed");
1898 ret = FALSE;
1899 break;
1900 }
1901
1902 cmd.bmp.bitmapDataLength = WINPR_ASSERTING_INT_CAST(UINT32, Stream_GetPosition(s));
1903 cmd.bmp.bitmapData = Stream_Buffer(s);
1904 first = (i == 0);
1905 last = ((i + 1) == numMessages);
1906
1907 if (!encoder->frameAck)
1908 IFCALLRET(update->SurfaceBits, ret, update->context, &cmd);
1909 else
1910 IFCALLRET(update->SurfaceFrameBits, ret, update->context, &cmd, first, last,
1911 frameId);
1912
1913 if (!ret)
1914 {
1915 WLog_ERR(TAG, "Send surface bits(RemoteFxCodec) failed");
1916 break;
1917 }
1918 }
1919
1920 rfx_message_list_free(messages);
1921 }
1922 else if (set_surface_bits_supported(settings) &&
1923 freerdp_settings_get_bool(settings, FreeRDP_NSCodec) && (nsID != 0))
1924 {
1925 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_NSCODEC) < 0)
1926 {
1927 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_NSCODEC");
1928 return FALSE;
1929 }
1930
1931 s = encoder->bs;
1932 Stream_ResetPosition(s);
1933 pSrcData = &pSrcData[(nYSrc * nSrcStep) + (nXSrc * 4)];
1934 if (!nsc_compose_message(encoder->nsc, s, pSrcData, nWidth, nHeight, nSrcStep))
1935 return FALSE;
1936
1937 cmd.cmdType = CMDTYPE_SET_SURFACE_BITS;
1938 cmd.bmp.bpp = 32;
1939 WINPR_ASSERT(nsID <= UINT16_MAX);
1940 cmd.bmp.codecID = (UINT16)nsID;
1941 cmd.destLeft = nXSrc;
1942 cmd.destTop = nYSrc;
1943 cmd.destRight = cmd.destLeft + nWidth;
1944 cmd.destBottom = cmd.destTop + nHeight;
1945 cmd.bmp.width = nWidth;
1946 cmd.bmp.height = nHeight;
1947
1948 cmd.bmp.bitmapDataLength = WINPR_ASSERTING_INT_CAST(UINT32, Stream_GetPosition(s));
1949 cmd.bmp.bitmapData = Stream_Buffer(s);
1950 first = TRUE;
1951 last = TRUE;
1952
1953 if (!encoder->frameAck)
1954 IFCALLRET(update->SurfaceBits, ret, update->context, &cmd);
1955 else
1956 IFCALLRET(update->SurfaceFrameBits, ret, update->context, &cmd, first, last, frameId);
1957
1958 if (!ret)
1959 {
1960 WLog_ERR(TAG, "Send surface bits(NSCodec) failed");
1961 }
1962 }
1963
1964 return ret;
1965}
1966
1972WINPR_ATTR_NODISCARD
1973static BOOL shadow_client_send_bitmap_update(rdpShadowClient* client, BYTE* pSrcData,
1974 UINT32 nSrcStep, UINT16 nXSrc, UINT16 nYSrc,
1975 UINT16 nWidth, UINT16 nHeight)
1976{
1977 BOOL ret = TRUE;
1978 BYTE* data = nullptr;
1979 BYTE* buffer = nullptr;
1980 UINT32 k = 0;
1981 UINT32 yIdx = 0;
1982 UINT32 xIdx = 0;
1983 UINT32 rows = 0;
1984 UINT32 cols = 0;
1985 UINT32 DstSize = 0;
1986 UINT32 SrcFormat = 0;
1987 BITMAP_DATA* bitmap = nullptr;
1988 rdpContext* context = (rdpContext*)client;
1989 UINT32 totalBitmapSize = 0;
1990 UINT32 updateSizeEstimate = 0;
1991 BITMAP_DATA* bitmapData = nullptr;
1992 BITMAP_UPDATE bitmapUpdate = WINPR_C_ARRAY_INIT;
1993
1994 if (!context || !pSrcData)
1995 return FALSE;
1996
1997 rdpUpdate* update = context->update;
1998 rdpSettings* settings = context->settings;
1999 rdpShadowEncoder* encoder = client->encoder;
2000
2001 if (!update || !settings || !encoder)
2002 return FALSE;
2003
2004 const UINT32 maxUpdateSize =
2005 freerdp_settings_get_uint32(settings, FreeRDP_MultifragMaxRequestSize);
2006 if (freerdp_settings_get_uint32(settings, FreeRDP_ColorDepth) < 32)
2007 {
2008 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_INTERLEAVED) < 0)
2009 {
2010 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_INTERLEAVED");
2011 return FALSE;
2012 }
2013 }
2014 else
2015 {
2016 if (shadow_encoder_prepare(encoder, FREERDP_CODEC_PLANAR) < 0)
2017 {
2018 WLog_ERR(TAG, "Failed to prepare encoder FREERDP_CODEC_PLANAR");
2019 return FALSE;
2020 }
2021 }
2022
2023 SrcFormat = PIXEL_FORMAT_BGRX32;
2024
2025 if ((nXSrc % 4) != 0)
2026 {
2027 nWidth += (nXSrc % 4);
2028 nXSrc -= (nXSrc % 4);
2029 }
2030
2031 if ((nYSrc % 4) != 0)
2032 {
2033 nHeight += (nYSrc % 4);
2034 nYSrc -= (nYSrc % 4);
2035 }
2036
2037 rows = (nHeight / 64) + ((nHeight % 64) ? 1 : 0);
2038 cols = (nWidth / 64) + ((nWidth % 64) ? 1 : 0);
2039 k = 0;
2040 totalBitmapSize = 0;
2041 bitmapUpdate.number = rows * cols;
2042
2043 if (!(bitmapData = (BITMAP_DATA*)calloc(bitmapUpdate.number, sizeof(BITMAP_DATA))))
2044 return FALSE;
2045
2046 bitmapUpdate.rectangles = bitmapData;
2047
2048 if ((nWidth % 4) != 0)
2049 {
2050 nWidth += (4 - (nWidth % 4));
2051 }
2052
2053 if ((nHeight % 4) != 0)
2054 {
2055 nHeight += (4 - (nHeight % 4));
2056 }
2057
2058 for (yIdx = 0; yIdx < rows; yIdx++)
2059 {
2060 for (xIdx = 0; xIdx < cols; xIdx++)
2061 {
2062 bitmap = &bitmapData[k];
2063 bitmap->width = 64;
2064 bitmap->height = 64;
2065 bitmap->destLeft = nXSrc + (xIdx * 64);
2066 bitmap->destTop = nYSrc + (yIdx * 64);
2067
2068 if (((INT64)bitmap->destLeft + bitmap->width) > (nXSrc + nWidth))
2069 bitmap->width = (UINT32)(nXSrc + nWidth) - bitmap->destLeft;
2070
2071 if (((INT64)bitmap->destTop + bitmap->height) > (nYSrc + nHeight))
2072 bitmap->height = (UINT32)(nYSrc + nHeight) - bitmap->destTop;
2073
2074 bitmap->destRight = bitmap->destLeft + bitmap->width - 1;
2075 bitmap->destBottom = bitmap->destTop + bitmap->height - 1;
2076 bitmap->compressed = TRUE;
2077
2078 if ((bitmap->width < 4) || (bitmap->height < 4))
2079 continue;
2080
2081 if (freerdp_settings_get_uint32(settings, FreeRDP_ColorDepth) < 32)
2082 {
2083 UINT32 bitsPerPixel = freerdp_settings_get_uint32(settings, FreeRDP_ColorDepth);
2084 UINT32 bytesPerPixel = (bitsPerPixel + 7) / 8;
2085 DstSize = 64 * 64 * 4;
2086 buffer = encoder->grid[k];
2087
2088 ret = interleaved_compress(
2089 encoder->interleaved, buffer, &DstSize, bitmap->width, bitmap->height, pSrcData,
2090 SrcFormat, nSrcStep, bitmap->destLeft, bitmap->destTop, nullptr, bitsPerPixel);
2091 if (!ret)
2092 goto out;
2093 bitmap->bitmapDataStream = buffer;
2094 bitmap->bitmapLength = DstSize;
2095 bitmap->bitsPerPixel = bitsPerPixel;
2096 bitmap->cbScanWidth = bitmap->width * bytesPerPixel;
2097 bitmap->cbUncompressedSize = bitmap->width * bitmap->height * bytesPerPixel;
2098 }
2099 else
2100 {
2101 UINT32 dstSize = 0;
2102 buffer = encoder->grid[k];
2103 data = &pSrcData[(bitmap->destTop * nSrcStep) + (bitmap->destLeft * 4)];
2104
2105 buffer =
2106 freerdp_bitmap_compress_planar(encoder->planar, data, SrcFormat, bitmap->width,
2107 bitmap->height, nSrcStep, buffer, &dstSize);
2108 bitmap->bitmapDataStream = buffer;
2109 bitmap->bitmapLength = dstSize;
2110 bitmap->bitsPerPixel = 32;
2111 bitmap->cbScanWidth = bitmap->width * 4;
2112 bitmap->cbUncompressedSize = bitmap->width * bitmap->height * 4;
2113 }
2114
2115 bitmap->cbCompFirstRowSize = 0;
2116 bitmap->cbCompMainBodySize = bitmap->bitmapLength;
2117 totalBitmapSize += bitmap->bitmapLength;
2118 k++;
2119 }
2120 }
2121
2122 bitmapUpdate.number = k;
2123 updateSizeEstimate = totalBitmapSize + (k * bitmapUpdate.number) + 16;
2124
2125 if (updateSizeEstimate > maxUpdateSize)
2126 {
2127 UINT32 i = 0;
2128 UINT32 j = 0;
2129 UINT32 updateSize = 0;
2130 UINT32 newUpdateSize = 0;
2131 BITMAP_DATA* fragBitmapData = nullptr;
2132
2133 if (k > 0)
2134 fragBitmapData = (BITMAP_DATA*)calloc(k, sizeof(BITMAP_DATA));
2135
2136 if (!fragBitmapData)
2137 {
2138 WLog_ERR(TAG, "Failed to allocate memory for fragBitmapData");
2139 ret = FALSE;
2140 goto out;
2141 }
2142
2143 bitmapUpdate.rectangles = fragBitmapData;
2144 i = j = 0;
2145 updateSize = 1024;
2146
2147 while (i < k)
2148 {
2149 newUpdateSize = updateSize + (bitmapData[i].bitmapLength + 16);
2150
2151 if (newUpdateSize < maxUpdateSize)
2152 {
2153 CopyMemory(&fragBitmapData[j++], &bitmapData[i++], sizeof(BITMAP_DATA));
2154 updateSize = newUpdateSize;
2155 }
2156
2157 if ((newUpdateSize >= maxUpdateSize) || (i + 1) >= k)
2158 {
2159 bitmapUpdate.number = j;
2160 ret = BitmapUpdateProxy(client, &bitmapUpdate);
2161
2162 if (!ret)
2163 break;
2164
2165 updateSize = 1024;
2166 j = 0;
2167 }
2168 }
2169
2170 free(fragBitmapData);
2171 }
2172 else
2173 {
2174 ret = BitmapUpdateProxy(client, &bitmapUpdate);
2175 }
2176
2177out:
2178 free(bitmapData);
2179 return ret;
2180}
2181
2187WINPR_ATTR_NODISCARD
2188static BOOL shadow_client_send_surface_update(rdpShadowClient* client, SHADOW_GFX_STATUS* pStatus)
2189{
2190 BOOL ret = TRUE;
2191 INT64 nXSrc = 0;
2192 INT64 nYSrc = 0;
2193 INT64 nWidth = 0;
2194 INT64 nHeight = 0;
2195 rdpContext* context = (rdpContext*)client;
2196 rdpSettings* settings = nullptr;
2197 rdpShadowServer* server = nullptr;
2198 rdpShadowSurface* surface = nullptr;
2199 REGION16 invalidRegion;
2200 RECTANGLE_16 surfaceRect;
2201 const RECTANGLE_16* extents = nullptr;
2202 BYTE* pSrcData = nullptr;
2203 UINT32 nSrcStep = 0;
2204 UINT32 SrcFormat = 0;
2205 UINT32 numRects = 0;
2206 const RECTANGLE_16* rects = nullptr;
2207
2208 if (!context || !pStatus)
2209 return FALSE;
2210
2211 settings = context->settings;
2212 server = client->server;
2213
2214 if (!settings || !server)
2215 return FALSE;
2216
2217 surface = client->inLobby ? server->lobby : server->surface;
2218
2219 if (!surface)
2220 return FALSE;
2221
2222 {
2223 EnterCriticalSection(&(client->lock));
2224 region16_init(&invalidRegion);
2225
2226 const BOOL res = region16_copy(&invalidRegion, &(client->invalidRegion));
2227 region16_clear(&(client->invalidRegion));
2228 LeaveCriticalSection(&(client->lock));
2229 if (!res)
2230 goto out;
2231 }
2232
2233 EnterCriticalSection(&surface->lock);
2234 rects = region16_rects(&(surface->invalidRegion), &numRects);
2235
2236 for (UINT32 index = 0; index < numRects; index++)
2237 {
2238 if (!region16_union_rect(&invalidRegion, &invalidRegion, &rects[index]))
2239 goto out;
2240 }
2241
2242 surfaceRect.left = 0;
2243 surfaceRect.top = 0;
2244 WINPR_ASSERT(surface->width <= UINT16_MAX);
2245 WINPR_ASSERT(surface->height <= UINT16_MAX);
2246 surfaceRect.right = (UINT16)surface->width;
2247 surfaceRect.bottom = (UINT16)surface->height;
2248 if (!region16_intersect_rect(&invalidRegion, &invalidRegion, &surfaceRect))
2249 goto out;
2250
2251 if (server->shareSubRect)
2252 {
2253 if (!region16_intersect_rect(&invalidRegion, &invalidRegion, &(server->subRect)))
2254 goto out;
2255 }
2256
2257 if (region16_is_empty(&invalidRegion))
2258 {
2259 /* No image region need to be updated. Success */
2260 goto out;
2261 }
2262
2263 extents = region16_extents(&invalidRegion);
2264 nXSrc = extents->left;
2265 nYSrc = extents->top;
2266 nWidth = extents->right - extents->left;
2267 nHeight = extents->bottom - extents->top;
2268 pSrcData = surface->data;
2269 nSrcStep = surface->scanline;
2270 SrcFormat = surface->format;
2271
2272 /* Move to new pSrcData / nXSrc / nYSrc according to sub rect */
2273 if (server->shareSubRect)
2274 {
2275 INT32 subX = 0;
2276 INT32 subY = 0;
2277 subX = server->subRect.left;
2278 subY = server->subRect.top;
2279 nXSrc -= subX;
2280 nYSrc -= subY;
2281 WINPR_ASSERT(nXSrc >= 0);
2282 WINPR_ASSERT(nXSrc <= UINT16_MAX);
2283 WINPR_ASSERT(nYSrc >= 0);
2284 WINPR_ASSERT(nYSrc <= UINT16_MAX);
2285 pSrcData = &pSrcData[((UINT16)subY * nSrcStep) + ((UINT16)subX * 4U)];
2286 }
2287
2288 // WLog_INFO(TAG, "shadow_client_send_surface_update: x: %" PRId64 " y: %" PRId64 " width: %"
2289 // PRId64 " height: %" PRId64 " right: %" PRId64 " bottom: %" PRId64, nXSrc, nYSrc, nWidth,
2290 // nHeight, nXSrc + nWidth, nYSrc + nHeight);
2291
2292 if (freerdp_settings_get_bool(settings, FreeRDP_SupportGraphicsPipeline))
2293 {
2294 if (pStatus->gfxOpened && client->areGfxCapsReady)
2295 {
2296 /* GFX/h264 always full screen encoded */
2297 nWidth = freerdp_settings_get_uint32(settings, FreeRDP_DesktopWidth);
2298 nHeight = freerdp_settings_get_uint32(settings, FreeRDP_DesktopHeight);
2299
2300 /* Create primary surface if have not */
2301 if (!pStatus->gfxSurfaceCreated)
2302 {
2303 /* Only init surface when we have h264 supported */
2304 if (!(ret = shadow_client_rdpgfx_reset_graphic(client)))
2305 goto out;
2306
2307 if (!(ret = shadow_client_rdpgfx_new_surface(client)))
2308 goto out;
2309
2310 pStatus->gfxSurfaceCreated = TRUE;
2311 }
2312
2313 WINPR_ASSERT(nWidth >= 0);
2314 WINPR_ASSERT(nWidth <= UINT16_MAX);
2315 WINPR_ASSERT(nHeight >= 0);
2316 WINPR_ASSERT(nHeight <= UINT16_MAX);
2317 ret = shadow_client_send_surface_gfx(client, pSrcData, nSrcStep, SrcFormat, 0, 0,
2318 (UINT16)nWidth, (UINT16)nHeight);
2319 }
2320 else
2321 {
2322 ret = TRUE;
2323 }
2324 }
2325 else if (is_surface_command_supported(settings))
2326 {
2327 WINPR_ASSERT(nXSrc >= 0);
2328 WINPR_ASSERT(nXSrc <= UINT16_MAX);
2329 WINPR_ASSERT(nYSrc >= 0);
2330 WINPR_ASSERT(nYSrc <= UINT16_MAX);
2331 WINPR_ASSERT(nWidth >= 0);
2332 WINPR_ASSERT(nWidth <= UINT16_MAX);
2333 WINPR_ASSERT(nHeight >= 0);
2334 WINPR_ASSERT(nHeight <= UINT16_MAX);
2335 ret = shadow_client_send_surface_bits(client, pSrcData, nSrcStep, (UINT16)nXSrc,
2336 (UINT16)nYSrc, (UINT16)nWidth, (UINT16)nHeight);
2337 }
2338 else
2339 {
2340 WINPR_ASSERT(nXSrc >= 0);
2341 WINPR_ASSERT(nXSrc <= UINT16_MAX);
2342 WINPR_ASSERT(nYSrc >= 0);
2343 WINPR_ASSERT(nYSrc <= UINT16_MAX);
2344 WINPR_ASSERT(nWidth >= 0);
2345 WINPR_ASSERT(nWidth <= UINT16_MAX);
2346 WINPR_ASSERT(nHeight >= 0);
2347 WINPR_ASSERT(nHeight <= UINT16_MAX);
2348 ret = shadow_client_send_bitmap_update(client, pSrcData, nSrcStep, (UINT16)nXSrc,
2349 (UINT16)nYSrc, (UINT16)nWidth, (UINT16)nHeight);
2350 }
2351
2352out:
2353 LeaveCriticalSection(&surface->lock);
2354 region16_uninit(&invalidRegion);
2355 return ret;
2356}
2357
2365WINPR_ATTR_NODISCARD
2366static BOOL shadow_client_send_resize(rdpShadowClient* client, SHADOW_GFX_STATUS* pStatus)
2367{
2368 rdpContext* context = (rdpContext*)client;
2369 rdpSettings* settings = nullptr;
2370 freerdp_peer* peer = nullptr;
2371
2372 if (!context || !pStatus)
2373 return FALSE;
2374
2375 peer = context->peer;
2376 settings = context->settings;
2377
2378 if (!peer || !settings)
2379 return FALSE;
2380
2386 client->activated = FALSE;
2387
2388 /* Close Gfx surfaces */
2389 if (pStatus->gfxSurfaceCreated)
2390 {
2391 if (!shadow_client_rdpgfx_release_surface(client))
2392 return FALSE;
2393
2394 pStatus->gfxSurfaceCreated = FALSE;
2395 }
2396
2397 /* Send Resize */
2398 if (!shadow_send_desktop_resize(client))
2399 return FALSE;
2400 shadow_reset_desktop_resize(client);
2401
2402 /* Clear my invalidRegion. shadow_client_activate refreshes fullscreen */
2403 EnterCriticalSection(&(client->lock));
2404 region16_clear(&(client->invalidRegion));
2405 LeaveCriticalSection(&(client->lock));
2406 return TRUE;
2407}
2408
2415WINPR_ATTR_NODISCARD
2416static BOOL shadow_client_surface_update(rdpShadowClient* client, REGION16* region)
2417{
2418 UINT32 numRects = 0;
2419 const RECTANGLE_16* rects = nullptr;
2420 rects = region16_rects(region, &numRects);
2421 shadow_client_mark_invalid(client, numRects, rects);
2422 return TRUE;
2423}
2424
2431WINPR_ATTR_NODISCARD
2432static inline BOOL shadow_client_no_surface_update(rdpShadowClient* client,
2433 SHADOW_GFX_STATUS* pStatus)
2434{
2435 rdpShadowServer* server = nullptr;
2436 rdpShadowSurface* surface = nullptr;
2437 WINPR_UNUSED(pStatus);
2438 WINPR_ASSERT(client);
2439 server = client->server;
2440 WINPR_ASSERT(server);
2441 surface = client->inLobby ? server->lobby : server->surface;
2442 EnterCriticalSection(&surface->lock);
2443 const BOOL rc = shadow_client_surface_update(client, &(surface->invalidRegion));
2444 LeaveCriticalSection(&surface->lock);
2445 return rc;
2446}
2447
2448WINPR_ATTR_NODISCARD
2449static int shadow_client_subsystem_process_message(rdpShadowClient* client, wMessage* message)
2450{
2451 rdpContext* context = (rdpContext*)client;
2452
2453 WINPR_ASSERT(message);
2454 WINPR_ASSERT(context);
2455
2456 rdpUpdate* update = context->update;
2457 WINPR_ASSERT(update);
2458
2459 /* FIXME: the pointer updates appear to be broken when used with bulk compression and mstsc */
2460
2461 switch (message->id)
2462 {
2463 case SHADOW_MSG_OUT_POINTER_POSITION_UPDATE_ID:
2464 {
2465 POINTER_POSITION_UPDATE pointerPosition;
2467 (const SHADOW_MSG_OUT_POINTER_POSITION_UPDATE*)message->wParam;
2468 pointerPosition.xPos = msg->xPos;
2469 pointerPosition.yPos = msg->yPos;
2470
2471 WINPR_ASSERT(client->server);
2472 if (client->server->shareSubRect)
2473 {
2474 pointerPosition.xPos -= client->server->subRect.left;
2475 pointerPosition.yPos -= client->server->subRect.top;
2476 }
2477
2478 if (client->activated)
2479 {
2480 if ((msg->xPos != client->pointerX) || (msg->yPos != client->pointerY))
2481 {
2482 WINPR_ASSERT(update->pointer);
2483 if (!IFCALLRESULT(TRUE, update->pointer->PointerPosition, context,
2484 &pointerPosition))
2485 return -1;
2486 client->pointerX = msg->xPos;
2487 client->pointerY = msg->yPos;
2488 }
2489 }
2490
2491 break;
2492 }
2493
2494 case SHADOW_MSG_OUT_POINTER_ALPHA_UPDATE_ID:
2495 {
2496 POINTER_NEW_UPDATE pointerNew = WINPR_C_ARRAY_INIT;
2497 POINTER_COLOR_UPDATE* pointerColor = WINPR_C_ARRAY_INIT;
2498 POINTER_CACHED_UPDATE pointerCached = WINPR_C_ARRAY_INIT;
2500 (const SHADOW_MSG_OUT_POINTER_ALPHA_UPDATE*)message->wParam;
2501
2502 WINPR_ASSERT(msg);
2503 pointerNew.xorBpp = 24;
2504 pointerColor = &(pointerNew.colorPtrAttr);
2505 pointerColor->cacheIndex = 0;
2506 pointerColor->hotSpotX = WINPR_ASSERTING_INT_CAST(UINT16, msg->xHot);
2507 pointerColor->hotSpotY = WINPR_ASSERTING_INT_CAST(UINT16, msg->yHot);
2508 pointerColor->width = WINPR_ASSERTING_INT_CAST(UINT16, msg->width);
2509 pointerColor->height = WINPR_ASSERTING_INT_CAST(UINT16, msg->height);
2510 pointerColor->lengthAndMask = WINPR_ASSERTING_INT_CAST(UINT16, msg->lengthAndMask);
2511 pointerColor->lengthXorMask = WINPR_ASSERTING_INT_CAST(UINT16, msg->lengthXorMask);
2512 pointerColor->xorMaskData = msg->xorMaskData;
2513 pointerColor->andMaskData = msg->andMaskData;
2514 pointerCached.cacheIndex = pointerColor->cacheIndex;
2515
2516 if (client->activated)
2517 {
2518 if (client->server->ShowMouseCursor)
2519 {
2520 if (!IFCALLRESULT(TRUE, update->pointer->PointerNew, context, &pointerNew))
2521 return -1;
2522 if (!IFCALLRESULT(TRUE, update->pointer->PointerCached, context,
2523 &pointerCached))
2524 return -1;
2525 }
2526 else
2527 {
2528 POINTER_SYSTEM_UPDATE pointer_system = WINPR_C_ARRAY_INIT;
2529 pointer_system.type = SYSPTR_DEFAULT;
2530 if (!IFCALLRESULT(TRUE, update->pointer->PointerSystem, context,
2531 &pointer_system))
2532 return -1;
2533 }
2534 }
2535
2536 break;
2537 }
2538
2539 case SHADOW_MSG_OUT_AUDIO_OUT_SAMPLES_ID:
2540 {
2542 (const SHADOW_MSG_OUT_AUDIO_OUT_SAMPLES*)message->wParam;
2543
2544 WINPR_ASSERT(msg);
2545
2546 if (client->activated && client->rdpsnd && client->rdpsnd->Activated)
2547 {
2548 client->rdpsnd->src_format = msg->audio_format;
2549
2550 const UINT error =
2551 IFCALLRESULT(CHANNEL_RC_OK, client->rdpsnd->SendSamples, client->rdpsnd,
2552 msg->buf, msg->nFrames, msg->wTimestamp);
2553 if (CHANNEL_RC_OK != error)
2554 return -1;
2555 }
2556
2557 break;
2558 }
2559
2560 case SHADOW_MSG_OUT_AUDIO_OUT_VOLUME_ID:
2561 {
2563 (const SHADOW_MSG_OUT_AUDIO_OUT_VOLUME*)message->wParam;
2564
2565 if (client->activated && client->rdpsnd && client->rdpsnd->Activated)
2566 {
2567 const UINT error = IFCALLRESULT(CHANNEL_RC_OK, client->rdpsnd->SetVolume,
2568 client->rdpsnd, msg->left, msg->right);
2569 if (CHANNEL_RC_OK != error)
2570 return -1;
2571 }
2572
2573 break;
2574 }
2575
2576 default:
2577 WLog_ERR(TAG, "Unknown message id: %" PRIu32 "", message->id);
2578 break;
2579 }
2580
2581 shadow_client_free_queued_message(message);
2582 return 1;
2583}
2584
2585WINPR_ATTR_NODISCARD
2586static DWORD WINAPI shadow_client_thread(LPVOID arg)
2587{
2588 rdpShadowClient* client = (rdpShadowClient*)arg;
2589 BOOL rc = FALSE;
2590 DWORD status = 0;
2591 wMessage message = WINPR_C_ARRAY_INIT;
2592 wMessage pointerPositionMsg = WINPR_C_ARRAY_INIT;
2593 wMessage pointerAlphaMsg = WINPR_C_ARRAY_INIT;
2594 wMessage audioVolumeMsg = WINPR_C_ARRAY_INIT;
2595 HANDLE ChannelEvent = nullptr;
2596 void* UpdateSubscriber = nullptr;
2597 HANDLE UpdateEvent = nullptr;
2598 freerdp_peer* peer = nullptr;
2599 rdpContext* context = nullptr;
2600 rdpSettings* settings = nullptr;
2601 rdpShadowServer* server = nullptr;
2602 rdpShadowSubsystem* subsystem = nullptr;
2603 wMessageQueue* MsgQueue = nullptr;
2604 /* This should only be visited in client thread */
2605 SHADOW_GFX_STATUS gfxstatus = WINPR_C_ARRAY_INIT;
2606 rdpUpdate* update = nullptr;
2607
2608 WINPR_ASSERT(client);
2609
2610 MsgQueue = client->MsgQueue;
2611 WINPR_ASSERT(MsgQueue);
2612
2613 server = client->server;
2614 WINPR_ASSERT(server);
2615 subsystem = server->subsystem;
2616 context = (rdpContext*)client;
2617 peer = context->peer;
2618 WINPR_ASSERT(peer);
2619 WINPR_ASSERT(peer->context);
2620
2621 settings = peer->context->settings;
2622 WINPR_ASSERT(settings);
2623
2624 peer->Capabilities = shadow_client_capabilities;
2625 peer->PostConnect = shadow_client_post_connect;
2626 peer->Activate = shadow_client_activate;
2627 peer->Logon = shadow_client_logon;
2628 shadow_input_register_callbacks(peer->context->input);
2629
2630 rc = peer->Initialize(peer);
2631 if (!rc)
2632 goto out;
2633
2634 update = peer->context->update;
2635 WINPR_ASSERT(update);
2636
2637 update->RefreshRect = shadow_client_refresh_rect;
2638 update->SuppressOutput = shadow_client_suppress_output;
2639 update->SurfaceFrameAcknowledge = shadow_client_surface_frame_acknowledge;
2640
2641 if ((!client->vcm) || (!subsystem->updateEvent))
2642 goto out;
2643
2644 UpdateSubscriber = shadow_multiclient_get_subscriber(subsystem->updateEvent);
2645
2646 if (!UpdateSubscriber)
2647 goto out;
2648
2649 UpdateEvent = shadow_multiclient_getevent(UpdateSubscriber);
2650 WINPR_ASSERT(UpdateEvent);
2651
2652 ChannelEvent = WTSVirtualChannelManagerGetEventHandle(client->vcm);
2653 WINPR_ASSERT(ChannelEvent);
2654
2655 rc = freerdp_settings_set_bool(settings, FreeRDP_UnicodeInput, TRUE);
2656 WINPR_ASSERT(rc);
2657 rc = freerdp_settings_set_bool(settings, FreeRDP_HasHorizontalWheel, TRUE);
2658 WINPR_ASSERT(rc);
2659 rc = freerdp_settings_set_bool(settings, FreeRDP_HasExtendedMouseEvent, TRUE);
2660 WINPR_ASSERT(rc);
2661 rc = freerdp_settings_set_bool(settings, FreeRDP_SupportMonitorLayoutPdu, TRUE);
2662 WINPR_ASSERT(rc);
2663 while (1)
2664 {
2665 HANDLE events[MAXIMUM_WAIT_OBJECTS] = WINPR_C_ARRAY_INIT;
2666 DWORD nCount = 0;
2667 events[nCount++] = UpdateEvent;
2668 {
2669 DWORD tmp = peer->GetEventHandles(peer, &events[nCount], 64 - nCount);
2670
2671 if (tmp == 0)
2672 {
2673 WLog_ERR(TAG, "Failed to get FreeRDP transport event handles");
2674 goto fail;
2675 }
2676
2677 nCount += tmp;
2678 }
2679 events[nCount++] = ChannelEvent;
2680 events[nCount++] = MessageQueue_Event(MsgQueue);
2681
2682#if defined(CHANNEL_RDPGFX_SERVER)
2683 HANDLE gfxevent = rdpgfx_server_get_event_handle(client->rdpgfx);
2684
2685 if (gfxevent)
2686 events[nCount++] = gfxevent;
2687#endif
2688
2689 status = WaitForMultipleObjects(nCount, events, FALSE, INFINITE);
2690
2691 if (status == WAIT_FAILED)
2692 goto fail;
2693
2694 if (WaitForSingleObject(UpdateEvent, 0) == WAIT_OBJECT_0)
2695 {
2696 /* The UpdateEvent means to start sending current frame. It is
2697 * triggered from subsystem implementation and it should ensure
2698 * that the screen and primary surface meta data (width, height,
2699 * scanline, invalid region, etc) is not changed until it is reset
2700 * (at shadow_multiclient_consume). As best practice, subsystem
2701 * implementation should invoke shadow_subsystem_frame_update which
2702 * triggers the event and then wait for completion */
2703 if (client->activated && !client->suppressOutput)
2704 {
2705 /* Send screen update or resize to this client */
2706
2707 /* Check resize */
2708 if (shadow_client_recalc_desktop_size(client))
2709 {
2710 /* Screen size changed, do resize */
2711 if (!shadow_client_send_resize(client, &gfxstatus))
2712 {
2713 WLog_ERR(TAG, "Failed to send resize message");
2714 break;
2715 }
2716 }
2717 else
2718 {
2719 /* Send frame */
2720 if (!shadow_client_send_surface_update(client, &gfxstatus))
2721 {
2722 WLog_ERR(TAG, "Failed to send surface update");
2723 break;
2724 }
2725 }
2726 }
2727 else
2728 {
2729 /* Our client don't receive graphic updates. Just save the invalid region */
2730 if (!shadow_client_no_surface_update(client, &gfxstatus))
2731 {
2732 WLog_ERR(TAG, "Failed to handle surface update");
2733 break;
2734 }
2735 }
2736
2737 /*
2738 * The return value of shadow_multiclient_consume is whether or not
2739 * the subscriber really consumes the event. It's not cared currently.
2740 */
2741 (void)shadow_multiclient_consume(UpdateSubscriber);
2742 }
2743
2744 WINPR_ASSERT(peer->CheckFileDescriptor);
2745 if (!peer->CheckFileDescriptor(peer))
2746 {
2747 WLog_ERR(TAG, "Failed to check FreeRDP file descriptor");
2748 goto fail;
2749 }
2750
2751 if (client->activated &&
2752 WTSVirtualChannelManagerIsChannelJoined(client->vcm, DRDYNVC_SVC_CHANNEL_NAME))
2753 {
2754 switch (WTSVirtualChannelManagerGetDrdynvcState(client->vcm))
2755 {
2756 /* Dynamic channel status may have been changed after processing */
2757 case DRDYNVC_STATE_NONE:
2758
2759 /* Call this routine to Initialize drdynvc channel */
2760 if (!WTSVirtualChannelManagerCheckFileDescriptor(client->vcm))
2761 {
2762 WLog_ERR(TAG, "Failed to initialize drdynvc channel");
2763 goto fail;
2764 }
2765
2766 break;
2767
2768 case DRDYNVC_STATE_READY:
2769#if defined(CHANNEL_AUDIN_SERVER)
2770 if (client->audin && !IFCALLRESULT(TRUE, client->audin->IsOpen, client->audin))
2771 {
2772 if (!IFCALLRESULT(FALSE, client->audin->Open, client->audin))
2773 {
2774 WLog_ERR(TAG, "Failed to initialize audin channel");
2775 goto fail;
2776 }
2777 }
2778#endif
2779
2780 /* Init RDPGFX dynamic channel */
2781 if (freerdp_settings_get_bool(settings, FreeRDP_SupportGraphicsPipeline) &&
2782 client->rdpgfx && !gfxstatus.gfxOpened)
2783 {
2784 client->rdpgfx->FrameAcknowledge = shadow_client_rdpgfx_frame_acknowledge;
2785 client->rdpgfx->CapsAdvertise = shadow_client_rdpgfx_caps_advertise;
2786
2787 if (!client->rdpgfx->Open(client->rdpgfx))
2788 {
2789 WLog_WARN(TAG, "Failed to open GraphicsPipeline");
2790 if (!freerdp_settings_set_bool(settings,
2791 FreeRDP_SupportGraphicsPipeline, FALSE))
2792 goto fail;
2793 }
2794 else
2795 {
2796 gfxstatus.gfxOpened = TRUE;
2797 WLog_INFO(TAG, "Gfx Pipeline Opened");
2798 }
2799 }
2800
2801 break;
2802
2803 default:
2804 break;
2805 }
2806 }
2807
2808 if (WaitForSingleObject(ChannelEvent, 0) == WAIT_OBJECT_0)
2809 {
2810 if (!WTSVirtualChannelManagerCheckFileDescriptor(client->vcm))
2811 {
2812 WLog_ERR(TAG, "WTSVirtualChannelManagerCheckFileDescriptor failure");
2813 goto fail;
2814 }
2815 }
2816
2817#if defined(CHANNEL_RDPGFX_SERVER)
2818 if (gfxevent)
2819 {
2820 if (WaitForSingleObject(gfxevent, 0) == WAIT_OBJECT_0)
2821 {
2822 const UINT error = rdpgfx_server_handle_messages(client->rdpgfx);
2823 if (error != CHANNEL_RC_OK)
2824 goto fail;
2825 }
2826 }
2827#endif
2828
2829 if (WaitForSingleObject(MessageQueue_Event(MsgQueue), 0) == WAIT_OBJECT_0)
2830 {
2831 /* Drain messages. Pointer update could be accumulated. */
2832 pointerPositionMsg.id = 0;
2833 pointerPositionMsg.Free = nullptr;
2834 pointerAlphaMsg.id = 0;
2835 pointerAlphaMsg.Free = nullptr;
2836 audioVolumeMsg.id = 0;
2837 audioVolumeMsg.Free = nullptr;
2838
2839 while (MessageQueue_Peek(MsgQueue, &message, TRUE))
2840 {
2841 if (message.id == WMQ_QUIT)
2842 {
2843 break;
2844 }
2845
2846 switch (message.id)
2847 {
2848 case SHADOW_MSG_OUT_POINTER_POSITION_UPDATE_ID:
2849 /* Abandon previous message */
2850 shadow_client_free_queued_message(&pointerPositionMsg);
2851 pointerPositionMsg = message;
2852 break;
2853
2854 case SHADOW_MSG_OUT_POINTER_ALPHA_UPDATE_ID:
2855 /* Abandon previous message */
2856 shadow_client_free_queued_message(&pointerAlphaMsg);
2857 pointerAlphaMsg = message;
2858 break;
2859
2860 case SHADOW_MSG_OUT_AUDIO_OUT_VOLUME_ID:
2861 /* Abandon previous message */
2862 shadow_client_free_queued_message(&audioVolumeMsg);
2863 audioVolumeMsg = message;
2864 break;
2865
2866 default:
2867 if (!shadow_client_subsystem_process_message(client, &message))
2868 goto fail;
2869 break;
2870 }
2871 }
2872
2873 if (message.id == WMQ_QUIT)
2874 {
2875 /* Release stored message */
2876 shadow_client_free_queued_message(&pointerPositionMsg);
2877 shadow_client_free_queued_message(&pointerAlphaMsg);
2878 shadow_client_free_queued_message(&audioVolumeMsg);
2879 goto fail;
2880 }
2881 else
2882 {
2883 /* Process accumulated messages if needed */
2884 if (pointerPositionMsg.id)
2885 {
2886 if (!shadow_client_subsystem_process_message(client, &pointerPositionMsg))
2887 goto fail;
2888 }
2889
2890 if (pointerAlphaMsg.id)
2891 {
2892 if (!shadow_client_subsystem_process_message(client, &pointerAlphaMsg))
2893 goto fail;
2894 }
2895
2896 if (audioVolumeMsg.id)
2897 {
2898 if (!shadow_client_subsystem_process_message(client, &audioVolumeMsg))
2899 goto fail;
2900 }
2901 }
2902 }
2903 }
2904
2905fail:
2906
2907 /* Free channels early because we establish channels in post connect */
2908#if defined(CHANNEL_AUDIN_SERVER)
2909 if (client->audin && !IFCALLRESULT(TRUE, client->audin->IsOpen, client->audin))
2910 {
2911 if (!IFCALLRESULT(FALSE, client->audin->Close, client->audin))
2912 {
2913 WLog_WARN(TAG, "AUDIN shutdown failure!");
2914 }
2915 }
2916#endif
2917
2918 if (gfxstatus.gfxOpened)
2919 {
2920 if (gfxstatus.gfxSurfaceCreated)
2921 {
2922 if (!shadow_client_rdpgfx_release_surface(client))
2923 WLog_WARN(TAG, "GFX release surface failure!");
2924 }
2925
2926 WINPR_ASSERT(client->rdpgfx);
2927 WINPR_ASSERT(client->rdpgfx->Close);
2928 rc = client->rdpgfx->Close(client->rdpgfx);
2929 WINPR_ASSERT(rc);
2930 }
2931
2932 shadow_client_channels_free(client);
2933
2934 if (UpdateSubscriber)
2935 {
2936 shadow_multiclient_release_subscriber(UpdateSubscriber);
2937 UpdateSubscriber = nullptr;
2938 }
2939
2940 if (peer->connected && subsystem->ClientDisconnect)
2941 {
2942 subsystem->ClientDisconnect(subsystem, client);
2943 }
2944
2945out:
2946 WINPR_ASSERT(peer->Disconnect);
2947 peer->Disconnect(peer);
2948 freerdp_peer_context_free(peer);
2949 freerdp_peer_free(peer);
2950 ExitThread(0);
2951 return 0;
2952}
2953
2954BOOL shadow_client_accepted(freerdp_listener* listener, freerdp_peer* peer)
2955{
2956 rdpShadowClient* client = nullptr;
2957 rdpShadowServer* server = nullptr;
2958
2959 if (!listener || !peer)
2960 return FALSE;
2961
2962 server = (rdpShadowServer*)listener->info;
2963 WINPR_ASSERT(server);
2964
2965 peer->ContextExtra = (void*)server;
2966 peer->ContextSize = sizeof(rdpShadowClient);
2967 peer->ContextNew = shadow_client_context_new;
2968 peer->ContextFree = shadow_client_context_free;
2969
2970 if (!freerdp_peer_context_new_ex(peer, server->settings))
2971 return FALSE;
2972
2973 client = (rdpShadowClient*)peer->context;
2974 WINPR_ASSERT(client);
2975
2976 if (!(client->thread = CreateThread(nullptr, 0, shadow_client_thread, client, 0, nullptr)))
2977 {
2978 freerdp_peer_context_free(peer);
2979 return FALSE;
2980 }
2981 else
2982 {
2983 /* Close the thread handle to make it detached. */
2984 (void)CloseHandle(client->thread);
2985 client->thread = nullptr;
2986 }
2987
2988 return TRUE;
2989}
2990
2991static void shadow_msg_out_addref(wMessage* message)
2992{
2993 SHADOW_MSG_OUT* msg = nullptr;
2994
2995 WINPR_ASSERT(message);
2996 msg = (SHADOW_MSG_OUT*)message->wParam;
2997 WINPR_ASSERT(msg);
2998
2999 InterlockedIncrement(&(msg->refCount));
3000}
3001
3002static void shadow_msg_out_release(wMessage* message)
3003{
3004 SHADOW_MSG_OUT* msg = nullptr;
3005
3006 WINPR_ASSERT(message);
3007 msg = (SHADOW_MSG_OUT*)message->wParam;
3008 WINPR_ASSERT(msg);
3009
3010 if (InterlockedDecrement(&(msg->refCount)) <= 0)
3011 {
3012 IFCALL(msg->Free, message->id, msg);
3013 }
3014}
3015
3016WINPR_ATTR_NODISCARD
3017static BOOL shadow_client_dispatch_msg(rdpShadowClient* client, wMessage* message)
3018{
3019 if (!client || !message)
3020 return FALSE;
3021
3022 /* Add reference when it is posted */
3023 shadow_msg_out_addref(message);
3024
3025 WINPR_ASSERT(client->MsgQueue);
3026 if (MessageQueue_Dispatch(client->MsgQueue, message))
3027 return TRUE;
3028 else
3029 {
3030 /* Release the reference since post failed */
3031 shadow_msg_out_release(message);
3032 return FALSE;
3033 }
3034}
3035
3036BOOL shadow_client_post_msg(rdpShadowClient* client, void* context, UINT32 type,
3037 SHADOW_MSG_OUT* msg, void* lParam)
3038{
3039 wMessage message = WINPR_C_ARRAY_INIT;
3040 message.context = context;
3041 message.id = type;
3042 message.wParam = (void*)msg;
3043 message.lParam = lParam;
3044 message.Free = shadow_msg_out_release;
3045 return shadow_client_dispatch_msg(client, &message);
3046}
3047
3048int shadow_client_boardcast_msg(rdpShadowServer* server, void* context, UINT32 type,
3049 SHADOW_MSG_OUT* msg, void* lParam)
3050{
3051 wMessage message = WINPR_C_ARRAY_INIT;
3052 int count = 0;
3053
3054 WINPR_ASSERT(server);
3055 WINPR_ASSERT(msg);
3056
3057 message.context = context;
3058 message.id = type;
3059 message.wParam = (void*)msg;
3060 message.lParam = lParam;
3061 message.Free = shadow_msg_out_release;
3062 /* First add reference as we reference it in this function.
3063 * Therefore it would not be free'ed during post. */
3064 shadow_msg_out_addref(&message);
3065
3066 WINPR_ASSERT(server->clients);
3067 ArrayList_Lock(server->clients);
3068
3069 for (size_t index = 0; index < ArrayList_Count(server->clients); index++)
3070 {
3071 rdpShadowClient* client = (rdpShadowClient*)ArrayList_GetItem(server->clients, index);
3072
3073 if (shadow_client_dispatch_msg(client, &message))
3074 {
3075 count++;
3076 }
3077 }
3078
3079 ArrayList_Unlock(server->clients);
3080 /* Release the reference for this function */
3081 shadow_msg_out_release(&message);
3082 return count;
3083}
3084
3085int shadow_client_boardcast_quit(rdpShadowServer* server, int nExitCode)
3086{
3087 wMessageQueue* queue = nullptr;
3088 int count = 0;
3089
3090 WINPR_ASSERT(server);
3091 WINPR_ASSERT(server->clients);
3092
3093 ArrayList_Lock(server->clients);
3094
3095 for (size_t index = 0; index < ArrayList_Count(server->clients); index++)
3096 {
3097 queue = ((rdpShadowClient*)ArrayList_GetItem(server->clients, index))->MsgQueue;
3098
3099 if (MessageQueue_PostQuit(queue, nExitCode))
3100 {
3101 count++;
3102 }
3103 }
3104
3105 ArrayList_Unlock(server->clients);
3106 return count;
3107}
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.
Definition rfx.h:44
This struct contains function pointer to initialize/free objects.
Definition collections.h:52