FreeRDP
Loading...
Searching...
No Matches
client/drdynvc_main.c
1
22#include <freerdp/config.h>
23
24#include <winpr/crt.h>
25#include <winpr/cast.h>
26#include <winpr/stream.h>
27#include <winpr/interlocked.h>
28
29#include <freerdp/freerdp.h>
30#include <freerdp/channels/drdynvc.h>
31#include <freerdp/utils/drdynvc.h>
32#include <freerdp/codec/zgfx.h>
33
34#include "drdynvc_main.h"
35
36#define TAG CHANNELS_TAG("drdynvc.client")
37
38static const char* channel_state2str(DVC_CHANNEL_STATE state)
39{
40 switch (state)
41 {
42 case DVC_CHANNEL_INIT:
43 return "DVC_CHANNEL_INIT";
44 case DVC_CHANNEL_RUNNING:
45 return "DVC_CHANNEL_RUNNING";
46 case DVC_CHANNEL_CLOSED:
47 return "DVC_CHANNEL_CLOSED";
48 default:
49 return "DVC_CHANNEL_UNKNOWN";
50 }
51}
52
53static void dvcman_channel_free(DVCMAN_CHANNEL* channel);
54static UINT dvcman_channel_close(DVCMAN_CHANNEL* channel, BOOL perRequest, BOOL fromHashTableFn);
55static void dvcman_free(drdynvcPlugin* drdynvc, IWTSVirtualChannelManager* pChannelMgr);
56static UINT drdynvc_write_data(drdynvcPlugin* drdynvc, UINT32 ChannelId, const BYTE* data,
57 UINT32 dataSize, BOOL* close);
58static UINT drdynvc_send(drdynvcPlugin* drdynvc, wStream* s);
59
60static void dvcman_wtslistener_free(DVCMAN_LISTENER* listener)
61{
62 if (listener)
63 free(listener->channel_name);
64 free(listener);
65}
66
72static UINT dvcman_get_configuration(IWTSListener* pListener, void** ppPropertyBag)
73{
74 WINPR_ASSERT(ppPropertyBag);
75 WINPR_UNUSED(pListener);
76 *ppPropertyBag = NULL;
77 return ERROR_INTERNAL_ERROR;
78}
79
85static UINT dvcman_create_listener(IWTSVirtualChannelManager* pChannelMgr,
86 const char* pszChannelName, ULONG ulFlags,
87 IWTSListenerCallback* pListenerCallback,
88 IWTSListener** ppListener)
89{
90 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
91 DVCMAN_LISTENER* listener = NULL;
92
93 WINPR_ASSERT(dvcman);
94 WLog_DBG(TAG, "create_listener: %" PRIuz ".%s.", HashTable_Count(dvcman->listeners) + 1,
95 pszChannelName);
96 listener = (DVCMAN_LISTENER*)calloc(1, sizeof(DVCMAN_LISTENER));
97
98 if (!listener)
99 {
100 WLog_ERR(TAG, "calloc failed!");
101 return CHANNEL_RC_NO_MEMORY;
102 }
103
104 listener->iface.GetConfiguration = dvcman_get_configuration;
105 listener->iface.pInterface = NULL;
106 listener->dvcman = dvcman;
107 listener->channel_name = _strdup(pszChannelName);
108
109 if (!listener->channel_name)
110 {
111 WLog_ERR(TAG, "_strdup failed!");
112 dvcman_wtslistener_free(listener);
113 return CHANNEL_RC_NO_MEMORY;
114 }
115
116 listener->flags = ulFlags;
117 listener->listener_callback = pListenerCallback;
118
119 if (ppListener)
120 *ppListener = (IWTSListener*)listener;
121
122 if (!HashTable_Insert(dvcman->listeners, listener->channel_name, listener))
123 {
124 dvcman_wtslistener_free(listener);
125 return ERROR_INTERNAL_ERROR;
126 }
127
128 // NOLINTNEXTLINE(clang-analyzer-unix.Malloc): HashTable_Insert takes ownership of listener
129 return CHANNEL_RC_OK;
130}
131
132static UINT dvcman_destroy_listener(IWTSVirtualChannelManager* pChannelMgr, IWTSListener* pListener)
133{
134 DVCMAN_LISTENER* listener = (DVCMAN_LISTENER*)pListener;
135
136 WINPR_UNUSED(pChannelMgr);
137
138 if (listener)
139 {
140 DVCMAN* dvcman = listener->dvcman;
141 if (dvcman)
142 HashTable_Remove(dvcman->listeners, listener->channel_name);
143 }
144
145 return CHANNEL_RC_OK;
146}
147
153static UINT dvcman_register_plugin(IDRDYNVC_ENTRY_POINTS* pEntryPoints, const char* name,
154 IWTSPlugin* pPlugin)
155{
156 WINPR_ASSERT(pEntryPoints);
157 DVCMAN* dvcman = ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->dvcman;
158
159 WINPR_ASSERT(dvcman);
160 if (!ArrayList_Append(dvcman->plugin_names, name))
161 return ERROR_INTERNAL_ERROR;
162 if (!ArrayList_Append(dvcman->plugins, pPlugin))
163 return ERROR_INTERNAL_ERROR;
164
165 WLog_DBG(TAG, "register_plugin: num_plugins %" PRIuz, ArrayList_Count(dvcman->plugins));
166 return CHANNEL_RC_OK;
167}
168
169static IWTSPlugin* dvcman_get_plugin(IDRDYNVC_ENTRY_POINTS* pEntryPoints, const char* name)
170{
171 IWTSPlugin* plugin = NULL;
172 size_t nc = 0;
173 size_t pc = 0;
174 WINPR_ASSERT(pEntryPoints);
175 DVCMAN* dvcman = ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->dvcman;
176 if (!dvcman || !pEntryPoints || !name)
177 return NULL;
178
179 nc = ArrayList_Count(dvcman->plugin_names);
180 pc = ArrayList_Count(dvcman->plugins);
181 if (nc != pc)
182 return NULL;
183
184 ArrayList_Lock(dvcman->plugin_names);
185 ArrayList_Lock(dvcman->plugins);
186 for (size_t i = 0; i < pc; i++)
187 {
188 const char* cur = ArrayList_GetItem(dvcman->plugin_names, i);
189 if (strcmp(cur, name) == 0)
190 {
191 plugin = ArrayList_GetItem(dvcman->plugins, i);
192 break;
193 }
194 }
195 ArrayList_Unlock(dvcman->plugin_names);
196 ArrayList_Unlock(dvcman->plugins);
197 return plugin;
198}
199
200static const ADDIN_ARGV* dvcman_get_plugin_data(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
201{
202 WINPR_ASSERT(pEntryPoints);
203 return ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->args;
204}
205
206static rdpContext* dvcman_get_rdp_context(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
207{
208 DVCMAN_ENTRY_POINTS* entry = (DVCMAN_ENTRY_POINTS*)pEntryPoints;
209 WINPR_ASSERT(entry);
210 return entry->context;
211}
212
213static rdpSettings* dvcman_get_rdp_settings(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
214{
215 rdpContext* context = dvcman_get_rdp_context(pEntryPoints);
216 WINPR_ASSERT(context);
217
218 return context->settings;
219}
220
221static UINT32 dvcman_get_channel_id(IWTSVirtualChannel* channel)
222{
223 DVCMAN_CHANNEL* dvc = (DVCMAN_CHANNEL*)channel;
224 WINPR_ASSERT(dvc);
225 return dvc->channel_id;
226}
227
228static const char* dvcman_get_channel_name(IWTSVirtualChannel* channel)
229{
230 DVCMAN_CHANNEL* dvc = (DVCMAN_CHANNEL*)channel;
231 WINPR_ASSERT(dvc);
232 return dvc->channel_name;
233}
234
235static DVCMAN_CHANNEL* dvcman_get_channel_by_id(IWTSVirtualChannelManager* pChannelMgr,
236 UINT32 ChannelId, BOOL doRef)
237{
238 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
239 DVCMAN_CHANNEL* dvcChannel = NULL;
240
241 WINPR_ASSERT(dvcman);
242 HashTable_Lock(dvcman->channelsById);
243 dvcChannel = HashTable_GetItemValue(dvcman->channelsById, &ChannelId);
244 if (dvcChannel)
245 {
246 if (doRef)
247 InterlockedIncrement(&dvcChannel->refCounter);
248 }
249
250 HashTable_Unlock(dvcman->channelsById);
251 return dvcChannel;
252}
253
254static IWTSVirtualChannel* dvcman_find_channel_by_id(IWTSVirtualChannelManager* pChannelMgr,
255 UINT32 ChannelId)
256{
257 DVCMAN_CHANNEL* channel = dvcman_get_channel_by_id(pChannelMgr, ChannelId, FALSE);
258 if (!channel)
259 return NULL;
260
261 return &channel->iface;
262}
263
264static void dvcman_plugin_terminate(void* plugin)
265{
266 IWTSPlugin* pPlugin = plugin;
267
268 WINPR_ASSERT(pPlugin);
269 UINT error = IFCALLRESULT(CHANNEL_RC_OK, pPlugin->Terminated, pPlugin);
270 if (error != CHANNEL_RC_OK)
271 WLog_ERR(TAG, "Terminated failed with error %" PRIu32 "!", error);
272}
273
274static void wts_listener_free(void* arg)
275{
276 DVCMAN_LISTENER* listener = (DVCMAN_LISTENER*)arg;
277 dvcman_wtslistener_free(listener);
278}
279
280static BOOL channelIdMatch(const void* k1, const void* k2)
281{
282 WINPR_ASSERT(k1);
283 WINPR_ASSERT(k2);
284 return *((const UINT32*)k1) == *((const UINT32*)k2);
285}
286
287static UINT32 channelIdHash(const void* id)
288{
289 WINPR_ASSERT(id);
290 return *((const UINT32*)id);
291}
292
293static void channelByIdCleanerFn(void* value)
294{
295 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*)value;
296 if (channel)
297 {
298 dvcman_channel_close(channel, FALSE, TRUE);
299 dvcman_channel_free(channel);
300 }
301}
302
303static IWTSVirtualChannelManager* dvcman_new(drdynvcPlugin* plugin)
304{
305 wObject* obj = NULL;
306 DVCMAN* dvcman = (DVCMAN*)calloc(1, sizeof(DVCMAN));
307
308 if (!dvcman)
309 return NULL;
310
311 dvcman->iface.CreateListener = dvcman_create_listener;
312 dvcman->iface.DestroyListener = dvcman_destroy_listener;
313 dvcman->iface.FindChannelById = dvcman_find_channel_by_id;
314 dvcman->iface.GetChannelId = dvcman_get_channel_id;
315 dvcman->iface.GetChannelName = dvcman_get_channel_name;
316 dvcman->drdynvc = plugin;
317 dvcman->channelsById = HashTable_New(TRUE);
318
319 if (!dvcman->channelsById)
320 goto fail;
321
322 HashTable_SetHashFunction(dvcman->channelsById, channelIdHash);
323 obj = HashTable_KeyObject(dvcman->channelsById);
324 WINPR_ASSERT(obj);
325 obj->fnObjectEquals = channelIdMatch;
326
327 obj = HashTable_ValueObject(dvcman->channelsById);
328 WINPR_ASSERT(obj);
329 obj->fnObjectFree = channelByIdCleanerFn;
330
331 dvcman->pool = StreamPool_New(TRUE, 10);
332 if (!dvcman->pool)
333 goto fail;
334
335 dvcman->listeners = HashTable_New(TRUE);
336 if (!dvcman->listeners)
337 goto fail;
338 HashTable_SetHashFunction(dvcman->listeners, HashTable_StringHash);
339
340 obj = HashTable_KeyObject(dvcman->listeners);
341 obj->fnObjectEquals = HashTable_StringCompare;
342
343 obj = HashTable_ValueObject(dvcman->listeners);
344 obj->fnObjectFree = wts_listener_free;
345
346 dvcman->plugin_names = ArrayList_New(TRUE);
347 if (!dvcman->plugin_names)
348 goto fail;
349 obj = ArrayList_Object(dvcman->plugin_names);
350 obj->fnObjectNew = winpr_ObjectStringClone;
351 obj->fnObjectFree = winpr_ObjectStringFree;
352
353 dvcman->plugins = ArrayList_New(TRUE);
354 if (!dvcman->plugins)
355 goto fail;
356 obj = ArrayList_Object(dvcman->plugins);
357 obj->fnObjectFree = dvcman_plugin_terminate;
358 return &dvcman->iface;
359fail:
360 dvcman_free(plugin, &dvcman->iface);
361 return NULL;
362}
363
369static UINT dvcman_load_addin(drdynvcPlugin* drdynvc, IWTSVirtualChannelManager* pChannelMgr,
370 const ADDIN_ARGV* args, rdpContext* context)
371{
372 WINPR_ASSERT(drdynvc);
373 WINPR_ASSERT(pChannelMgr);
374 WINPR_ASSERT(args);
375 WINPR_ASSERT(context);
376
377 WLog_Print(drdynvc->log, WLOG_INFO, "Loading Dynamic Virtual Channel %s", args->argv[0]);
378
379 PVIRTUALCHANNELENTRY pvce =
380 freerdp_load_channel_addin_entry(args->argv[0], NULL, NULL, FREERDP_ADDIN_CHANNEL_DYNAMIC);
381 PDVC_PLUGIN_ENTRY pDVCPluginEntry = WINPR_FUNC_PTR_CAST(pvce, PDVC_PLUGIN_ENTRY);
382
383 if (pDVCPluginEntry)
384 {
385 DVCMAN_ENTRY_POINTS entryPoints = { 0 };
386
387 entryPoints.iface.RegisterPlugin = dvcman_register_plugin;
388 entryPoints.iface.GetPlugin = dvcman_get_plugin;
389 entryPoints.iface.GetPluginData = dvcman_get_plugin_data;
390 entryPoints.iface.GetRdpSettings = dvcman_get_rdp_settings;
391 entryPoints.iface.GetRdpContext = dvcman_get_rdp_context;
392 entryPoints.dvcman = (DVCMAN*)pChannelMgr;
393 entryPoints.args = args;
394 entryPoints.context = context;
395 return pDVCPluginEntry(&entryPoints.iface);
396 }
397
398 return ERROR_INVALID_FUNCTION;
399}
400
401static void dvcman_channel_free(DVCMAN_CHANNEL* channel)
402{
403 if (!channel)
404 return;
405
406 if (channel->dvc_data)
407 Stream_Release(channel->dvc_data);
408
409 zgfx_context_free(channel->decompressor);
410 DeleteCriticalSection(&(channel->lock));
411 free(channel->channel_name);
412 free(channel);
413}
414
415static void dvcman_channel_unref(DVCMAN_CHANNEL* channel)
416{
417 WINPR_ASSERT(channel);
418 if (InterlockedDecrement(&channel->refCounter))
419 return;
420
421 DVCMAN* dvcman = channel->dvcman;
422 if (dvcman)
423 HashTable_Remove(dvcman->channelsById, &channel->channel_id);
424}
425
426static UINT dvcchannel_send_close(DVCMAN_CHANNEL* channel)
427{
428 WINPR_ASSERT(channel);
429 DVCMAN* dvcman = channel->dvcman;
430 drdynvcPlugin* drdynvc = dvcman->drdynvc;
431 wStream* s = StreamPool_Take(dvcman->pool, 5);
432
433 if (!s)
434 {
435 WLog_Print(drdynvc->log, WLOG_ERROR, "StreamPool_Take failed!");
436 return CHANNEL_RC_NO_MEMORY;
437 }
438
439 Stream_Write_UINT8(s, (CLOSE_REQUEST_PDU << 4) | 0x02);
440 Stream_Write_UINT32(s, channel->channel_id);
441 return drdynvc_send(drdynvc, s);
442}
443
444static void check_open_close_receive(DVCMAN_CHANNEL* channel)
445{
446 WINPR_ASSERT(channel);
447
448 IWTSVirtualChannelCallback* cb = channel->channel_callback;
449 const char* name = channel->channel_name;
450 const UINT32 id = channel->channel_id;
451
452 WINPR_ASSERT(cb);
453 if (!cb->OnOpen || !cb->OnClose || !cb->OnDataReceived)
454 WLog_VRB(TAG, "{%s:%" PRIu32 "} OnOpen=%p, OnClose=%p, OnDataReceived=%p", name, id,
455 WINPR_CXX_COMPAT_CAST(const void*, cb->OnOpen),
456 WINPR_CXX_COMPAT_CAST(const void*, cb->OnClose),
457 WINPR_CXX_COMPAT_CAST(const void*, cb->OnDataReceived));
458}
459
460static UINT dvcman_call_on_receive(DVCMAN_CHANNEL* channel, wStream* data)
461{
462 WINPR_ASSERT(channel);
463 WINPR_ASSERT(data);
464
465 IWTSVirtualChannelCallback* cb = channel->channel_callback;
466 WINPR_ASSERT(cb);
467
468 check_open_close_receive(channel);
469 WINPR_ASSERT(cb->OnDataReceived);
470 return cb->OnDataReceived(cb, data);
471}
472
473static UINT dvcman_channel_close(DVCMAN_CHANNEL* channel, BOOL perRequest, BOOL fromHashTableFn)
474{
475 UINT error = CHANNEL_RC_OK;
476 DrdynvcClientContext* context = NULL;
477
478 WINPR_ASSERT(channel);
479 switch (channel->state)
480 {
481 case DVC_CHANNEL_INIT:
482 break;
483 case DVC_CHANNEL_RUNNING:
484 if (channel->dvcman)
485 {
486 drdynvcPlugin* drdynvc = channel->dvcman->drdynvc;
487 WINPR_ASSERT(drdynvc);
488 context = drdynvc->context;
489 if (perRequest)
490 WLog_Print(drdynvc->log, WLOG_DEBUG, "sending close confirm for '%s'",
491 channel->channel_name);
492
493 error = dvcchannel_send_close(channel);
494 if (error != CHANNEL_RC_OK)
495 {
496 if (perRequest)
497 WLog_Print(drdynvc->log, WLOG_DEBUG,
498 "error when sending closeRequest for '%s'",
499 channel->channel_name);
500 else
501 WLog_Print(drdynvc->log, WLOG_DEBUG,
502 "error when sending close confirm for '%s'",
503 channel->channel_name);
504 }
505 WLog_Print(drdynvc->log, WLOG_DEBUG, "listener %s destroyed channel %" PRIu32 "",
506 channel->channel_name, channel->channel_id);
507 }
508
509 channel->state = DVC_CHANNEL_CLOSED;
510
511 {
512 check_open_close_receive(channel);
513
514 IWTSVirtualChannelCallback* cb = channel->channel_callback;
515 channel->channel_callback = NULL;
516 if (cb)
517 {
518 IFCALL(cb->OnClose, cb);
519 }
520 }
521
522 if (channel->dvcman && channel->dvcman->drdynvc)
523 {
524 if (context)
525 {
526 IFCALLRET(context->OnChannelDisconnected, error, context, channel->channel_name,
527 channel->pInterface);
528 }
529 }
530
531 if (!fromHashTableFn)
532 dvcman_channel_unref(channel);
533 break;
534 case DVC_CHANNEL_CLOSED:
535 break;
536 default:
537 break;
538 }
539
540 return error;
541}
542
543static DVCMAN_CHANNEL* dvcman_channel_new(WINPR_ATTR_UNUSED drdynvcPlugin* drdynvc,
544 IWTSVirtualChannelManager* pChannelMgr, UINT32 ChannelId,
545 const char* ChannelName)
546{
547 DVCMAN_CHANNEL* channel = NULL;
548
549 WINPR_ASSERT(drdynvc);
550 WINPR_ASSERT(pChannelMgr);
551 channel = (DVCMAN_CHANNEL*)calloc(1, sizeof(DVCMAN_CHANNEL));
552
553 if (!channel)
554 return NULL;
555
556 channel->dvcman = (DVCMAN*)pChannelMgr;
557 channel->channel_id = ChannelId;
558 channel->refCounter = 1;
559 channel->state = DVC_CHANNEL_INIT;
560 channel->channel_name = _strdup(ChannelName);
561 if (!channel->channel_name)
562 goto fail;
563
564 channel->decompressor = zgfx_context_new(FALSE);
565 if (!channel->decompressor)
566 goto fail;
567
568 if (!InitializeCriticalSectionEx(&(channel->lock), 0, 0))
569 goto fail;
570
571 return channel;
572fail:
573 dvcman_channel_free(channel);
574 return NULL;
575}
576
577static void dvcman_clear(drdynvcPlugin* drdynvc, IWTSVirtualChannelManager* pChannelMgr)
578{
579 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
580
581 WINPR_ASSERT(dvcman);
582 WINPR_UNUSED(drdynvc);
583
584 HashTable_Clear(dvcman->channelsById);
585 ArrayList_Clear(dvcman->plugins);
586 ArrayList_Clear(dvcman->plugin_names);
587 HashTable_Clear(dvcman->listeners);
588}
589static void dvcman_free(drdynvcPlugin* drdynvc, IWTSVirtualChannelManager* pChannelMgr)
590{
591 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
592
593 WINPR_ASSERT(dvcman);
594 WINPR_UNUSED(drdynvc);
595
596 HashTable_Free(dvcman->channelsById);
597 ArrayList_Free(dvcman->plugins);
598 ArrayList_Free(dvcman->plugin_names);
599 HashTable_Free(dvcman->listeners);
600
601 StreamPool_Free(dvcman->pool);
602 free(dvcman);
603}
604
610static UINT dvcman_init(drdynvcPlugin* drdynvc, IWTSVirtualChannelManager* pChannelMgr)
611{
612 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
613 UINT error = CHANNEL_RC_OK;
614
615 WINPR_ASSERT(dvcman);
616 ArrayList_Lock(dvcman->plugins);
617 for (size_t i = 0; i < ArrayList_Count(dvcman->plugins); i++)
618 {
619 IWTSPlugin* pPlugin = ArrayList_GetItem(dvcman->plugins, i);
620
621 error = IFCALLRESULT(CHANNEL_RC_OK, pPlugin->Initialize, pPlugin, pChannelMgr);
622 if (error != CHANNEL_RC_OK)
623 {
624 WLog_Print(drdynvc->log, WLOG_ERROR, "Initialize failed with error %" PRIu32 "!",
625 error);
626 goto fail;
627 }
628 }
629
630fail:
631 ArrayList_Unlock(dvcman->plugins);
632 return error;
633}
634
640static UINT dvcman_write_channel(IWTSVirtualChannel* pChannel, ULONG cbSize, const BYTE* pBuffer,
641 void* pReserved)
642{
643 BOOL close = FALSE;
644 UINT status = 0;
645 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*)pChannel;
646
647 WINPR_UNUSED(pReserved);
648 if (!channel || !channel->dvcman)
649 return CHANNEL_RC_BAD_CHANNEL;
650
651 EnterCriticalSection(&(channel->lock));
652 status =
653 drdynvc_write_data(channel->dvcman->drdynvc, channel->channel_id, pBuffer, cbSize, &close);
654 LeaveCriticalSection(&(channel->lock));
655 /* Close delayed, it removes the channel struct */
656 if (close)
657 dvcman_channel_close(channel, FALSE, FALSE);
658
659 return status;
660}
661
667static UINT dvcman_close_channel_iface(IWTSVirtualChannel* pChannel)
668{
669 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*)pChannel;
670
671 if (!channel)
672 return CHANNEL_RC_BAD_CHANNEL;
673
674 WLog_DBG(TAG, "close_channel_iface: id=%" PRIu32 "", channel->channel_id);
675 return dvcman_channel_close(channel, FALSE, FALSE);
676}
677
683static DVCMAN_CHANNEL* dvcman_create_channel(drdynvcPlugin* drdynvc,
684 IWTSVirtualChannelManager* pChannelMgr,
685 UINT32 ChannelId, const char* ChannelName, UINT* res)
686{
687 BOOL bAccept = 0;
688 DVCMAN_CHANNEL* channel = NULL;
689 DrdynvcClientContext* context = NULL;
690 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
691 DVCMAN_LISTENER* listener = NULL;
692 IWTSVirtualChannelCallback* pCallback = NULL;
693
694 WINPR_ASSERT(dvcman);
695 WINPR_ASSERT(res);
696
697 HashTable_Lock(dvcman->listeners);
698 listener = (DVCMAN_LISTENER*)HashTable_GetItemValue(dvcman->listeners, ChannelName);
699 if (!listener)
700 {
701 *res = ERROR_NOT_FOUND;
702 goto out;
703 }
704
705 channel = dvcman_get_channel_by_id(pChannelMgr, ChannelId, FALSE);
706 if (channel)
707 {
708 switch (channel->state)
709 {
710 case DVC_CHANNEL_RUNNING:
711 WLog_Print(drdynvc->log, WLOG_ERROR,
712 "Protocol error: Duplicated ChannelId %" PRIu32 " (%s)!", ChannelId,
713 ChannelName);
714 *res = CHANNEL_RC_ALREADY_OPEN;
715 goto out;
716
717 case DVC_CHANNEL_CLOSED:
718 case DVC_CHANNEL_INIT:
719 default:
720 {
721 WLog_Print(drdynvc->log, WLOG_ERROR, "not expecting a createChannel from state %s",
722 channel_state2str(channel->state));
723 *res = CHANNEL_RC_INITIALIZATION_ERROR;
724 goto out;
725 }
726 }
727 }
728 else
729 {
730 if (!(channel = dvcman_channel_new(drdynvc, pChannelMgr, ChannelId, ChannelName)))
731 {
732 WLog_Print(drdynvc->log, WLOG_ERROR, "dvcman_channel_new failed!");
733 *res = CHANNEL_RC_NO_MEMORY;
734 goto out;
735 }
736 }
737
738 if (!HashTable_Insert(dvcman->channelsById, &channel->channel_id, channel))
739 {
740 WLog_Print(drdynvc->log, WLOG_ERROR, "unable to register channel in our channel list");
741 *res = ERROR_INTERNAL_ERROR;
742 dvcman_channel_free(channel);
743 channel = NULL;
744 goto out;
745 }
746
747 channel->iface.Write = dvcman_write_channel;
748 channel->iface.Close = dvcman_close_channel_iface;
749 bAccept = TRUE;
750
751 *res = listener->listener_callback->OnNewChannelConnection(
752 listener->listener_callback, &channel->iface, NULL, &bAccept, &pCallback);
753
754 if (*res != CHANNEL_RC_OK)
755 {
756 WLog_Print(drdynvc->log, WLOG_ERROR,
757 "OnNewChannelConnection failed with error %" PRIu32 "!", *res);
758 *res = ERROR_INTERNAL_ERROR;
759 dvcman_channel_unref(channel);
760 goto out;
761 }
762
763 if (!bAccept)
764 {
765 WLog_Print(drdynvc->log, WLOG_ERROR, "OnNewChannelConnection returned with bAccept FALSE!");
766 *res = ERROR_INTERNAL_ERROR;
767 dvcman_channel_unref(channel);
768 channel = NULL;
769 goto out;
770 }
771
772 WLog_Print(drdynvc->log, WLOG_DEBUG, "listener %s created new channel %" PRIu32 "",
773 listener->channel_name, channel->channel_id);
774 channel->state = DVC_CHANNEL_RUNNING;
775 channel->channel_callback = pCallback;
776 channel->pInterface = listener->iface.pInterface;
777 context = dvcman->drdynvc->context;
778
779 IFCALLRET(context->OnChannelConnected, *res, context, ChannelName, listener->iface.pInterface);
780 if (*res != CHANNEL_RC_OK)
781 {
782 WLog_Print(drdynvc->log, WLOG_ERROR,
783 "context.OnChannelConnected failed with error %" PRIu32 "", *res);
784 }
785
786out:
787 HashTable_Unlock(dvcman->listeners);
788
789 return channel;
790}
791
797static UINT dvcman_open_channel(drdynvcPlugin* drdynvc, DVCMAN_CHANNEL* channel)
798{
799 UINT error = CHANNEL_RC_OK;
800
801 WINPR_ASSERT(drdynvc);
802 WINPR_ASSERT(channel);
803 if (channel->state == DVC_CHANNEL_RUNNING)
804 {
805 IWTSVirtualChannelCallback* pCallback = channel->channel_callback;
806
807 if (pCallback->OnOpen)
808 {
809 check_open_close_receive(channel);
810 error = pCallback->OnOpen(pCallback);
811 if (error)
812 {
813 WLog_Print(drdynvc->log, WLOG_ERROR, "OnOpen failed with error %" PRIu32 "!",
814 error);
815 goto out;
816 }
817 }
818
819 WLog_Print(drdynvc->log, WLOG_DEBUG, "open_channel: ChannelId %" PRIu32 "",
820 channel->channel_id);
821 }
822
823out:
824 return error;
825}
826
832static UINT dvcman_receive_channel_data_first(DVCMAN_CHANNEL* channel, UINT32 length)
833{
834 WINPR_ASSERT(channel);
835 WINPR_ASSERT(channel->dvcman);
836 if (channel->dvc_data)
837 Stream_Release(channel->dvc_data);
838
839 channel->dvc_data = StreamPool_Take(channel->dvcman->pool, length);
840
841 if (!channel->dvc_data)
842 {
843 drdynvcPlugin* drdynvc = channel->dvcman->drdynvc;
844 WLog_Print(drdynvc->log, WLOG_ERROR, "StreamPool_Take failed!");
845 return CHANNEL_RC_NO_MEMORY;
846 }
847
848 channel->dvc_data_length = length;
849 return CHANNEL_RC_OK;
850}
851
857static UINT dvcman_receive_channel_data(DVCMAN_CHANNEL* channel, wStream* data,
858 WINPR_ATTR_UNUSED UINT32 ThreadingFlags)
859{
860 UINT status = CHANNEL_RC_OK;
861 size_t dataSize = Stream_GetRemainingLength(data);
862
863 WINPR_ASSERT(channel);
864 WINPR_ASSERT(channel->dvcman);
865 if (channel->dvc_data)
866 {
867 drdynvcPlugin* drdynvc = channel->dvcman->drdynvc;
868
869 /* Fragmented data */
870 if (Stream_GetPosition(channel->dvc_data) + dataSize > channel->dvc_data_length)
871 {
872 WLog_Print(drdynvc->log, WLOG_ERROR, "data exceeding declared length!");
873 Stream_Release(channel->dvc_data);
874 channel->dvc_data = NULL;
875 status = ERROR_INVALID_DATA;
876 goto out;
877 }
878
879 Stream_Copy(data, channel->dvc_data, dataSize);
880
881 if (Stream_GetPosition(channel->dvc_data) >= channel->dvc_data_length)
882 {
883 Stream_SealLength(channel->dvc_data);
884 Stream_SetPosition(channel->dvc_data, 0);
885
886 status = dvcman_call_on_receive(channel, channel->dvc_data);
887 Stream_Release(channel->dvc_data);
888 channel->dvc_data = NULL;
889 }
890 }
891 else
892 status = dvcman_call_on_receive(channel, data);
893
894out:
895 return status;
896}
897
898static UINT8 drdynvc_write_variable_uint(wStream* s, UINT32 val)
899{
900 UINT8 cb = 0;
901
902 if (val <= 0xFF)
903 {
904 cb = 0;
905 Stream_Write_UINT8(s, (UINT8)val);
906 }
907 else if (val <= 0xFFFF)
908 {
909 cb = 1;
910 Stream_Write_UINT16(s, (UINT16)val);
911 }
912 else
913 {
914 cb = 2;
915 Stream_Write_UINT32(s, val);
916 }
917
918 return cb;
919}
920
926static UINT drdynvc_send(drdynvcPlugin* drdynvc, wStream* s)
927{
928 UINT status = 0;
929
930 if (!drdynvc)
931 status = CHANNEL_RC_BAD_CHANNEL_HANDLE;
932 else
933 {
934 WINPR_ASSERT(drdynvc->channelEntryPoints.pVirtualChannelWriteEx);
935 status = drdynvc->channelEntryPoints.pVirtualChannelWriteEx(
936 drdynvc->InitHandle, drdynvc->OpenHandle, Stream_Buffer(s),
937 (UINT32)Stream_GetPosition(s), s);
938 }
939
940 switch (status)
941 {
942 case CHANNEL_RC_OK:
943 return CHANNEL_RC_OK;
944
945 case CHANNEL_RC_NOT_CONNECTED:
946 Stream_Release(s);
947 return CHANNEL_RC_OK;
948
949 case CHANNEL_RC_BAD_CHANNEL_HANDLE:
950 Stream_Release(s);
951 WLog_ERR(TAG, "VirtualChannelWriteEx failed with CHANNEL_RC_BAD_CHANNEL_HANDLE");
952 return status;
953
954 default:
955 Stream_Release(s);
956 WLog_Print(drdynvc->log, WLOG_ERROR,
957 "VirtualChannelWriteEx failed with %s [%08" PRIX32 "]",
958 WTSErrorToString(status), status);
959 return status;
960 }
961}
962
968static UINT drdynvc_write_data(drdynvcPlugin* drdynvc, UINT32 ChannelId, const BYTE* data,
969 UINT32 dataSize, BOOL* close)
970{
971 wStream* data_out = NULL;
972 size_t pos = 0;
973 UINT8 cbChId = 0;
974 UINT8 cbLen = 0;
975 UINT status = CHANNEL_RC_BAD_INIT_HANDLE;
976 DVCMAN* dvcman = NULL;
977
978 if (!drdynvc)
979 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
980
981 dvcman = (DVCMAN*)drdynvc->channel_mgr;
982 WINPR_ASSERT(dvcman);
983
984 WLog_Print(drdynvc->log, WLOG_TRACE, "write_data: ChannelId=%" PRIu32 " size=%" PRIu32 "",
985 ChannelId, dataSize);
986 data_out = StreamPool_Take(dvcman->pool, CHANNEL_CHUNK_LENGTH);
987
988 if (!data_out)
989 {
990 WLog_Print(drdynvc->log, WLOG_ERROR, "StreamPool_Take failed!");
991 return CHANNEL_RC_NO_MEMORY;
992 }
993
994 Stream_SetPosition(data_out, 1);
995 cbChId = drdynvc_write_variable_uint(data_out, ChannelId);
996 pos = Stream_GetPosition(data_out);
997
998 if (dataSize == 0)
999 {
1000 /* TODO: shall treat that case with write(0) that do a close */
1001 *close = TRUE;
1002 Stream_Release(data_out);
1003 }
1004 else if (dataSize <= CHANNEL_CHUNK_LENGTH - pos)
1005 {
1006 Stream_SetPosition(data_out, 0);
1007 Stream_Write_UINT8(data_out, (DATA_PDU << 4) | cbChId);
1008 Stream_SetPosition(data_out, pos);
1009 Stream_Write(data_out, data, dataSize);
1010 status = drdynvc_send(drdynvc, data_out);
1011 }
1012 else
1013 {
1014 /* Fragment the data */
1015 cbLen = drdynvc_write_variable_uint(data_out, dataSize);
1016 pos = Stream_GetPosition(data_out);
1017 Stream_SetPosition(data_out, 0);
1018
1019 const INT32 pdu = (DATA_FIRST_PDU << 4) | cbChId | (cbLen << 2);
1020 Stream_Write_UINT8(data_out, WINPR_ASSERTING_INT_CAST(UINT8, pdu));
1021 Stream_SetPosition(data_out, pos);
1022
1023 {
1024 WINPR_ASSERT(pos <= CHANNEL_CHUNK_LENGTH);
1025 const uint32_t chunkLength =
1026 CHANNEL_CHUNK_LENGTH - WINPR_ASSERTING_INT_CAST(uint32_t, pos);
1027 Stream_Write(data_out, data, chunkLength);
1028
1029 data += chunkLength;
1030 dataSize -= chunkLength;
1031 }
1032 status = drdynvc_send(drdynvc, data_out);
1033
1034 while (status == CHANNEL_RC_OK && dataSize > 0)
1035 {
1036 data_out = StreamPool_Take(dvcman->pool, CHANNEL_CHUNK_LENGTH);
1037
1038 if (!data_out)
1039 {
1040 WLog_Print(drdynvc->log, WLOG_ERROR, "StreamPool_Take failed!");
1041 return CHANNEL_RC_NO_MEMORY;
1042 }
1043
1044 Stream_SetPosition(data_out, 1);
1045 cbChId = drdynvc_write_variable_uint(data_out, ChannelId);
1046 pos = Stream_GetPosition(data_out);
1047 Stream_SetPosition(data_out, 0);
1048 Stream_Write_UINT8(data_out, (DATA_PDU << 4) | cbChId);
1049 Stream_SetPosition(data_out, pos);
1050
1051 uint32_t chunkLength = dataSize;
1052
1053 WINPR_ASSERT(pos <= CHANNEL_CHUNK_LENGTH);
1054 if (chunkLength > CHANNEL_CHUNK_LENGTH - pos)
1055 chunkLength = CHANNEL_CHUNK_LENGTH - WINPR_ASSERTING_INT_CAST(uint32_t, pos);
1056
1057 Stream_Write(data_out, data, chunkLength);
1058 data += chunkLength;
1059 dataSize -= chunkLength;
1060 status = drdynvc_send(drdynvc, data_out);
1061 }
1062 }
1063
1064 if (status != CHANNEL_RC_OK)
1065 {
1066 WLog_Print(drdynvc->log, WLOG_ERROR, "VirtualChannelWriteEx failed with %s [%08" PRIX32 "]",
1067 WTSErrorToString(status), status);
1068 return status;
1069 }
1070
1071 return CHANNEL_RC_OK;
1072}
1073
1079static UINT drdynvc_send_capability_response(drdynvcPlugin* drdynvc)
1080{
1081 UINT status = 0;
1082 wStream* s = NULL;
1083 DVCMAN* dvcman = NULL;
1084
1085 if (!drdynvc)
1086 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1087
1088 dvcman = (DVCMAN*)drdynvc->channel_mgr;
1089 WINPR_ASSERT(dvcman);
1090
1091 WLog_Print(drdynvc->log, WLOG_TRACE, "capability_response");
1092 s = StreamPool_Take(dvcman->pool, 4);
1093
1094 if (!s)
1095 {
1096 WLog_Print(drdynvc->log, WLOG_ERROR, "Stream_Ndrdynvc_write_variable_uintew failed!");
1097 return CHANNEL_RC_NO_MEMORY;
1098 }
1099
1100 Stream_Write_UINT16(s, 0x0050); /* Cmd+Sp+cbChId+Pad. Note: MSTSC sends 0x005c */
1101 Stream_Write_UINT16(s, drdynvc->version);
1102 status = drdynvc_send(drdynvc, s);
1103
1104 if (status != CHANNEL_RC_OK)
1105 {
1106 WLog_Print(drdynvc->log, WLOG_ERROR, "VirtualChannelWriteEx failed with %s [%08" PRIX32 "]",
1107 WTSErrorToString(status), status);
1108 }
1109
1110 return status;
1111}
1112
1118static UINT drdynvc_process_capability_request(drdynvcPlugin* drdynvc, int Sp, int cbChId,
1119 wStream* s)
1120{
1121 UINT status = 0;
1122
1123 if (!drdynvc)
1124 return CHANNEL_RC_BAD_INIT_HANDLE;
1125
1126 if (!Stream_CheckAndLogRequiredLength(TAG, s, 3))
1127 return ERROR_INVALID_DATA;
1128
1129 WLog_Print(drdynvc->log, WLOG_TRACE, "capability_request Sp=%d cbChId=%d", Sp, cbChId);
1130 Stream_Seek(s, 1); /* pad */
1131 Stream_Read_UINT16(s, drdynvc->version);
1132
1133 /* RDP8 servers offer version 3, though Microsoft forgot to document it
1134 * in their early documents. It behaves the same as version 2.
1135 */
1136 if ((drdynvc->version == 2) || (drdynvc->version == 3))
1137 {
1138 if (!Stream_CheckAndLogRequiredLength(TAG, s, 8))
1139 return ERROR_INVALID_DATA;
1140
1141 Stream_Read_UINT16(s, drdynvc->PriorityCharge0);
1142 Stream_Read_UINT16(s, drdynvc->PriorityCharge1);
1143 Stream_Read_UINT16(s, drdynvc->PriorityCharge2);
1144 Stream_Read_UINT16(s, drdynvc->PriorityCharge3);
1145 }
1146
1147 status = drdynvc_send_capability_response(drdynvc);
1148 drdynvc->state = DRDYNVC_STATE_READY;
1149 return status;
1150}
1151
1152static UINT32 drdynvc_cblen_to_bytes(int cbLen)
1153{
1154 switch (cbLen)
1155 {
1156 case 0:
1157 return 1;
1158
1159 case 1:
1160 return 2;
1161
1162 default:
1163 return 4;
1164 }
1165}
1166
1167static UINT32 drdynvc_read_variable_uint(wStream* s, int cbLen)
1168{
1169 UINT32 val = 0;
1170
1171 switch (cbLen)
1172 {
1173 case 0:
1174 Stream_Read_UINT8(s, val);
1175 break;
1176
1177 case 1:
1178 Stream_Read_UINT16(s, val);
1179 break;
1180
1181 default:
1182 Stream_Read_UINT32(s, val);
1183 break;
1184 }
1185
1186 return val;
1187}
1188
1194static UINT drdynvc_process_create_request(drdynvcPlugin* drdynvc, UINT8 Sp, UINT8 cbChId,
1195 wStream* s)
1196{
1197 UINT status = 0;
1198 wStream* data_out = NULL;
1199 UINT channel_status = 0;
1200 DVCMAN* dvcman = NULL;
1201 DVCMAN_CHANNEL* channel = NULL;
1202 INT32 retStatus = 0;
1203
1204 WINPR_UNUSED(Sp);
1205 if (!drdynvc)
1206 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1207
1208 dvcman = (DVCMAN*)drdynvc->channel_mgr;
1209 WINPR_ASSERT(dvcman);
1210
1211 if (drdynvc->state == DRDYNVC_STATE_CAPABILITIES)
1212 {
1218 drdynvc->version = 3;
1219
1220 if ((status = drdynvc_send_capability_response(drdynvc)))
1221 {
1222 WLog_Print(drdynvc->log, WLOG_ERROR, "drdynvc_send_capability_response failed!");
1223 return status;
1224 }
1225
1226 drdynvc->state = DRDYNVC_STATE_READY;
1227 }
1228
1229 if (!Stream_CheckAndLogRequiredLength(TAG, s, drdynvc_cblen_to_bytes(cbChId)))
1230 return ERROR_INVALID_DATA;
1231
1232 const UINT32 ChannelId = drdynvc_read_variable_uint(s, cbChId);
1233 const size_t pos = Stream_GetPosition(s);
1234 const char* name = Stream_ConstPointer(s);
1235 const size_t length = Stream_GetRemainingLength(s);
1236
1237 if (strnlen(name, length) >= length)
1238 return ERROR_INVALID_DATA;
1239
1240 WLog_Print(drdynvc->log, WLOG_DEBUG,
1241 "process_create_request: ChannelId=%" PRIu32 " ChannelName=%s", ChannelId, name);
1242
1243 data_out = StreamPool_Take(dvcman->pool, pos + 4);
1244 if (!data_out)
1245 {
1246 WLog_Print(drdynvc->log, WLOG_ERROR, "StreamPool_Take failed!");
1247 return CHANNEL_RC_NO_MEMORY;
1248 }
1249
1250 Stream_Write_UINT8(data_out, (CREATE_REQUEST_PDU << 4) | cbChId);
1251 Stream_SetPosition(s, 1);
1252 Stream_Copy(s, data_out, pos - 1);
1253
1254 channel =
1255 dvcman_create_channel(drdynvc, drdynvc->channel_mgr, ChannelId, name, &channel_status);
1256 switch (channel_status)
1257 {
1258 case CHANNEL_RC_OK:
1259 WLog_Print(drdynvc->log, WLOG_DEBUG, "channel created");
1260 retStatus = 0;
1261 break;
1262 case CHANNEL_RC_NO_MEMORY:
1263 WLog_Print(drdynvc->log, WLOG_DEBUG, "not enough memory for channel creation");
1264 retStatus = STATUS_NO_MEMORY;
1265 break;
1266 case ERROR_NOT_FOUND:
1267 WLog_Print(drdynvc->log, WLOG_DEBUG, "no listener for '%s'", name);
1268 retStatus = STATUS_NOT_FOUND; /* same code used by mstsc, STATUS_UNSUCCESSFUL */
1269 break;
1270 default:
1271 WLog_Print(drdynvc->log, WLOG_DEBUG, "channel creation error");
1272 retStatus = STATUS_UNSUCCESSFUL; /* same code used by mstsc, STATUS_UNSUCCESSFUL */
1273 break;
1274 }
1275 Stream_Write_INT32(data_out, retStatus);
1276
1277 status = drdynvc_send(drdynvc, data_out);
1278 if (status != CHANNEL_RC_OK)
1279 {
1280 WLog_Print(drdynvc->log, WLOG_ERROR, "VirtualChannelWriteEx failed with %s [%08" PRIX32 "]",
1281 WTSErrorToString(status), status);
1282 dvcman_channel_unref(channel);
1283 return status;
1284 }
1285
1286 if (channel_status == CHANNEL_RC_OK)
1287 {
1288 if ((status = dvcman_open_channel(drdynvc, channel)))
1289 {
1290 WLog_Print(drdynvc->log, WLOG_ERROR,
1291 "dvcman_open_channel failed with error %" PRIu32 "!", status);
1292 return status;
1293 }
1294 }
1295
1296 return status;
1297}
1298
1304static UINT drdynvc_process_data_first(drdynvcPlugin* drdynvc, int Sp, int cbChId, wStream* s,
1305 BOOL compressed, UINT32 ThreadingFlags)
1306{
1307 WINPR_ASSERT(drdynvc);
1308 if (!Stream_CheckAndLogRequiredLength(
1309 TAG, s, drdynvc_cblen_to_bytes(cbChId) + drdynvc_cblen_to_bytes(Sp)))
1310 return ERROR_INVALID_DATA;
1311
1312 UINT32 ChannelId = drdynvc_read_variable_uint(s, cbChId);
1313 UINT32 Length = drdynvc_read_variable_uint(s, Sp);
1314 WLog_Print(drdynvc->log, WLOG_TRACE,
1315 "process_data_first: Sp=%d cbChId=%d, ChannelId=%" PRIu32 " Length=%" PRIu32 "", Sp,
1316 cbChId, ChannelId, Length);
1317
1318 DVCMAN_CHANNEL* channel = dvcman_get_channel_by_id(drdynvc->channel_mgr, ChannelId, TRUE);
1319 if (!channel)
1320 {
1326 WLog_Print(drdynvc->log, WLOG_ERROR, "ChannelId %" PRIu32 " not found!", ChannelId);
1327 return CHANNEL_RC_OK;
1328 }
1329
1330 UINT status = CHANNEL_RC_OK;
1331 BOOL shouldFree = FALSE;
1332 if (channel->state != DVC_CHANNEL_RUNNING)
1333 goto out;
1334
1335 if (compressed)
1336 {
1337 BYTE* data = NULL;
1338 UINT32 dataSize = 0;
1339 if (zgfx_decompress(channel->decompressor, Stream_Pointer(s),
1340 WINPR_ASSERTING_INT_CAST(UINT32, Stream_GetRemainingLength(s)), &data,
1341 &dataSize, 0) < 0)
1342 {
1343 status = ERROR_INVALID_DATA;
1344 WLog_Print(drdynvc->log, WLOG_ERROR, "error de-compressing first packet");
1345 goto out;
1346 }
1347
1348 s = Stream_New(data, dataSize);
1349 if (!s)
1350 {
1351 status = CHANNEL_RC_NO_MEMORY;
1352 WLog_Print(drdynvc->log, WLOG_ERROR, "error allocating new Stream(len=%" PRIu32 ")",
1353 dataSize);
1354 free(data);
1355 goto out;
1356 }
1357 shouldFree = TRUE;
1358 }
1359
1360 status = dvcman_receive_channel_data_first(channel, Length);
1361
1362 if (status == CHANNEL_RC_OK)
1363 status = dvcman_receive_channel_data(channel, s, ThreadingFlags);
1364
1365 if (status != CHANNEL_RC_OK)
1366 status = dvcman_channel_close(channel, FALSE, FALSE);
1367
1368out:
1369 if (shouldFree)
1370 Stream_Free(s, TRUE);
1371 dvcman_channel_unref(channel);
1372 return status;
1373}
1374
1380static UINT drdynvc_process_data(drdynvcPlugin* drdynvc, int Sp, int cbChId, wStream* s,
1381 BOOL compressed, UINT32 ThreadingFlags)
1382{
1383 WINPR_ASSERT(drdynvc);
1384 if (!Stream_CheckAndLogRequiredLength(TAG, s, drdynvc_cblen_to_bytes(cbChId)))
1385 return ERROR_INVALID_DATA;
1386
1387 UINT32 ChannelId = drdynvc_read_variable_uint(s, cbChId);
1388 WLog_Print(drdynvc->log, WLOG_TRACE, "process_data: Sp=%d cbChId=%d, ChannelId=%" PRIu32 "", Sp,
1389 cbChId, ChannelId);
1390
1391 DVCMAN_CHANNEL* channel = dvcman_get_channel_by_id(drdynvc->channel_mgr, ChannelId, TRUE);
1392 if (!channel)
1393 {
1399 WLog_Print(drdynvc->log, WLOG_ERROR, "ChannelId %" PRIu32 " not found!", ChannelId);
1400 return CHANNEL_RC_OK;
1401 }
1402
1403 BOOL shouldFree = FALSE;
1404 UINT status = CHANNEL_RC_OK;
1405 if (channel->state != DVC_CHANNEL_RUNNING)
1406 goto out;
1407
1408 if (compressed)
1409 {
1410 BYTE* data = NULL;
1411 UINT32 dataSize = 0;
1412
1413 if (zgfx_decompress(channel->decompressor, Stream_Pointer(s),
1414 WINPR_ASSERTING_INT_CAST(UINT32, Stream_GetRemainingLength(s)), &data,
1415 &dataSize, 0) < 0)
1416 {
1417 status = ERROR_INVALID_DATA;
1418 WLog_Print(drdynvc->log, WLOG_ERROR, "error de-compressing data packet");
1419 goto out;
1420 }
1421
1422 s = Stream_New(data, dataSize);
1423 if (!s)
1424 {
1425 status = CHANNEL_RC_NO_MEMORY;
1426 WLog_Print(drdynvc->log, WLOG_ERROR, "error allocating new Stream(len=%" PRIu32 ")",
1427 dataSize);
1428 free(data);
1429 goto out;
1430 }
1431 shouldFree = TRUE;
1432 }
1433
1434 status = dvcman_receive_channel_data(channel, s, ThreadingFlags);
1435 if (status != CHANNEL_RC_OK)
1436 status = dvcman_channel_close(channel, FALSE, FALSE);
1437
1438out:
1439 if (shouldFree)
1440 Stream_Free(s, TRUE);
1441 dvcman_channel_unref(channel);
1442 return status;
1443}
1444
1450static UINT drdynvc_process_close_request(drdynvcPlugin* drdynvc, int Sp, int cbChId, wStream* s)
1451{
1452 UINT32 ChannelId = 0;
1453 DVCMAN_CHANNEL* channel = NULL;
1454
1455 WINPR_ASSERT(drdynvc);
1456 if (!Stream_CheckAndLogRequiredLength(TAG, s, drdynvc_cblen_to_bytes(cbChId)))
1457 return ERROR_INVALID_DATA;
1458
1459 ChannelId = drdynvc_read_variable_uint(s, cbChId);
1460 WLog_Print(drdynvc->log, WLOG_DEBUG,
1461 "process_close_request: Sp=%d cbChId=%d, ChannelId=%" PRIu32 "", Sp, cbChId,
1462 ChannelId);
1463
1464 channel = dvcman_get_channel_by_id(drdynvc->channel_mgr, ChannelId, TRUE);
1465 if (!channel)
1466 {
1467 WLog_Print(drdynvc->log, WLOG_ERROR, "dvcman_close_request channel %" PRIu32 " not present",
1468 ChannelId);
1469 return CHANNEL_RC_OK;
1470 }
1471
1472 dvcman_channel_close(channel, TRUE, FALSE);
1473 dvcman_channel_unref(channel);
1474 return CHANNEL_RC_OK;
1475}
1476
1482static UINT drdynvc_order_recv(drdynvcPlugin* drdynvc, wStream* s, UINT32 ThreadingFlags)
1483{
1484 WINPR_ASSERT(drdynvc);
1485 if (!Stream_CheckAndLogRequiredLength(TAG, s, 1))
1486 return ERROR_INVALID_DATA;
1487
1488 UINT8 value = Stream_Get_UINT8(s);
1489 const UINT8 Cmd = (value & 0xf0) >> 4;
1490 const UINT8 Sp = (value & 0x0c) >> 2;
1491 const UINT8 cbChId = (value & 0x03) >> 0;
1492 WLog_Print(drdynvc->log, WLOG_TRACE, "order_recv: Cmd=%s, Sp=%" PRIu8 " cbChId=%" PRIu8,
1493 drdynvc_get_packet_type(Cmd), Sp, cbChId);
1494
1495 switch (Cmd)
1496 {
1497 case CAPABILITY_REQUEST_PDU:
1498 return drdynvc_process_capability_request(drdynvc, Sp, cbChId, s);
1499
1500 case CREATE_REQUEST_PDU:
1501 return drdynvc_process_create_request(drdynvc, Sp, cbChId, s);
1502
1503 case DATA_FIRST_PDU:
1504 case DATA_FIRST_COMPRESSED_PDU:
1505 return drdynvc_process_data_first(drdynvc, Sp, cbChId, s,
1506 (Cmd == DATA_FIRST_COMPRESSED_PDU), ThreadingFlags);
1507
1508 case DATA_PDU:
1509 case DATA_COMPRESSED_PDU:
1510 return drdynvc_process_data(drdynvc, Sp, cbChId, s, (Cmd == DATA_COMPRESSED_PDU),
1511 ThreadingFlags);
1512
1513 case CLOSE_REQUEST_PDU:
1514 return drdynvc_process_close_request(drdynvc, Sp, cbChId, s);
1515
1516 case SOFT_SYNC_RESPONSE_PDU:
1517 WLog_Print(drdynvc->log, WLOG_ERROR,
1518 "not expecting a SOFT_SYNC_RESPONSE_PDU as a client");
1519 return ERROR_INTERNAL_ERROR;
1520
1521 default:
1522 WLog_Print(drdynvc->log, WLOG_ERROR, "unknown drdynvc cmd 0x%x", Cmd);
1523 return ERROR_INTERNAL_ERROR;
1524 }
1525}
1526
1532static UINT drdynvc_virtual_channel_event_data_received(drdynvcPlugin* drdynvc, void* pData,
1533 UINT32 dataLength, UINT32 totalLength,
1534 UINT32 dataFlags)
1535{
1536 wStream* data_in = NULL;
1537
1538 WINPR_ASSERT(drdynvc);
1539 if ((dataFlags & CHANNEL_FLAG_SUSPEND) || (dataFlags & CHANNEL_FLAG_RESUME))
1540 {
1541 return CHANNEL_RC_OK;
1542 }
1543
1544 if (dataFlags & CHANNEL_FLAG_FIRST)
1545 {
1546 DVCMAN* mgr = (DVCMAN*)drdynvc->channel_mgr;
1547 if (drdynvc->data_in)
1548 Stream_Release(drdynvc->data_in);
1549
1550 drdynvc->data_in = StreamPool_Take(mgr->pool, totalLength);
1551 }
1552
1553 if (!(data_in = drdynvc->data_in))
1554 {
1555 WLog_Print(drdynvc->log, WLOG_ERROR, "StreamPool_Take failed!");
1556 return CHANNEL_RC_NO_MEMORY;
1557 }
1558
1559 if (!Stream_EnsureRemainingCapacity(data_in, dataLength))
1560 {
1561 WLog_Print(drdynvc->log, WLOG_ERROR, "Stream_EnsureRemainingCapacity failed!");
1562 Stream_Release(drdynvc->data_in);
1563 drdynvc->data_in = NULL;
1564 return ERROR_INTERNAL_ERROR;
1565 }
1566
1567 Stream_Write(data_in, pData, dataLength);
1568
1569 if (dataFlags & CHANNEL_FLAG_LAST)
1570 {
1571 const size_t cap = Stream_Capacity(data_in);
1572 const size_t pos = Stream_GetPosition(data_in);
1573 if (cap < pos)
1574 {
1575 WLog_Print(drdynvc->log, WLOG_ERROR, "drdynvc_plugin_process_received: read error");
1576 return ERROR_INVALID_DATA;
1577 }
1578
1579 drdynvc->data_in = NULL;
1580 Stream_SealLength(data_in);
1581 Stream_SetPosition(data_in, 0);
1582
1583 if (drdynvc->async)
1584 {
1585 if (!MessageQueue_Post(drdynvc->queue, NULL, 0, (void*)data_in, NULL))
1586 {
1587 WLog_Print(drdynvc->log, WLOG_ERROR, "MessageQueue_Post failed!");
1588 return ERROR_INTERNAL_ERROR;
1589 }
1590 }
1591 else
1592 {
1593 UINT error = drdynvc_order_recv(drdynvc, data_in, TRUE);
1594 Stream_Release(data_in);
1595
1596 if (error)
1597 {
1598 WLog_Print(drdynvc->log, WLOG_WARN,
1599 "drdynvc_order_recv failed with error %" PRIu32 "!", error);
1600 return error;
1601 }
1602 }
1603 }
1604
1605 return CHANNEL_RC_OK;
1606}
1607
1608static void VCAPITYPE drdynvc_virtual_channel_open_event_ex(LPVOID lpUserParam, DWORD openHandle,
1609 UINT event, LPVOID pData,
1610 UINT32 dataLength, UINT32 totalLength,
1611 UINT32 dataFlags)
1612{
1613 UINT error = CHANNEL_RC_OK;
1614 drdynvcPlugin* drdynvc = (drdynvcPlugin*)lpUserParam;
1615
1616 WINPR_ASSERT(drdynvc);
1617 switch (event)
1618 {
1619 case CHANNEL_EVENT_DATA_RECEIVED:
1620 if (!drdynvc || (drdynvc->OpenHandle != openHandle))
1621 {
1622 WLog_ERR(TAG, "drdynvc_virtual_channel_open_event: error no match");
1623 return;
1624 }
1625 if ((error = drdynvc_virtual_channel_event_data_received(drdynvc, pData, dataLength,
1626 totalLength, dataFlags)))
1627 WLog_Print(drdynvc->log, WLOG_ERROR,
1628 "drdynvc_virtual_channel_event_data_received failed with error %" PRIu32
1629 "",
1630 error);
1631
1632 break;
1633
1634 case CHANNEL_EVENT_WRITE_CANCELLED:
1635 case CHANNEL_EVENT_WRITE_COMPLETE:
1636 {
1637 wStream* s = (wStream*)pData;
1638 Stream_Release(s);
1639 }
1640 break;
1641
1642 case CHANNEL_EVENT_USER:
1643 break;
1644 default:
1645 break;
1646 }
1647
1648 if (error && drdynvc && drdynvc->rdpcontext)
1649 setChannelError(drdynvc->rdpcontext, error,
1650 "drdynvc_virtual_channel_open_event reported an error");
1651}
1652
1653static DWORD WINAPI drdynvc_virtual_channel_client_thread(LPVOID arg)
1654{
1655 /* TODO: rewrite this */
1656 wStream* data = NULL;
1657 wMessage message = { 0 };
1658 UINT error = CHANNEL_RC_OK;
1659 drdynvcPlugin* drdynvc = (drdynvcPlugin*)arg;
1660
1661 if (!drdynvc)
1662 {
1663 ExitThread((DWORD)CHANNEL_RC_BAD_CHANNEL_HANDLE);
1664 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1665 }
1666
1667 while (1)
1668 {
1669 if (!MessageQueue_Wait(drdynvc->queue))
1670 {
1671 WLog_Print(drdynvc->log, WLOG_ERROR, "MessageQueue_Wait failed!");
1672 error = ERROR_INTERNAL_ERROR;
1673 break;
1674 }
1675
1676 if (!MessageQueue_Peek(drdynvc->queue, &message, TRUE))
1677 {
1678 WLog_Print(drdynvc->log, WLOG_ERROR, "MessageQueue_Peek failed!");
1679 error = ERROR_INTERNAL_ERROR;
1680 break;
1681 }
1682
1683 if (message.id == WMQ_QUIT)
1684 break;
1685
1686 if (message.id == 0)
1687 {
1688 UINT32 ThreadingFlags = TRUE;
1689 data = (wStream*)message.wParam;
1690
1691 if ((error = drdynvc_order_recv(drdynvc, data, ThreadingFlags)))
1692 {
1693 WLog_Print(drdynvc->log, WLOG_WARN,
1694 "drdynvc_order_recv failed with error %" PRIu32 "!", error);
1695 }
1696
1697 Stream_Release(data);
1698 }
1699 }
1700
1701 {
1702 /* Disconnect remaining dynamic channels that the server did not.
1703 * This is required to properly shut down channels by calling the appropriate
1704 * event handlers. */
1705 DVCMAN* drdynvcMgr = (DVCMAN*)drdynvc->channel_mgr;
1706
1707 HashTable_Clear(drdynvcMgr->channelsById);
1708 }
1709
1710 if (error && drdynvc->rdpcontext)
1711 setChannelError(drdynvc->rdpcontext, error,
1712 "drdynvc_virtual_channel_client_thread reported an error");
1713
1714 ExitThread((DWORD)error);
1715 return error;
1716}
1717
1718static void drdynvc_queue_object_free(void* obj)
1719{
1720 wStream* s = NULL;
1721 wMessage* msg = (wMessage*)obj;
1722
1723 if (!msg || (msg->id != 0))
1724 return;
1725
1726 s = (wStream*)msg->wParam;
1727
1728 if (s)
1729 Stream_Release(s);
1730}
1731
1732static UINT drdynvc_virtual_channel_event_initialized(drdynvcPlugin* drdynvc, LPVOID pData,
1733 UINT32 dataLength)
1734{
1735 wObject* obj = NULL;
1736 WINPR_UNUSED(pData);
1737 WINPR_UNUSED(dataLength);
1738
1739 if (!drdynvc)
1740 goto error;
1741
1742 drdynvc->queue = MessageQueue_New(NULL);
1743
1744 if (!drdynvc->queue)
1745 {
1746 WLog_Print(drdynvc->log, WLOG_ERROR, "MessageQueue_New failed!");
1747 goto error;
1748 }
1749
1750 obj = MessageQueue_Object(drdynvc->queue);
1751 obj->fnObjectFree = drdynvc_queue_object_free;
1752 drdynvc->channel_mgr = dvcman_new(drdynvc);
1753
1754 if (!drdynvc->channel_mgr)
1755 {
1756 WLog_Print(drdynvc->log, WLOG_ERROR, "dvcman_new failed!");
1757 goto error;
1758 }
1759
1760 return CHANNEL_RC_OK;
1761error:
1762 return ERROR_INTERNAL_ERROR;
1763}
1764
1770static UINT drdynvc_virtual_channel_event_connected(drdynvcPlugin* drdynvc, LPVOID pData,
1771 UINT32 dataLength)
1772{
1773 UINT error = 0;
1774 UINT32 status = 0;
1775 rdpSettings* settings = NULL;
1776
1777 WINPR_ASSERT(drdynvc);
1778 WINPR_UNUSED(pData);
1779 WINPR_UNUSED(dataLength);
1780
1781 if (!drdynvc)
1782 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1783
1784 WINPR_ASSERT(drdynvc->channelEntryPoints.pVirtualChannelOpenEx);
1785 status = drdynvc->channelEntryPoints.pVirtualChannelOpenEx(
1786 drdynvc->InitHandle, &drdynvc->OpenHandle, drdynvc->channelDef.name,
1787 drdynvc_virtual_channel_open_event_ex);
1788
1789 if (status != CHANNEL_RC_OK)
1790 {
1791 WLog_Print(drdynvc->log, WLOG_ERROR, "pVirtualChannelOpen failed with %s [%08" PRIX32 "]",
1792 WTSErrorToString(status), status);
1793 return status;
1794 }
1795
1796 WINPR_ASSERT(drdynvc->rdpcontext);
1797 settings = drdynvc->rdpcontext->settings;
1798 WINPR_ASSERT(settings);
1799
1800 for (UINT32 index = 0;
1801 index < freerdp_settings_get_uint32(settings, FreeRDP_DynamicChannelCount); index++)
1802 {
1803 const ADDIN_ARGV* args =
1804 freerdp_settings_get_pointer_array(settings, FreeRDP_DynamicChannelArray, index);
1805 error = dvcman_load_addin(drdynvc, drdynvc->channel_mgr, args, drdynvc->rdpcontext);
1806
1807 if (CHANNEL_RC_OK != error)
1808 goto error;
1809 }
1810
1811 if ((error = dvcman_init(drdynvc, drdynvc->channel_mgr)))
1812 {
1813 WLog_Print(drdynvc->log, WLOG_ERROR, "dvcman_init failed with error %" PRIu32 "!", error);
1814 goto error;
1815 }
1816
1817 drdynvc->state = DRDYNVC_STATE_CAPABILITIES;
1818
1819 if (drdynvc->async)
1820 {
1821 if (!(drdynvc->thread = CreateThread(NULL, 0, drdynvc_virtual_channel_client_thread,
1822 (void*)drdynvc, 0, NULL)))
1823 {
1824 error = ERROR_INTERNAL_ERROR;
1825 WLog_Print(drdynvc->log, WLOG_ERROR, "CreateThread failed!");
1826 goto error;
1827 }
1828
1829 if (!SetThreadPriority(drdynvc->thread, THREAD_PRIORITY_HIGHEST))
1830 WLog_Print(drdynvc->log, WLOG_WARN, "SetThreadPriority failed, ignoring.");
1831 }
1832
1833error:
1834 return error;
1835}
1836
1842static UINT drdynvc_virtual_channel_event_disconnected(drdynvcPlugin* drdynvc)
1843{
1844 UINT status = 0;
1845
1846 if (!drdynvc)
1847 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1848
1849 if (drdynvc->OpenHandle == 0)
1850 return CHANNEL_RC_OK;
1851
1852 if (drdynvc->queue)
1853 {
1854 if (!MessageQueue_PostQuit(drdynvc->queue, 0))
1855 {
1856 status = GetLastError();
1857 WLog_Print(drdynvc->log, WLOG_ERROR,
1858 "MessageQueue_PostQuit failed with error %" PRIu32 "", status);
1859 return status;
1860 }
1861 }
1862
1863 if (drdynvc->thread)
1864 {
1865 if (WaitForSingleObject(drdynvc->thread, INFINITE) != WAIT_OBJECT_0)
1866 {
1867 status = GetLastError();
1868 WLog_Print(drdynvc->log, WLOG_ERROR,
1869 "WaitForSingleObject failed with error %" PRIu32 "", status);
1870 return status;
1871 }
1872
1873 (void)CloseHandle(drdynvc->thread);
1874 drdynvc->thread = NULL;
1875 }
1876 else
1877 {
1878 {
1879 /* Disconnect remaining dynamic channels that the server did not.
1880 * This is required to properly shut down channels by calling the appropriate
1881 * event handlers. */
1882 DVCMAN* drdynvcMgr = (DVCMAN*)drdynvc->channel_mgr;
1883
1884 HashTable_Clear(drdynvcMgr->channelsById);
1885 }
1886 }
1887
1888 WINPR_ASSERT(drdynvc->channelEntryPoints.pVirtualChannelCloseEx);
1889 status = drdynvc->channelEntryPoints.pVirtualChannelCloseEx(drdynvc->InitHandle,
1890 drdynvc->OpenHandle);
1891
1892 if (status != CHANNEL_RC_OK)
1893 {
1894 WLog_Print(drdynvc->log, WLOG_ERROR, "pVirtualChannelClose failed with %s [%08" PRIX32 "]",
1895 WTSErrorToString(status), status);
1896 }
1897
1898 dvcman_clear(drdynvc, drdynvc->channel_mgr);
1899 if (drdynvc->queue)
1900 MessageQueue_Clear(drdynvc->queue);
1901 drdynvc->OpenHandle = 0;
1902
1903 if (drdynvc->data_in)
1904 {
1905 Stream_Release(drdynvc->data_in);
1906 drdynvc->data_in = NULL;
1907 }
1908
1909 return status;
1910}
1911
1917static UINT drdynvc_virtual_channel_event_terminated(drdynvcPlugin* drdynvc)
1918{
1919 if (!drdynvc)
1920 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1921
1922 MessageQueue_Free(drdynvc->queue);
1923 drdynvc->queue = NULL;
1924
1925 if (drdynvc->channel_mgr)
1926 {
1927 dvcman_free(drdynvc, drdynvc->channel_mgr);
1928 drdynvc->channel_mgr = NULL;
1929 }
1930 drdynvc->InitHandle = 0;
1931 free(drdynvc->context);
1932 free(drdynvc);
1933 return CHANNEL_RC_OK;
1934}
1935
1936static UINT drdynvc_virtual_channel_event_attached(drdynvcPlugin* drdynvc)
1937{
1938 UINT error = CHANNEL_RC_OK;
1939 DVCMAN* dvcman = NULL;
1940
1941 if (!drdynvc)
1942 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1943
1944 dvcman = (DVCMAN*)drdynvc->channel_mgr;
1945
1946 if (!dvcman)
1947 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1948
1949 ArrayList_Lock(dvcman->plugins);
1950 for (size_t i = 0; i < ArrayList_Count(dvcman->plugins); i++)
1951 {
1952 IWTSPlugin* pPlugin = ArrayList_GetItem(dvcman->plugins, i);
1953
1954 error = IFCALLRESULT(CHANNEL_RC_OK, pPlugin->Attached, pPlugin);
1955 if (error != CHANNEL_RC_OK)
1956 {
1957 WLog_Print(drdynvc->log, WLOG_ERROR, "Attach failed with error %" PRIu32 "!", error);
1958 goto fail;
1959 }
1960 }
1961
1962fail:
1963 ArrayList_Unlock(dvcman->plugins);
1964 return error;
1965}
1966
1967static UINT drdynvc_virtual_channel_event_detached(drdynvcPlugin* drdynvc)
1968{
1969 UINT error = CHANNEL_RC_OK;
1970 DVCMAN* dvcman = NULL;
1971
1972 if (!drdynvc)
1973 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1974
1975 dvcman = (DVCMAN*)drdynvc->channel_mgr;
1976
1977 if (!dvcman)
1978 return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1979
1980 ArrayList_Lock(dvcman->plugins);
1981 for (size_t i = 0; i < ArrayList_Count(dvcman->plugins); i++)
1982 {
1983 IWTSPlugin* pPlugin = ArrayList_GetItem(dvcman->plugins, i);
1984
1985 error = IFCALLRESULT(CHANNEL_RC_OK, pPlugin->Detached, pPlugin);
1986 if (error != CHANNEL_RC_OK)
1987 {
1988 WLog_Print(drdynvc->log, WLOG_ERROR, "Detach failed with error %" PRIu32 "!", error);
1989 goto fail;
1990 }
1991 }
1992
1993fail:
1994 ArrayList_Unlock(dvcman->plugins);
1995
1996 return error;
1997}
1998
1999static VOID VCAPITYPE drdynvc_virtual_channel_init_event_ex(LPVOID lpUserParam, LPVOID pInitHandle,
2000 UINT event, LPVOID pData,
2001 UINT dataLength)
2002{
2003 UINT error = CHANNEL_RC_OK;
2004 drdynvcPlugin* drdynvc = (drdynvcPlugin*)lpUserParam;
2005
2006 if (!drdynvc || (drdynvc->InitHandle != pInitHandle))
2007 {
2008 WLog_ERR(TAG, "drdynvc_virtual_channel_init_event: error no match");
2009 return;
2010 }
2011
2012 switch (event)
2013 {
2014 case CHANNEL_EVENT_INITIALIZED:
2015 error = drdynvc_virtual_channel_event_initialized(drdynvc, pData, dataLength);
2016 break;
2017 case CHANNEL_EVENT_CONNECTED:
2018 if ((error = drdynvc_virtual_channel_event_connected(drdynvc, pData, dataLength)))
2019 WLog_Print(drdynvc->log, WLOG_ERROR,
2020 "drdynvc_virtual_channel_event_connected failed with error %" PRIu32 "",
2021 error);
2022
2023 break;
2024
2025 case CHANNEL_EVENT_DISCONNECTED:
2026 if ((error = drdynvc_virtual_channel_event_disconnected(drdynvc)))
2027 WLog_Print(drdynvc->log, WLOG_ERROR,
2028 "drdynvc_virtual_channel_event_disconnected failed with error %" PRIu32
2029 "",
2030 error);
2031
2032 break;
2033
2034 case CHANNEL_EVENT_TERMINATED:
2035 if ((error = drdynvc_virtual_channel_event_terminated(drdynvc)))
2036 WLog_Print(drdynvc->log, WLOG_ERROR,
2037 "drdynvc_virtual_channel_event_terminated failed with error %" PRIu32 "",
2038 error);
2039
2040 break;
2041
2042 case CHANNEL_EVENT_ATTACHED:
2043 if ((error = drdynvc_virtual_channel_event_attached(drdynvc)))
2044 WLog_Print(drdynvc->log, WLOG_ERROR,
2045 "drdynvc_virtual_channel_event_attached failed with error %" PRIu32 "",
2046 error);
2047
2048 break;
2049
2050 case CHANNEL_EVENT_DETACHED:
2051 if ((error = drdynvc_virtual_channel_event_detached(drdynvc)))
2052 WLog_Print(drdynvc->log, WLOG_ERROR,
2053 "drdynvc_virtual_channel_event_detached failed with error %" PRIu32 "",
2054 error);
2055
2056 break;
2057
2058 default:
2059 break;
2060 }
2061
2062 if (error && drdynvc->rdpcontext)
2063 setChannelError(drdynvc->rdpcontext, error,
2064 "drdynvc_virtual_channel_init_event_ex reported an error");
2065}
2066
2071static int drdynvc_get_version(DrdynvcClientContext* context)
2072{
2073 WINPR_ASSERT(context);
2074 drdynvcPlugin* drdynvc = (drdynvcPlugin*)context->handle;
2075 WINPR_ASSERT(drdynvc);
2076 return drdynvc->version;
2077}
2078
2079/* drdynvc is always built-in */
2080#define VirtualChannelEntryEx drdynvc_VirtualChannelEntryEx
2081
2082FREERDP_ENTRY_POINT(BOOL VCAPITYPE VirtualChannelEntryEx(PCHANNEL_ENTRY_POINTS_EX pEntryPoints,
2083 PVOID pInitHandle))
2084{
2085 UINT rc = 0;
2086 drdynvcPlugin* drdynvc = NULL;
2087 DrdynvcClientContext* context = NULL;
2088 CHANNEL_ENTRY_POINTS_FREERDP_EX* pEntryPointsEx = NULL;
2089 drdynvc = (drdynvcPlugin*)calloc(1, sizeof(drdynvcPlugin));
2090
2091 WINPR_ASSERT(pEntryPoints);
2092 if (!drdynvc)
2093 {
2094 WLog_ERR(TAG, "calloc failed!");
2095 return FALSE;
2096 }
2097
2098 drdynvc->channelDef.options =
2099 CHANNEL_OPTION_INITIALIZED | CHANNEL_OPTION_ENCRYPT_RDP | CHANNEL_OPTION_COMPRESS_RDP;
2100 (void)sprintf_s(drdynvc->channelDef.name, ARRAYSIZE(drdynvc->channelDef.name),
2101 DRDYNVC_SVC_CHANNEL_NAME);
2102 drdynvc->state = DRDYNVC_STATE_INITIAL;
2103 pEntryPointsEx = (CHANNEL_ENTRY_POINTS_FREERDP_EX*)pEntryPoints;
2104
2105 if ((pEntryPointsEx->cbSize >= sizeof(CHANNEL_ENTRY_POINTS_FREERDP_EX)) &&
2106 (pEntryPointsEx->MagicNumber == FREERDP_CHANNEL_MAGIC_NUMBER))
2107 {
2108 context = (DrdynvcClientContext*)calloc(1, sizeof(DrdynvcClientContext));
2109
2110 if (!context)
2111 {
2112 WLog_Print(drdynvc->log, WLOG_ERROR, "calloc failed!");
2113 free(drdynvc);
2114 return FALSE;
2115 }
2116
2117 context->handle = (void*)drdynvc;
2118 context->custom = NULL;
2119 drdynvc->context = context;
2120 context->GetVersion = drdynvc_get_version;
2121 drdynvc->rdpcontext = pEntryPointsEx->context;
2122 if (!freerdp_settings_get_bool(drdynvc->rdpcontext->settings,
2123 FreeRDP_TransportDumpReplay) &&
2124 !freerdp_settings_get_bool(drdynvc->rdpcontext->settings,
2125 FreeRDP_SynchronousDynamicChannels))
2126 drdynvc->async = TRUE;
2127 }
2128
2129 drdynvc->log = WLog_Get(TAG);
2130 WLog_Print(drdynvc->log, WLOG_DEBUG, "VirtualChannelEntryEx");
2131 CopyMemory(&(drdynvc->channelEntryPoints), pEntryPoints,
2133 drdynvc->InitHandle = pInitHandle;
2134
2135 WINPR_ASSERT(drdynvc->channelEntryPoints.pVirtualChannelInitEx);
2136 rc = drdynvc->channelEntryPoints.pVirtualChannelInitEx(
2137 drdynvc, context, pInitHandle, &drdynvc->channelDef, 1, VIRTUAL_CHANNEL_VERSION_WIN2000,
2138 drdynvc_virtual_channel_init_event_ex);
2139
2140 if (CHANNEL_RC_OK != rc)
2141 {
2142 WLog_Print(drdynvc->log, WLOG_ERROR, "pVirtualChannelInit failed with %s [%08" PRIX32 "]",
2143 WTSErrorToString(rc), rc);
2144 free(drdynvc->context);
2145 free(drdynvc);
2146 return FALSE;
2147 }
2148
2149 drdynvc->channelEntryPoints.pInterface = context;
2150 return TRUE;
2151}
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_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.
Definition svc.h:60
This struct contains function pointer to initialize/free objects.
Definition collections.h:52
OBJECT_FREE_FN fnObjectFree
Definition collections.h:58
OBJECT_EQUALS_FN fnObjectEquals
Definition collections.h:59
OBJECT_NEW_FN fnObjectNew
Definition collections.h:53