Coverage Report

Created: 2026-05-30 06:46

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/FreeRDP/libfreerdp/core/client.c
Line
Count
Source
1
/**
2
 * FreeRDP: A Remote Desktop Protocol Implementation
3
 * Client Channels
4
 *
5
 * Copyright 2014 Marc-Andre Moreau <marcandre.moreau@gmail.com>
6
 * Copyright 2015 Thincast Technologies GmbH
7
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
8
 *
9
 * Licensed under the Apache License, Version 2.0 (the "License");
10
 * you may not use this file except in compliance with the License.
11
 * You may obtain a copy of the License at
12
 *
13
 *     http://www.apache.org/licenses/LICENSE-2.0
14
 *
15
 * Unless required by applicable law or agreed to in writing, software
16
 * distributed under the License is distributed on an "AS IS" BASIS,
17
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18
 * See the License for the specific language governing permissions and
19
 * limitations under the License.
20
 */
21
22
#include <freerdp/config.h>
23
24
#include "settings.h"
25
26
#include <winpr/assert.h>
27
28
#include <freerdp/log.h>
29
#include <freerdp/channels/drdynvc.h>
30
31
#include "rdp.h"
32
#include "client.h"
33
34
#define TAG FREERDP_TAG("core.client")
35
36
typedef struct
37
{
38
  freerdp_channel_handle_fkt_t fkt;
39
  void* userdata;
40
} ChannelEventEntry;
41
42
/* Use this instance to get access to channels in VirtualChannelInit. It is set during
43
 * freerdp_connect so channels that use VirtualChannelInit must be initialized from the same thread
44
 * as freerdp_connect was called */
45
static WINPR_TLS freerdp* g_Instance = nullptr;
46
47
/* use global counter to ensure uniqueness across channel manager instances */
48
static volatile LONG g_OpenHandleSeq = 1;
49
50
/* HashTable mapping channel handles to CHANNEL_OPEN_DATA */
51
static INIT_ONCE g_ChannelHandlesOnce = INIT_ONCE_STATIC_INIT;
52
static wHashTable* g_ChannelHandles = nullptr;
53
54
static BOOL freerdp_channels_process_message_free(wMessage* message, DWORD type);
55
56
static CHANNEL_OPEN_DATA* freerdp_channels_find_channel_open_data_by_name(rdpChannels* channels,
57
                                                                          const char* name)
58
0
{
59
0
  for (int index = 0; index < channels->openDataCount; index++)
60
0
  {
61
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[index];
62
63
0
    if (strncmp(name, pChannelOpenData->name, CHANNEL_NAME_LEN + 1) == 0)
64
0
      return pChannelOpenData;
65
0
  }
66
67
0
  return nullptr;
68
0
}
69
70
/* returns rdpChannel for the channel name passed in */
71
static rdpMcsChannel* freerdp_channels_find_channel_by_name(rdpRdp* rdp, const char* name)
72
0
{
73
0
  rdpMcs* mcs = nullptr;
74
75
0
  if (!rdp)
76
0
    return nullptr;
77
78
0
  mcs = rdp->mcs;
79
80
0
  for (UINT32 index = 0; index < mcs->channelCount; index++)
81
0
  {
82
0
    rdpMcsChannel* channel = &mcs->channels[index];
83
84
0
    if (strncmp(name, channel->Name, CHANNEL_NAME_LEN + 1) == 0)
85
0
    {
86
0
      return channel;
87
0
    }
88
0
  }
89
90
0
  return nullptr;
91
0
}
92
93
static rdpMcsChannel* freerdp_channels_find_channel_by_id(rdpRdp* rdp, UINT16 channel_id)
94
0
{
95
0
  rdpMcsChannel* channel = nullptr;
96
0
  rdpMcs* mcs = nullptr;
97
98
0
  if (!rdp)
99
0
    return nullptr;
100
101
0
  mcs = rdp->mcs;
102
103
0
  for (UINT32 index = 0; index < mcs->channelCount; index++)
104
0
  {
105
0
    channel = &mcs->channels[index];
106
107
0
    if (channel->ChannelId == channel_id)
108
0
    {
109
0
      return channel;
110
0
    }
111
0
  }
112
113
0
  return nullptr;
114
0
}
115
116
static void channel_queue_message_free(wMessage* msg)
117
0
{
118
0
  CHANNEL_OPEN_EVENT* ev = nullptr;
119
120
0
  if (!msg || (msg->id != 0))
121
0
    return;
122
123
0
  ev = (CHANNEL_OPEN_EVENT*)msg->wParam;
124
0
  free(ev);
125
0
}
126
127
static void channel_queue_free(void* obj)
128
0
{
129
0
  wMessage* msg = (wMessage*)obj;
130
0
  freerdp_channels_process_message_free(msg, CHANNEL_EVENT_WRITE_CANCELLED);
131
0
  channel_queue_message_free(msg);
132
0
}
133
134
static BOOL CALLBACK init_channel_handles_table(WINPR_ATTR_UNUSED PINIT_ONCE once,
135
                                                WINPR_ATTR_UNUSED PVOID param,
136
                                                WINPR_ATTR_UNUSED PVOID* context)
137
1
{
138
1
  g_ChannelHandles = HashTable_New(TRUE);
139
1
  return TRUE;
140
1
}
141
142
static void* channel_event_entry_clone(const void* data)
143
0
{
144
0
  const ChannelEventEntry* entry = data;
145
0
  if (!entry)
146
0
    return nullptr;
147
148
0
  ChannelEventEntry* copy = calloc(1, sizeof(ChannelEventEntry));
149
0
  if (!copy)
150
0
    return nullptr;
151
0
  *copy = *entry;
152
0
  return copy;
153
0
}
154
155
rdpChannels* freerdp_channels_new(freerdp* instance)
156
9.26k
{
157
9.26k
  if (!InitOnceExecuteOnce(&g_ChannelHandlesOnce, init_channel_handles_table, nullptr, nullptr))
158
0
    return nullptr;
159
160
9.26k
  wObject* obj = nullptr;
161
9.26k
  rdpChannels* channels = nullptr;
162
9.26k
  channels = (rdpChannels*)calloc(1, sizeof(rdpChannels));
163
164
9.26k
  if (!channels)
165
0
    return nullptr;
166
167
9.26k
  if (!g_ChannelHandles)
168
0
    goto error;
169
9.26k
  if (!InitializeCriticalSectionAndSpinCount(&channels->channelsLock, 4000))
170
0
    goto error;
171
172
9.26k
  channels->instance = instance;
173
9.26k
  channels->queue = MessageQueue_New(nullptr);
174
175
9.26k
  if (!channels->queue)
176
0
    goto error;
177
178
9.26k
  obj = MessageQueue_Object(channels->queue);
179
9.26k
  obj->fnObjectFree = channel_queue_free;
180
181
9.26k
  channels->channelEvents = HashTable_New(FALSE);
182
9.26k
  if (!channels->channelEvents)
183
0
    goto error;
184
185
9.26k
  obj = HashTable_ValueObject(channels->channelEvents);
186
9.26k
  WINPR_ASSERT(obj);
187
9.26k
  obj->fnObjectFree = free;
188
9.26k
  obj->fnObjectNew = channel_event_entry_clone;
189
9.26k
  return channels;
190
0
error:
191
0
  WINPR_PRAGMA_DIAG_PUSH
192
0
  WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
193
0
  freerdp_channels_free(channels);
194
0
  WINPR_PRAGMA_DIAG_POP
195
0
  return nullptr;
196
9.26k
}
197
198
void freerdp_channels_free(rdpChannels* channels)
199
9.26k
{
200
9.26k
  if (!channels)
201
0
    return;
202
203
9.26k
  HashTable_Free(channels->channelEvents);
204
205
9.26k
  DeleteCriticalSection(&channels->channelsLock);
206
207
9.26k
  if (channels->queue)
208
9.26k
  {
209
9.26k
    MessageQueue_Free(channels->queue);
210
9.26k
    channels->queue = nullptr;
211
9.26k
  }
212
213
9.26k
  free(channels);
214
9.26k
}
215
216
/**
217
 * Function description
218
 *
219
 * @return 0 on success, otherwise a Win32 error code
220
 */
221
static UINT freerdp_drdynvc_on_channel_connected(DrdynvcClientContext* context, const char* name,
222
                                                 void* pInterface)
223
0
{
224
0
  UINT status = CHANNEL_RC_OK;
225
0
  ChannelConnectedEventArgs e = WINPR_C_ARRAY_INIT;
226
227
0
  WINPR_ASSERT(context);
228
229
0
  rdpChannels* channels = (rdpChannels*)context->custom;
230
0
  WINPR_ASSERT(channels);
231
232
0
  freerdp* instance = channels->instance;
233
0
  WINPR_ASSERT(instance);
234
0
  WINPR_ASSERT(instance->context);
235
236
0
  EventArgsInit(&e, "freerdp");
237
0
  e.name = name;
238
0
  e.pInterface = pInterface;
239
0
  if (PubSub_OnChannelConnected(instance->context->pubSub, instance->context, &e) < 0)
240
0
    return ERROR_INTERNAL_ERROR;
241
0
  return status;
242
0
}
243
244
/**
245
 * Function description
246
 *
247
 * @return 0 on success, otherwise a Win32 error code
248
 */
249
static UINT freerdp_drdynvc_on_channel_disconnected(DrdynvcClientContext* context, const char* name,
250
                                                    void* pInterface)
251
0
{
252
0
  UINT status = CHANNEL_RC_OK;
253
0
  ChannelDisconnectedEventArgs e = WINPR_C_ARRAY_INIT;
254
255
0
  WINPR_ASSERT(context);
256
0
  rdpChannels* channels = (rdpChannels*)context->custom;
257
0
  WINPR_ASSERT(channels);
258
259
0
  freerdp* instance = channels->instance;
260
0
  WINPR_ASSERT(instance);
261
0
  WINPR_ASSERT(instance->context);
262
263
0
  EventArgsInit(&e, "freerdp");
264
0
  e.name = name;
265
0
  e.pInterface = pInterface;
266
0
  if (PubSub_OnChannelDisconnected(instance->context->pubSub, instance->context, &e) < 0)
267
0
    return ERROR_INTERNAL_ERROR;
268
0
  return status;
269
0
}
270
271
static UINT freerdp_drdynvc_on_channel_attached(DrdynvcClientContext* context, const char* name,
272
                                                void* pInterface)
273
0
{
274
0
  UINT status = CHANNEL_RC_OK;
275
0
  ChannelAttachedEventArgs e = WINPR_C_ARRAY_INIT;
276
277
0
  WINPR_ASSERT(context);
278
0
  rdpChannels* channels = (rdpChannels*)context->custom;
279
0
  WINPR_ASSERT(channels);
280
281
0
  freerdp* instance = channels->instance;
282
0
  WINPR_ASSERT(instance);
283
0
  WINPR_ASSERT(instance->context);
284
285
0
  EventArgsInit(&e, "freerdp");
286
0
  e.name = name;
287
0
  e.pInterface = pInterface;
288
0
  if (PubSub_OnChannelAttached(instance->context->pubSub, instance->context, &e) < 0)
289
0
    return ERROR_INTERNAL_ERROR;
290
0
  return status;
291
0
}
292
293
static UINT freerdp_drdynvc_on_channel_detached(DrdynvcClientContext* context, const char* name,
294
                                                void* pInterface)
295
0
{
296
0
  UINT status = CHANNEL_RC_OK;
297
0
  ChannelDetachedEventArgs e = WINPR_C_ARRAY_INIT;
298
299
0
  WINPR_ASSERT(context);
300
0
  rdpChannels* channels = (rdpChannels*)context->custom;
301
0
  WINPR_ASSERT(channels);
302
303
0
  freerdp* instance = channels->instance;
304
0
  WINPR_ASSERT(instance);
305
0
  WINPR_ASSERT(instance->context);
306
307
0
  EventArgsInit(&e, "freerdp");
308
0
  e.name = name;
309
0
  e.pInterface = pInterface;
310
0
  if (PubSub_OnChannelDetached(instance->context->pubSub, instance->context, &e) < 0)
311
0
    return ERROR_INTERNAL_ERROR;
312
0
  return status;
313
0
}
314
315
void freerdp_channels_register_instance(WINPR_ATTR_UNUSED rdpChannels* channels, freerdp* instance)
316
0
{
317
  /* store instance in TLS so future VirtualChannelInit calls can use it */
318
0
  g_Instance = instance;
319
0
}
320
321
/**
322
 * go through and inform all the libraries that we are initialized
323
 * called only from main thread
324
 */
325
UINT freerdp_channels_pre_connect(rdpChannels* channels, freerdp* instance)
326
0
{
327
0
  UINT error = CHANNEL_RC_OK;
328
329
0
  MessageQueue_Clear(channels->queue);
330
331
0
  for (int index = 0; index < channels->clientDataCount; index++)
332
0
  {
333
0
    CHANNEL_CLIENT_DATA* pChannelClientData = &channels->clientDataList[index];
334
335
0
    if (pChannelClientData->pChannelInitEventProc)
336
0
    {
337
0
      pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
338
0
                                                CHANNEL_EVENT_INITIALIZED, nullptr, 0);
339
0
    }
340
0
    else if (pChannelClientData->pChannelInitEventProcEx)
341
0
    {
342
0
      pChannelClientData->pChannelInitEventProcEx(pChannelClientData->lpUserParam,
343
0
                                                  pChannelClientData->pInitHandle,
344
0
                                                  CHANNEL_EVENT_INITIALIZED, nullptr, 0);
345
0
    }
346
347
0
    error = getChannelError(instance->context);
348
0
    if (CHANNEL_RC_OK != error)
349
0
      break;
350
351
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[index];
352
0
    CHANNEL_INIT_DATA* pChannelInitData = &channels->initDataList[index];
353
0
    ChannelInitializedEventArgs e = WINPR_C_ARRAY_INIT;
354
0
    EventArgsInit(&e, "freerdp");
355
0
    e.name = pChannelOpenData->name;
356
0
    e.pInterface = pChannelInitData->pInterface;
357
0
    if (PubSub_OnChannelInitialized(instance->context->pubSub, instance->context, &e) < 0)
358
0
    {
359
0
      error = ERROR_INTERNAL_ERROR;
360
0
      break;
361
0
    }
362
0
  }
363
364
0
  return error;
365
0
}
366
367
UINT freerdp_channels_attach(freerdp* instance)
368
0
{
369
0
  UINT error = CHANNEL_RC_OK;
370
0
  const char* hostname = nullptr;
371
0
  size_t hostnameLength = 0;
372
0
  rdpChannels* channels = nullptr;
373
0
  CHANNEL_CLIENT_DATA* pChannelClientData = nullptr;
374
375
0
  WINPR_ASSERT(instance);
376
0
  WINPR_ASSERT(instance->context);
377
0
  WINPR_ASSERT(instance->context->settings);
378
379
0
  channels = instance->context->channels;
380
0
  hostname = freerdp_settings_get_string(instance->context->settings, FreeRDP_ServerHostname);
381
0
  WINPR_ASSERT(hostname);
382
0
  hostnameLength = strnlen(hostname, MAX_PATH);
383
384
0
  for (int index = 0; index < channels->clientDataCount; index++)
385
0
  {
386
0
    union
387
0
    {
388
0
      const void* cpv;
389
0
      void* pv;
390
0
    } cnv;
391
0
    ChannelAttachedEventArgs e = WINPR_C_ARRAY_INIT;
392
393
0
    cnv.cpv = hostname;
394
0
    pChannelClientData = &channels->clientDataList[index];
395
396
0
    if (pChannelClientData->pChannelInitEventProc)
397
0
    {
398
399
0
      pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
400
0
                                                CHANNEL_EVENT_ATTACHED, cnv.pv,
401
0
                                                (UINT)hostnameLength);
402
0
    }
403
0
    else if (pChannelClientData->pChannelInitEventProcEx)
404
0
    {
405
0
      pChannelClientData->pChannelInitEventProcEx(
406
0
          pChannelClientData->lpUserParam, pChannelClientData->pInitHandle,
407
0
          CHANNEL_EVENT_ATTACHED, cnv.pv, (UINT)hostnameLength);
408
0
    }
409
410
0
    error = getChannelError(instance->context);
411
0
    if (error != CHANNEL_RC_OK)
412
0
      goto fail;
413
414
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[index];
415
0
    EventArgsInit(&e, "freerdp");
416
0
    e.name = pChannelOpenData->name;
417
0
    e.pInterface = pChannelOpenData->pInterface;
418
0
    if (PubSub_OnChannelAttached(instance->context->pubSub, instance->context, &e) < 0)
419
0
    {
420
0
      error = ERROR_INTERNAL_ERROR;
421
0
      goto fail;
422
0
    }
423
0
  }
424
425
0
fail:
426
0
  return error;
427
0
}
428
429
UINT freerdp_channels_detach(freerdp* instance)
430
0
{
431
0
  UINT error = CHANNEL_RC_OK;
432
0
  const char* hostname = nullptr;
433
0
  size_t hostnameLength = 0;
434
0
  rdpChannels* channels = nullptr;
435
0
  rdpContext* context = nullptr;
436
0
  CHANNEL_CLIENT_DATA* pChannelClientData = nullptr;
437
438
0
  WINPR_ASSERT(instance);
439
440
0
  context = instance->context;
441
0
  WINPR_ASSERT(context);
442
443
0
  channels = context->channels;
444
0
  WINPR_ASSERT(channels);
445
446
0
  WINPR_ASSERT(context->settings);
447
0
  hostname = freerdp_settings_get_string(context->settings, FreeRDP_ServerHostname);
448
0
  WINPR_ASSERT(hostname);
449
0
  hostnameLength = strnlen(hostname, MAX_PATH);
450
451
0
  for (int index = 0; index < channels->clientDataCount; index++)
452
0
  {
453
0
    union
454
0
    {
455
0
      const void* cpv;
456
0
      void* pv;
457
0
    } cnv;
458
459
0
    ChannelDetachedEventArgs e = WINPR_C_ARRAY_INIT;
460
461
0
    cnv.cpv = hostname;
462
0
    pChannelClientData = &channels->clientDataList[index];
463
464
0
    if (pChannelClientData->pChannelInitEventProc)
465
0
    {
466
0
      pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
467
0
                                                CHANNEL_EVENT_DETACHED, cnv.pv,
468
0
                                                (UINT)hostnameLength);
469
0
    }
470
0
    else if (pChannelClientData->pChannelInitEventProcEx)
471
0
    {
472
0
      pChannelClientData->pChannelInitEventProcEx(
473
0
          pChannelClientData->lpUserParam, pChannelClientData->pInitHandle,
474
0
          CHANNEL_EVENT_DETACHED, cnv.pv, (UINT)hostnameLength);
475
0
    }
476
477
0
    error = getChannelError(context);
478
0
    if (error != CHANNEL_RC_OK)
479
0
      goto fail;
480
481
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[index];
482
0
    EventArgsInit(&e, "freerdp");
483
0
    e.name = pChannelOpenData->name;
484
0
    e.pInterface = pChannelOpenData->pInterface;
485
0
    if (PubSub_OnChannelDetached(context->pubSub, context, &e) < 0)
486
0
    {
487
0
      error = ERROR_INTERNAL_ERROR;
488
0
      goto fail;
489
0
    }
490
0
  }
491
492
0
fail:
493
0
  return error;
494
0
}
495
496
/**
497
 * go through and inform all the libraries that we are connected
498
 * this will tell the libraries that its ok to call MyVirtualChannelOpen
499
 * called only from main thread
500
 */
501
UINT freerdp_channels_post_connect(rdpChannels* channels, freerdp* instance)
502
0
{
503
0
  UINT error = CHANNEL_RC_OK;
504
505
0
  WINPR_ASSERT(channels);
506
0
  WINPR_ASSERT(instance);
507
0
  WINPR_ASSERT(instance->context);
508
0
  WINPR_ASSERT(instance->context->settings);
509
510
0
  channels->connected = TRUE;
511
0
  const char* hostname =
512
0
      freerdp_settings_get_string(instance->context->settings, FreeRDP_ServerHostname);
513
0
  WINPR_ASSERT(hostname);
514
0
  const size_t hostnameLength = strnlen(hostname, MAX_PATH);
515
516
0
  for (int index = 0; index < channels->clientDataCount; index++)
517
0
  {
518
0
    union
519
0
    {
520
0
      const void* pcb;
521
0
      void* pb;
522
0
    } cnv;
523
524
0
    CHANNEL_CLIENT_DATA* pChannelClientData = &channels->clientDataList[index];
525
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[index];
526
527
0
    cnv.pcb = hostname;
528
0
    if (pChannelClientData->pChannelInitEventProc)
529
0
    {
530
0
      pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
531
0
                                                CHANNEL_EVENT_CONNECTED, cnv.pb,
532
0
                                                (UINT)hostnameLength);
533
0
    }
534
0
    else if (pChannelClientData->pChannelInitEventProcEx)
535
0
    {
536
0
      pChannelClientData->pChannelInitEventProcEx(
537
0
          pChannelClientData->lpUserParam, pChannelClientData->pInitHandle,
538
0
          CHANNEL_EVENT_CONNECTED, cnv.pb, (UINT)hostnameLength);
539
0
    }
540
541
0
    error = getChannelError(instance->context);
542
0
    if (error != CHANNEL_RC_OK)
543
0
      goto fail;
544
545
0
    ChannelConnectedEventArgs e = WINPR_C_ARRAY_INIT;
546
0
    EventArgsInit(&e, "freerdp");
547
0
    e.name = pChannelOpenData->name;
548
0
    e.pInterface = pChannelOpenData->pInterface;
549
0
    if (PubSub_OnChannelConnected(instance->context->pubSub, instance->context, &e) < 0)
550
0
    {
551
0
      error = ERROR_INTERNAL_ERROR;
552
0
      goto fail;
553
0
    }
554
0
  }
555
556
0
  channels->drdynvc = (DrdynvcClientContext*)freerdp_channels_get_static_channel_interface(
557
0
      channels, DRDYNVC_SVC_CHANNEL_NAME);
558
559
0
  if (channels->drdynvc)
560
0
  {
561
0
    channels->drdynvc->custom = (void*)channels;
562
0
    channels->drdynvc->OnChannelConnected = freerdp_drdynvc_on_channel_connected;
563
0
    channels->drdynvc->OnChannelDisconnected = freerdp_drdynvc_on_channel_disconnected;
564
0
    channels->drdynvc->OnChannelAttached = freerdp_drdynvc_on_channel_attached;
565
0
    channels->drdynvc->OnChannelDetached = freerdp_drdynvc_on_channel_detached;
566
0
  }
567
568
0
fail:
569
0
  return error;
570
0
}
571
572
BOOL freerdp_channels_data(freerdp* instance, UINT16 channelId, const BYTE* cdata, size_t dataSize,
573
                           UINT32 flags, size_t totalSize)
574
0
{
575
0
  rdpMcs* mcs = nullptr;
576
0
  rdpChannels* channels = nullptr;
577
0
  rdpMcsChannel* channel = nullptr;
578
0
  CHANNEL_OPEN_DATA* pChannelOpenData = nullptr;
579
0
  union
580
0
  {
581
0
    const BYTE* pcb;
582
0
    BYTE* pb;
583
0
  } data;
584
585
0
  data.pcb = cdata;
586
0
  if (!instance || !data.pcb)
587
0
  {
588
0
    WLog_ERR(TAG, "(%p, %" PRIu16 ", %p, 0x%08x): Invalid arguments",
589
0
             WINPR_CXX_COMPAT_CAST(const void*, instance), channelId,
590
0
             WINPR_CXX_COMPAT_CAST(const void*, data.pcb), flags);
591
0
    return FALSE;
592
0
  }
593
594
0
  mcs = instance->context->rdp->mcs;
595
0
  channels = instance->context->channels;
596
597
0
  if (!channels || !mcs)
598
0
  {
599
0
    return FALSE;
600
0
  }
601
602
0
  for (UINT32 index = 0; index < mcs->channelCount; index++)
603
0
  {
604
0
    rdpMcsChannel* cur = &mcs->channels[index];
605
606
0
    if (cur->ChannelId == channelId)
607
0
    {
608
0
      channel = cur;
609
0
      break;
610
0
    }
611
0
  }
612
613
0
  if (!channel)
614
0
  {
615
0
    return FALSE;
616
0
  }
617
618
0
  pChannelOpenData = freerdp_channels_find_channel_open_data_by_name(channels, channel->Name);
619
620
0
  if (!pChannelOpenData)
621
0
  {
622
0
    return FALSE;
623
0
  }
624
625
0
  if (pChannelOpenData->pChannelOpenEventProc)
626
0
  {
627
0
    pChannelOpenData->pChannelOpenEventProc(pChannelOpenData->OpenHandle,
628
0
                                            CHANNEL_EVENT_DATA_RECEIVED, data.pb,
629
0
                                            (UINT32)dataSize, (UINT32)totalSize, flags);
630
0
  }
631
0
  else if (pChannelOpenData->pChannelOpenEventProcEx)
632
0
  {
633
0
    pChannelOpenData->pChannelOpenEventProcEx(
634
0
        pChannelOpenData->lpUserParam, pChannelOpenData->OpenHandle,
635
0
        CHANNEL_EVENT_DATA_RECEIVED, data.pb, (UINT32)dataSize, (UINT32)totalSize, flags);
636
0
  }
637
638
0
  return TRUE;
639
0
}
640
641
UINT16 freerdp_channels_get_id_by_name(freerdp* instance, const char* channel_name)
642
0
{
643
0
  if (!instance || !channel_name)
644
0
    return 0;
645
646
0
  rdpMcsChannel* mcsChannel =
647
0
      freerdp_channels_find_channel_by_name(instance->context->rdp, channel_name);
648
0
  if (!mcsChannel)
649
0
    return 0;
650
651
0
  return mcsChannel->ChannelId;
652
0
}
653
654
const char* freerdp_channels_get_name_by_id(freerdp* instance, UINT16 channelId)
655
0
{
656
0
  rdpMcsChannel* mcsChannel = nullptr;
657
0
  if (!instance)
658
0
    return nullptr;
659
660
0
  mcsChannel = freerdp_channels_find_channel_by_id(instance->context->rdp, channelId);
661
0
  if (!mcsChannel)
662
0
    return nullptr;
663
664
0
  return mcsChannel->Name;
665
0
}
666
667
BOOL freerdp_channels_process_message_free(wMessage* message, DWORD type)
668
0
{
669
0
  if (message->id == WMQ_QUIT)
670
0
  {
671
0
    return FALSE;
672
0
  }
673
674
0
  if (message->id == 0)
675
0
  {
676
0
    CHANNEL_OPEN_DATA* pChannelOpenData = nullptr;
677
0
    CHANNEL_OPEN_EVENT* item = (CHANNEL_OPEN_EVENT*)message->wParam;
678
679
0
    if (!item)
680
0
      return FALSE;
681
682
0
    pChannelOpenData = item->pChannelOpenData;
683
684
0
    if (pChannelOpenData->pChannelOpenEventProc)
685
0
    {
686
0
      pChannelOpenData->pChannelOpenEventProc(pChannelOpenData->OpenHandle, type,
687
0
                                              item->UserData, item->DataLength,
688
0
                                              item->DataLength, 0);
689
0
    }
690
0
    else if (pChannelOpenData->pChannelOpenEventProcEx)
691
0
    {
692
0
      pChannelOpenData->pChannelOpenEventProcEx(
693
0
          pChannelOpenData->lpUserParam, pChannelOpenData->OpenHandle, type, item->UserData,
694
0
          item->DataLength, item->DataLength, 0);
695
0
    }
696
0
  }
697
698
0
  return TRUE;
699
0
}
700
701
static BOOL freerdp_channels_process_message(freerdp* instance, wMessage* message)
702
0
{
703
0
  BOOL ret = TRUE;
704
0
  BOOL rc = FALSE;
705
706
0
  WINPR_ASSERT(instance);
707
0
  WINPR_ASSERT(message);
708
709
0
  if (message->id == WMQ_QUIT)
710
0
    goto fail;
711
0
  else if (message->id == 0)
712
0
  {
713
0
    rdpMcsChannel* channel = nullptr;
714
0
    CHANNEL_OPEN_DATA* pChannelOpenData = nullptr;
715
0
    CHANNEL_OPEN_EVENT* item = (CHANNEL_OPEN_EVENT*)message->wParam;
716
717
0
    if (!item)
718
0
      goto fail;
719
720
0
    pChannelOpenData = item->pChannelOpenData;
721
0
    if (pChannelOpenData->flags != 2)
722
0
    {
723
0
      freerdp_channels_process_message_free(message, CHANNEL_EVENT_WRITE_CANCELLED);
724
0
      goto fail;
725
0
    }
726
0
    channel =
727
0
        freerdp_channels_find_channel_by_name(instance->context->rdp, pChannelOpenData->name);
728
729
0
    if (channel)
730
0
      ret = instance->SendChannelData(instance, channel->ChannelId, item->Data,
731
0
                                      item->DataLength);
732
0
  }
733
734
0
  if (!freerdp_channels_process_message_free(message, CHANNEL_EVENT_WRITE_COMPLETE))
735
0
    goto fail;
736
737
0
  rc = ret;
738
739
0
fail:
740
0
  IFCALL(message->Free, message);
741
0
  return rc;
742
0
}
743
744
/**
745
 * called only from main thread
746
 */
747
static BOOL freerdp_channels_process_sync(rdpChannels* channels, freerdp* instance)
748
0
{
749
0
  BOOL status = TRUE;
750
0
  wMessage message = WINPR_C_ARRAY_INIT;
751
752
0
  WINPR_ASSERT(channels);
753
754
0
  while (MessageQueue_Peek(channels->queue, &message, TRUE))
755
0
  {
756
0
    if (!freerdp_channels_process_message(instance, &message))
757
0
      status = FALSE;
758
0
  }
759
760
0
  return status;
761
0
}
762
763
/**
764
 * called only from main thread
765
 */
766
#if defined(WITH_FREERDP_DEPRECATED)
767
BOOL freerdp_channels_get_fds(rdpChannels* channels, WINPR_ATTR_UNUSED freerdp* instance,
768
                              void** read_fds, int* read_count, WINPR_ATTR_UNUSED void** write_fds,
769
                              WINPR_ATTR_UNUSED int* write_count)
770
{
771
  void* pfd = nullptr;
772
  pfd = GetEventWaitObject(MessageQueue_Event(channels->queue));
773
774
  if (pfd)
775
  {
776
    read_fds[*read_count] = pfd;
777
    (*read_count)++;
778
  }
779
780
  return TRUE;
781
}
782
#endif
783
784
void* freerdp_channels_get_static_channel_interface(rdpChannels* channels, const char* name)
785
0
{
786
0
  void* pInterface = nullptr;
787
0
  CHANNEL_OPEN_DATA* pChannelOpenData =
788
0
      freerdp_channels_find_channel_open_data_by_name(channels, name);
789
790
0
  if (pChannelOpenData)
791
0
    pInterface = pChannelOpenData->pInterface;
792
793
0
  return pInterface;
794
0
}
795
796
HANDLE freerdp_channels_get_event_handle(freerdp* instance)
797
0
{
798
0
  if (!instance)
799
0
    return INVALID_HANDLE_VALUE;
800
801
0
  WINPR_ASSERT(instance->context);
802
803
0
  rdpChannels* channels = instance->context->channels;
804
0
  WINPR_ASSERT(channels);
805
806
0
  return MessageQueue_Event(channels->queue);
807
0
}
808
809
static BOOL channels_process(const void* key, void* value, void* arg)
810
0
{
811
0
  ChannelEventEntry* entry = value;
812
0
  rdpContext* context = arg;
813
814
0
  WINPR_UNUSED(key);
815
816
0
  if (!entry->fkt)
817
0
    return FALSE;
818
0
  return entry->fkt(context, entry->userdata);
819
0
}
820
821
int freerdp_channels_process_pending_messages(freerdp* instance)
822
0
{
823
0
  if (!instance)
824
0
    return -1;
825
826
0
  WINPR_ASSERT(instance->context);
827
828
0
  rdpChannels* channels = instance->context->channels;
829
0
  WINPR_ASSERT(channels);
830
831
0
  const DWORD status = WaitForSingleObject(MessageQueue_Event(channels->queue), 0);
832
0
  if (status == WAIT_OBJECT_0)
833
0
  {
834
0
    if (!freerdp_channels_process_sync(channels, instance))
835
0
      return -1;
836
0
  }
837
838
0
  if (!HashTable_Foreach(channels->channelEvents, channels_process, instance->context))
839
0
    return -1;
840
841
0
  return 1;
842
0
}
843
844
/**
845
 * called only from main thread
846
 */
847
BOOL freerdp_channels_check_fds(rdpChannels* channels, freerdp* instance)
848
0
{
849
0
  WINPR_ASSERT(channels);
850
0
  WINPR_UNUSED(channels);
851
852
0
  const int rc = freerdp_channels_process_pending_messages(instance);
853
0
  return rc == 1;
854
0
}
855
856
BOOL freerdp_client_channel_register(rdpChannels* channels, HANDLE handle,
857
                                     freerdp_channel_handle_fkt_t fkt, void* userdata)
858
0
{
859
0
  if (!channels || (handle == INVALID_HANDLE_VALUE) || !fkt)
860
0
  {
861
0
    WLog_ERR(TAG, "Invalid function arguments (channels=%p, handle=%p, fkt=%p, userdata=%p",
862
0
             WINPR_FUNC_PTR_CAST(channels, const void*), handle,
863
0
             WINPR_FUNC_PTR_CAST(fkt, const void*), userdata);
864
0
    return FALSE;
865
0
  }
866
867
0
  ChannelEventEntry entry = { .fkt = fkt, .userdata = userdata };
868
0
  return HashTable_Insert(channels->channelEvents, handle, &entry);
869
0
}
870
871
BOOL freerdp_client_channel_unregister(rdpChannels* channels, HANDLE handle)
872
0
{
873
0
  if (!channels || (handle == INVALID_HANDLE_VALUE))
874
0
  {
875
0
    WLog_ERR(TAG, "Invalid function arguments (channels=%p, handle=%p",
876
0
             WINPR_CXX_COMPAT_CAST(const void*, channels), handle);
877
0
    return FALSE;
878
0
  }
879
880
0
  return HashTable_Remove(channels->channelEvents, handle);
881
0
}
882
883
SSIZE_T freerdp_client_channel_get_registered_event_handles(rdpChannels* channels, HANDLE* events,
884
                                                            DWORD count)
885
0
{
886
0
  SSIZE_T rc = -1;
887
888
0
  WINPR_ASSERT(channels);
889
0
  WINPR_ASSERT(events || (count == 0));
890
891
0
  HashTable_Lock(channels->channelEvents);
892
0
  size_t len = HashTable_Count(channels->channelEvents);
893
0
  if (len <= count)
894
0
  {
895
0
    ULONG_PTR* keys = nullptr;
896
0
    const size_t nrKeys = HashTable_GetKeys(channels->channelEvents, &keys);
897
0
    if ((nrKeys <= SSIZE_MAX) && (nrKeys == len))
898
0
    {
899
0
      for (size_t x = 0; x < nrKeys; x++)
900
0
      {
901
0
        HANDLE cur = (HANDLE)keys[x];
902
0
        events[x] = cur;
903
0
      }
904
0
      rc = (SSIZE_T)nrKeys;
905
0
    }
906
0
    free(keys);
907
0
  }
908
0
  HashTable_Unlock(channels->channelEvents);
909
0
  return rc;
910
0
}
911
912
UINT freerdp_channels_disconnect(rdpChannels* channels, freerdp* instance)
913
0
{
914
0
  UINT error = CHANNEL_RC_OK;
915
0
  CHANNEL_CLIENT_DATA* pChannelClientData = nullptr;
916
917
0
  WINPR_ASSERT(channels);
918
919
0
  if (!channels->connected)
920
0
    return 0;
921
922
0
  freerdp_channels_check_fds(channels, instance);
923
924
  /* tell all libraries we are shutting down */
925
0
  for (int index = 0; index < channels->clientDataCount; index++)
926
0
  {
927
0
    ChannelDisconnectedEventArgs e = WINPR_C_ARRAY_INIT;
928
0
    pChannelClientData = &channels->clientDataList[index];
929
930
0
    if (pChannelClientData->pChannelInitEventProc)
931
0
    {
932
0
      pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
933
0
                                                CHANNEL_EVENT_DISCONNECTED, nullptr, 0);
934
0
    }
935
0
    else if (pChannelClientData->pChannelInitEventProcEx)
936
0
    {
937
0
      pChannelClientData->pChannelInitEventProcEx(pChannelClientData->lpUserParam,
938
0
                                                  pChannelClientData->pInitHandle,
939
0
                                                  CHANNEL_EVENT_DISCONNECTED, nullptr, 0);
940
0
    }
941
942
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[index];
943
0
    EventArgsInit(&e, "freerdp");
944
0
    e.name = pChannelOpenData->name;
945
0
    e.pInterface = pChannelOpenData->pInterface;
946
0
    if (PubSub_OnChannelDisconnected(instance->context->pubSub, instance->context, &e) < 0)
947
0
      error = ERROR_INTERNAL_ERROR;
948
0
  }
949
950
0
  channels->connected = FALSE;
951
952
  /* Flush pending messages */
953
0
  freerdp_channels_check_fds(channels, instance);
954
0
  return error;
955
0
}
956
957
void freerdp_channels_close(rdpChannels* channels, freerdp* instance)
958
0
{
959
0
  WINPR_ASSERT(channels);
960
0
  WINPR_ASSERT(instance);
961
962
0
  MessageQueue_PostQuit(channels->queue, 0);
963
0
  freerdp_channels_check_fds(channels, instance);
964
965
  /* tell all libraries we are shutting down */
966
0
  for (int index = 0; index < channels->clientDataCount; index++)
967
0
  {
968
0
    CHANNEL_CLIENT_DATA* pChannelClientData = &channels->clientDataList[index];
969
970
0
    if (pChannelClientData->pChannelInitEventProc)
971
0
    {
972
0
      pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
973
0
                                                CHANNEL_EVENT_TERMINATED, nullptr, 0);
974
0
    }
975
0
    else if (pChannelClientData->pChannelInitEventProcEx)
976
0
    {
977
0
      pChannelClientData->pChannelInitEventProcEx(pChannelClientData->lpUserParam,
978
0
                                                  pChannelClientData->pInitHandle,
979
0
                                                  CHANNEL_EVENT_TERMINATED, nullptr, 0);
980
0
    }
981
982
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[index];
983
0
    CHANNEL_INIT_DATA* pChannelInitData = &channels->initDataList[index];
984
0
    ChannelTerminatedEventArgs e = WINPR_C_ARRAY_INIT;
985
0
    EventArgsInit(&e, "freerdp");
986
0
    e.name = pChannelOpenData->name;
987
0
    e.pInterface = pChannelInitData->pInterface;
988
0
    if (PubSub_OnChannelTerminated(instance->context->pubSub, instance->context, &e) < 0)
989
0
    {
990
0
      WLog_WARN(TAG, "PubSub_OnChannelTerminated(%s) failed", pChannelOpenData->name);
991
0
    }
992
0
  }
993
994
0
  for (int index = 0; index < channels->openDataCount; index++)
995
0
  {
996
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[index];
997
0
    HashTable_Remove(g_ChannelHandles, (void*)(UINT_PTR)pChannelOpenData->OpenHandle);
998
0
  }
999
1000
0
  channels->openDataCount = 0;
1001
0
  channels->initDataCount = 0;
1002
0
  channels->clientDataCount = 0;
1003
1004
0
  WINPR_ASSERT(instance->context);
1005
0
  WINPR_ASSERT(instance->context->settings);
1006
0
  instance->context->settings->ChannelCount = 0;
1007
0
  g_Instance = nullptr;
1008
0
}
1009
1010
static UINT VCAPITYPE FreeRDP_VirtualChannelInitEx(
1011
    LPVOID lpUserParam, LPVOID clientContext, LPVOID pInitHandle, PCHANNEL_DEF pChannel,
1012
    INT channelCount, ULONG versionRequested, PCHANNEL_INIT_EVENT_EX_FN pChannelInitEventProcEx)
1013
0
{
1014
0
  rdpSettings* settings = nullptr;
1015
0
  CHANNEL_CLIENT_DATA* pChannelClientData = nullptr;
1016
1017
0
  if (!pInitHandle)
1018
0
    return CHANNEL_RC_BAD_INIT_HANDLE;
1019
1020
0
  if (!pChannel)
1021
0
    return CHANNEL_RC_BAD_CHANNEL;
1022
1023
0
  if ((channelCount <= 0) || !pChannelInitEventProcEx)
1024
0
    return CHANNEL_RC_INITIALIZATION_ERROR;
1025
1026
0
  CHANNEL_INIT_DATA* pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
1027
0
  WINPR_ASSERT(pChannelInitData);
1028
1029
0
  rdpChannels* channels = pChannelInitData->channels;
1030
0
  WINPR_ASSERT(channels);
1031
1032
0
  if (!channels->can_call_init)
1033
0
    return CHANNEL_RC_NOT_IN_VIRTUALCHANNELENTRY;
1034
1035
0
  if ((channels->openDataCount + channelCount) > CHANNEL_MAX_COUNT)
1036
0
    return CHANNEL_RC_TOO_MANY_CHANNELS;
1037
1038
0
  if (channels->connected)
1039
0
    return CHANNEL_RC_ALREADY_CONNECTED;
1040
1041
0
  if (versionRequested != VIRTUAL_CHANNEL_VERSION_WIN2000)
1042
0
  {
1043
0
  }
1044
1045
0
  for (int index = 0; index < channelCount; index++)
1046
0
  {
1047
0
    const PCHANNEL_DEF pChannelDef = &pChannel[index];
1048
1049
0
    if (freerdp_channels_find_channel_open_data_by_name(channels, pChannelDef->name) != nullptr)
1050
0
    {
1051
0
      return CHANNEL_RC_BAD_CHANNEL;
1052
0
    }
1053
0
  }
1054
1055
0
  pChannelInitData->pInterface = clientContext;
1056
0
  pChannelClientData = &channels->clientDataList[channels->clientDataCount];
1057
0
  pChannelClientData->pChannelInitEventProcEx = pChannelInitEventProcEx;
1058
0
  pChannelClientData->pInitHandle = pInitHandle;
1059
0
  pChannelClientData->lpUserParam = lpUserParam;
1060
0
  channels->clientDataCount++;
1061
1062
0
  WINPR_ASSERT(channels->instance);
1063
0
  WINPR_ASSERT(channels->instance->context);
1064
0
  settings = channels->instance->context->settings;
1065
0
  WINPR_ASSERT(settings);
1066
1067
0
  for (int index = 0; index < channelCount; index++)
1068
0
  {
1069
0
    const PCHANNEL_DEF pChannelDef = &pChannel[index];
1070
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[channels->openDataCount];
1071
1072
0
    WINPR_ASSERT(pChannelOpenData);
1073
1074
0
    const LONG hdl = InterlockedIncrement(&g_OpenHandleSeq);
1075
0
    pChannelOpenData->OpenHandle = WINPR_ASSERTING_INT_CAST(uint32_t, hdl);
1076
0
    pChannelOpenData->channels = channels;
1077
0
    pChannelOpenData->lpUserParam = lpUserParam;
1078
0
    if (!HashTable_Insert(g_ChannelHandles, (void*)(UINT_PTR)pChannelOpenData->OpenHandle,
1079
0
                          (void*)pChannelOpenData))
1080
0
    {
1081
0
      pChannelInitData->pInterface = nullptr;
1082
0
      return CHANNEL_RC_INITIALIZATION_ERROR;
1083
0
    }
1084
0
    pChannelOpenData->flags = 1; /* init */
1085
0
    strncpy(pChannelOpenData->name, pChannelDef->name, CHANNEL_NAME_LEN);
1086
0
    pChannelOpenData->options = pChannelDef->options;
1087
1088
0
    const UINT32 max = freerdp_settings_get_uint32(settings, FreeRDP_ChannelDefArraySize);
1089
0
    WINPR_ASSERT(max >= CHANNEL_MAX_COUNT);
1090
0
    if (settings->ChannelCount < max)
1091
0
    {
1092
0
      CHANNEL_DEF* channel = freerdp_settings_get_pointer_array_writable(
1093
0
          settings, FreeRDP_ChannelDefArray, settings->ChannelCount);
1094
0
      if (!channel)
1095
0
        continue;
1096
0
      strncpy(channel->name, pChannelDef->name, CHANNEL_NAME_LEN);
1097
0
      channel->options = pChannelDef->options;
1098
0
      settings->ChannelCount++;
1099
0
    }
1100
1101
0
    channels->openDataCount++;
1102
0
  }
1103
1104
0
  return CHANNEL_RC_OK;
1105
0
}
1106
1107
static UINT VCAPITYPE FreeRDP_VirtualChannelInit(LPVOID* ppInitHandle, PCHANNEL_DEF pChannel,
1108
                                                 INT channelCount, ULONG versionRequested,
1109
                                                 PCHANNEL_INIT_EVENT_FN pChannelInitEventProc)
1110
0
{
1111
0
  CHANNEL_DEF* channel = nullptr;
1112
0
  rdpSettings* settings = nullptr;
1113
0
  PCHANNEL_DEF pChannelDef = nullptr;
1114
0
  CHANNEL_INIT_DATA* pChannelInitData = nullptr;
1115
0
  CHANNEL_CLIENT_DATA* pChannelClientData = nullptr;
1116
0
  rdpChannels* channels = nullptr;
1117
1118
  /* g_Instance should have been set during freerdp_connect - otherwise VirtualChannelInit was
1119
   * called from a different thread */
1120
0
  if (!g_Instance || !g_Instance->context)
1121
0
    return CHANNEL_RC_NOT_INITIALIZED;
1122
1123
0
  channels = g_Instance->context->channels;
1124
1125
0
  if (!ppInitHandle || !channels)
1126
0
    return CHANNEL_RC_BAD_INIT_HANDLE;
1127
1128
0
  if (!pChannel)
1129
0
    return CHANNEL_RC_BAD_CHANNEL;
1130
1131
0
  if ((channelCount <= 0) || !pChannelInitEventProc)
1132
0
    return CHANNEL_RC_INITIALIZATION_ERROR;
1133
1134
0
  pChannelInitData = &(channels->initDataList[channels->initDataCount]);
1135
0
  *ppInitHandle = pChannelInitData;
1136
0
  channels->initDataCount++;
1137
0
  pChannelInitData->channels = channels;
1138
0
  pChannelInitData->pInterface = nullptr;
1139
1140
0
  if (!channels->can_call_init)
1141
0
    return CHANNEL_RC_NOT_IN_VIRTUALCHANNELENTRY;
1142
1143
0
  if (channels->openDataCount + channelCount > CHANNEL_MAX_COUNT)
1144
0
    return CHANNEL_RC_TOO_MANY_CHANNELS;
1145
1146
0
  if (channels->connected)
1147
0
    return CHANNEL_RC_ALREADY_CONNECTED;
1148
1149
0
  if (versionRequested != VIRTUAL_CHANNEL_VERSION_WIN2000)
1150
0
  {
1151
0
  }
1152
1153
0
  for (int index = 0; index < channelCount; index++)
1154
0
  {
1155
0
    pChannelDef = &pChannel[index];
1156
1157
0
    if (freerdp_channels_find_channel_open_data_by_name(channels, pChannelDef->name) != nullptr)
1158
0
    {
1159
0
      return CHANNEL_RC_BAD_CHANNEL;
1160
0
    }
1161
0
  }
1162
1163
0
  pChannelClientData = &channels->clientDataList[channels->clientDataCount];
1164
0
  pChannelClientData->pChannelInitEventProc = pChannelInitEventProc;
1165
0
  pChannelClientData->pInitHandle = *ppInitHandle;
1166
0
  channels->clientDataCount++;
1167
0
  settings = channels->instance->context->settings;
1168
1169
0
  for (int index = 0; index < channelCount; index++)
1170
0
  {
1171
0
    UINT32 ChannelCount = freerdp_settings_get_uint32(settings, FreeRDP_ChannelCount);
1172
1173
0
    pChannelDef = &pChannel[index];
1174
1175
0
    CHANNEL_OPEN_DATA* pChannelOpenData = &channels->openDataList[channels->openDataCount];
1176
1177
0
    const LONG hdl = InterlockedIncrement(&g_OpenHandleSeq);
1178
0
    pChannelOpenData->OpenHandle = WINPR_ASSERTING_INT_CAST(uint32_t, hdl);
1179
0
    pChannelOpenData->channels = channels;
1180
0
    if (!HashTable_Insert(g_ChannelHandles, (void*)(UINT_PTR)pChannelOpenData->OpenHandle,
1181
0
                          (void*)pChannelOpenData))
1182
0
      return CHANNEL_RC_INITIALIZATION_ERROR;
1183
0
    pChannelOpenData->flags = 1; /* init */
1184
0
    strncpy(pChannelOpenData->name, pChannelDef->name, CHANNEL_NAME_LEN);
1185
0
    pChannelOpenData->options = pChannelDef->options;
1186
1187
0
    if (ChannelCount < CHANNEL_MAX_COUNT)
1188
0
    {
1189
0
      channel = freerdp_settings_get_pointer_array_writable(settings, FreeRDP_ChannelDefArray,
1190
0
                                                            ChannelCount++);
1191
0
      strncpy(channel->name, pChannelDef->name, CHANNEL_NAME_LEN);
1192
0
      channel->options = pChannelDef->options;
1193
0
      if (!freerdp_settings_set_uint32(settings, FreeRDP_ChannelCount, ChannelCount))
1194
0
        return ERROR_INTERNAL_ERROR;
1195
0
    }
1196
1197
0
    channels->openDataCount++;
1198
0
  }
1199
1200
0
  return CHANNEL_RC_OK;
1201
0
}
1202
1203
static UINT VCAPITYPE
1204
FreeRDP_VirtualChannelOpenEx(LPVOID pInitHandle, LPDWORD pOpenHandle, PCHAR pChannelName,
1205
                             PCHANNEL_OPEN_EVENT_EX_FN pChannelOpenEventProcEx)
1206
0
{
1207
0
  CHANNEL_INIT_DATA* pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
1208
0
  rdpChannels* channels = pChannelInitData->channels;
1209
0
  void* pInterface = pChannelInitData->pInterface;
1210
1211
0
  if (!pOpenHandle)
1212
0
    return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1213
1214
0
  if (!pChannelOpenEventProcEx)
1215
0
    return CHANNEL_RC_BAD_PROC;
1216
1217
0
  if (!channels->connected)
1218
0
    return CHANNEL_RC_NOT_CONNECTED;
1219
1220
0
  CHANNEL_OPEN_DATA* pChannelOpenData =
1221
0
      freerdp_channels_find_channel_open_data_by_name(channels, pChannelName);
1222
1223
0
  if (!pChannelOpenData)
1224
0
    return CHANNEL_RC_UNKNOWN_CHANNEL_NAME;
1225
1226
0
  if (pChannelOpenData->flags == 2)
1227
0
    return CHANNEL_RC_ALREADY_OPEN;
1228
1229
0
  pChannelOpenData->flags = 2; /* open */
1230
0
  pChannelOpenData->pInterface = pInterface;
1231
0
  pChannelOpenData->pChannelOpenEventProcEx = pChannelOpenEventProcEx;
1232
0
  *pOpenHandle = pChannelOpenData->OpenHandle;
1233
0
  return CHANNEL_RC_OK;
1234
0
}
1235
1236
static UINT VCAPITYPE FreeRDP_VirtualChannelOpen(LPVOID pInitHandle, LPDWORD pOpenHandle,
1237
                                                 PCHAR pChannelName,
1238
                                                 PCHANNEL_OPEN_EVENT_FN pChannelOpenEventProc)
1239
0
{
1240
0
  void* pInterface = nullptr;
1241
0
  rdpChannels* channels = nullptr;
1242
0
  CHANNEL_INIT_DATA* pChannelInitData = nullptr;
1243
0
  CHANNEL_OPEN_DATA* pChannelOpenData = nullptr;
1244
0
  pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
1245
0
  channels = pChannelInitData->channels;
1246
0
  pInterface = pChannelInitData->pInterface;
1247
1248
0
  if (!pOpenHandle)
1249
0
    return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1250
1251
0
  if (!pChannelOpenEventProc)
1252
0
    return CHANNEL_RC_BAD_PROC;
1253
1254
0
  if (!channels->connected)
1255
0
    return CHANNEL_RC_NOT_CONNECTED;
1256
1257
0
  pChannelOpenData = freerdp_channels_find_channel_open_data_by_name(channels, pChannelName);
1258
1259
0
  if (!pChannelOpenData)
1260
0
    return CHANNEL_RC_UNKNOWN_CHANNEL_NAME;
1261
1262
0
  if (pChannelOpenData->flags == 2)
1263
0
    return CHANNEL_RC_ALREADY_OPEN;
1264
1265
0
  pChannelOpenData->flags = 2; /* open */
1266
0
  pChannelOpenData->pInterface = pInterface;
1267
0
  pChannelOpenData->pChannelOpenEventProc = pChannelOpenEventProc;
1268
0
  *pOpenHandle = pChannelOpenData->OpenHandle;
1269
0
  return CHANNEL_RC_OK;
1270
0
}
1271
1272
static UINT VCAPITYPE FreeRDP_VirtualChannelCloseEx(LPVOID pInitHandle, DWORD openHandle)
1273
0
{
1274
0
  CHANNEL_OPEN_DATA* pChannelOpenData = nullptr;
1275
1276
0
  if (!pInitHandle)
1277
0
    return CHANNEL_RC_BAD_INIT_HANDLE;
1278
1279
0
  pChannelOpenData = HashTable_GetItemValue(g_ChannelHandles, (void*)(UINT_PTR)openHandle);
1280
1281
0
  if (!pChannelOpenData)
1282
0
    return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1283
1284
0
  if (pChannelOpenData->flags != 2)
1285
0
    return CHANNEL_RC_NOT_OPEN;
1286
1287
0
  pChannelOpenData->flags = 0;
1288
0
  return CHANNEL_RC_OK;
1289
0
}
1290
1291
static UINT VCAPITYPE FreeRDP_VirtualChannelClose(DWORD openHandle)
1292
0
{
1293
0
  CHANNEL_OPEN_DATA* pChannelOpenData = nullptr;
1294
1295
0
  pChannelOpenData = HashTable_GetItemValue(g_ChannelHandles, (void*)(UINT_PTR)openHandle);
1296
1297
0
  if (!pChannelOpenData)
1298
0
    return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1299
1300
0
  if (pChannelOpenData->flags != 2)
1301
0
    return CHANNEL_RC_NOT_OPEN;
1302
1303
0
  pChannelOpenData->flags = 0;
1304
0
  return CHANNEL_RC_OK;
1305
0
}
1306
1307
static UINT VCAPITYPE FreeRDP_VirtualChannelWriteEx(LPVOID pInitHandle, DWORD openHandle,
1308
                                                    LPVOID pData, ULONG dataLength,
1309
                                                    LPVOID pUserData)
1310
0
{
1311
0
  rdpChannels* channels = nullptr;
1312
0
  CHANNEL_INIT_DATA* pChannelInitData = nullptr;
1313
0
  CHANNEL_OPEN_DATA* pChannelOpenData = nullptr;
1314
0
  CHANNEL_OPEN_EVENT* pChannelOpenEvent = nullptr;
1315
0
  wMessage message = WINPR_C_ARRAY_INIT;
1316
1317
0
  if (!pInitHandle)
1318
0
    return CHANNEL_RC_BAD_INIT_HANDLE;
1319
1320
0
  pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
1321
0
  channels = pChannelInitData->channels;
1322
1323
0
  if (!channels)
1324
0
    return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1325
1326
0
  pChannelOpenData = HashTable_GetItemValue(g_ChannelHandles, (void*)(UINT_PTR)openHandle);
1327
1328
0
  if (!pChannelOpenData)
1329
0
    return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1330
1331
0
  if (!channels->connected)
1332
0
    return CHANNEL_RC_NOT_CONNECTED;
1333
1334
0
  if (!pData)
1335
0
    return CHANNEL_RC_NULL_DATA;
1336
1337
0
  if (!dataLength)
1338
0
    return CHANNEL_RC_ZERO_LENGTH;
1339
1340
0
  if (pChannelOpenData->flags != 2)
1341
0
    return CHANNEL_RC_NOT_OPEN;
1342
1343
0
  pChannelOpenEvent = (CHANNEL_OPEN_EVENT*)malloc(sizeof(CHANNEL_OPEN_EVENT));
1344
1345
0
  if (!pChannelOpenEvent)
1346
0
    return CHANNEL_RC_NO_MEMORY;
1347
1348
0
  pChannelOpenEvent->Data = pData;
1349
0
  pChannelOpenEvent->DataLength = dataLength;
1350
0
  pChannelOpenEvent->UserData = pUserData;
1351
0
  pChannelOpenEvent->pChannelOpenData = pChannelOpenData;
1352
0
  message.context = channels;
1353
0
  message.id = 0;
1354
0
  message.wParam = pChannelOpenEvent;
1355
0
  message.lParam = nullptr;
1356
0
  message.Free = channel_queue_message_free;
1357
1358
0
  if (!MessageQueue_Dispatch(channels->queue, &message))
1359
0
  {
1360
0
    free(pChannelOpenEvent);
1361
0
    return CHANNEL_RC_NO_MEMORY;
1362
0
  }
1363
1364
0
  return CHANNEL_RC_OK;
1365
0
}
1366
1367
static UINT VCAPITYPE FreeRDP_VirtualChannelWrite(DWORD openHandle, LPVOID pData, ULONG dataLength,
1368
                                                  LPVOID pUserData)
1369
0
{
1370
0
  wMessage message = WINPR_C_ARRAY_INIT;
1371
0
  CHANNEL_OPEN_DATA* pChannelOpenData = nullptr;
1372
0
  CHANNEL_OPEN_EVENT* pChannelOpenEvent = nullptr;
1373
0
  rdpChannels* channels = nullptr;
1374
1375
0
  pChannelOpenData = HashTable_GetItemValue(g_ChannelHandles, (void*)(UINT_PTR)openHandle);
1376
1377
0
  if (!pChannelOpenData)
1378
0
    return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1379
1380
0
  channels = pChannelOpenData->channels;
1381
0
  if (!channels)
1382
0
    return CHANNEL_RC_BAD_CHANNEL_HANDLE;
1383
1384
0
  if (!channels->connected)
1385
0
    return CHANNEL_RC_NOT_CONNECTED;
1386
1387
0
  if (!pData)
1388
0
    return CHANNEL_RC_NULL_DATA;
1389
1390
0
  if (!dataLength)
1391
0
    return CHANNEL_RC_ZERO_LENGTH;
1392
1393
0
  if (pChannelOpenData->flags != 2)
1394
0
    return CHANNEL_RC_NOT_OPEN;
1395
1396
0
  pChannelOpenEvent = (CHANNEL_OPEN_EVENT*)malloc(sizeof(CHANNEL_OPEN_EVENT));
1397
1398
0
  if (!pChannelOpenEvent)
1399
0
    return CHANNEL_RC_NO_MEMORY;
1400
1401
0
  pChannelOpenEvent->Data = pData;
1402
0
  pChannelOpenEvent->DataLength = dataLength;
1403
0
  pChannelOpenEvent->UserData = pUserData;
1404
0
  pChannelOpenEvent->pChannelOpenData = pChannelOpenData;
1405
0
  message.context = channels;
1406
0
  message.id = 0;
1407
0
  message.wParam = pChannelOpenEvent;
1408
0
  message.lParam = nullptr;
1409
0
  message.Free = channel_queue_message_free;
1410
1411
0
  if (!MessageQueue_Dispatch(channels->queue, &message))
1412
0
  {
1413
0
    free(pChannelOpenEvent);
1414
0
    return CHANNEL_RC_NO_MEMORY;
1415
0
  }
1416
1417
0
  return CHANNEL_RC_OK;
1418
0
}
1419
1420
static BOOL freerdp_channels_is_loaded(rdpChannels* channels, PVIRTUALCHANNELENTRY entry)
1421
0
{
1422
0
  for (int i = 0; i < channels->clientDataCount; i++)
1423
0
  {
1424
0
    CHANNEL_CLIENT_DATA* pChannelClientData = &channels->clientDataList[i];
1425
1426
0
    if (pChannelClientData->entry == entry)
1427
0
      return TRUE;
1428
0
  }
1429
1430
0
  return FALSE;
1431
0
}
1432
1433
static BOOL freerdp_channels_is_loaded_ex(rdpChannels* channels, PVIRTUALCHANNELENTRYEX entryEx)
1434
0
{
1435
0
  for (int i = 0; i < channels->clientDataCount; i++)
1436
0
  {
1437
0
    CHANNEL_CLIENT_DATA* pChannelClientData = &channels->clientDataList[i];
1438
1439
0
    if (pChannelClientData->entryEx == entryEx)
1440
0
      return TRUE;
1441
0
  }
1442
1443
0
  return FALSE;
1444
0
}
1445
1446
int freerdp_channels_client_load(rdpChannels* channels, WINPR_ATTR_UNUSED rdpSettings* settings,
1447
                                 PVIRTUALCHANNELENTRY entry, void* data)
1448
0
{
1449
0
  CHANNEL_CLIENT_DATA* pChannelClientData = nullptr;
1450
1451
0
  WINPR_ASSERT(channels);
1452
0
  WINPR_ASSERT(channels->instance);
1453
0
  WINPR_ASSERT(channels->instance->context);
1454
0
  WINPR_ASSERT(entry);
1455
1456
0
  if (channels->clientDataCount + 1 > CHANNEL_MAX_COUNT)
1457
0
  {
1458
0
    WLog_ERR(TAG, "error: too many channels");
1459
0
    return 1;
1460
0
  }
1461
1462
0
  if (freerdp_channels_is_loaded(channels, entry))
1463
0
  {
1464
0
    WLog_WARN(TAG, "Skipping, channel already loaded");
1465
0
    return 0;
1466
0
  }
1467
1468
0
  pChannelClientData = &channels->clientDataList[channels->clientDataCount];
1469
0
  pChannelClientData->entry = entry;
1470
1471
0
  CHANNEL_ENTRY_POINTS_FREERDP EntryPoints = { .cbSize = sizeof(EntryPoints),
1472
0
                                             .protocolVersion = VIRTUAL_CHANNEL_VERSION_WIN2000,
1473
0
                                             .pVirtualChannelInit = FreeRDP_VirtualChannelInit,
1474
0
                                             .pVirtualChannelOpen = FreeRDP_VirtualChannelOpen,
1475
0
                                             .pVirtualChannelClose =
1476
0
                                                 FreeRDP_VirtualChannelClose,
1477
0
                                             .pVirtualChannelWrite =
1478
0
                                                 FreeRDP_VirtualChannelWrite,
1479
0
                                             .MagicNumber = FREERDP_CHANNEL_MAGIC_NUMBER,
1480
0
                                             .pExtendedData = data,
1481
0
                                             .context = channels->instance->context };
1482
  /* enable VirtualChannelInit */
1483
0
  union
1484
0
  {
1485
0
    PCHANNEL_ENTRY_POINTS_FREERDP pfx;
1486
0
    PCHANNEL_ENTRY_POINTS px;
1487
0
  } ptr;
1488
1489
0
  ptr.pfx = &EntryPoints;
1490
1491
0
  channels->can_call_init = TRUE;
1492
0
  EnterCriticalSection(&channels->channelsLock);
1493
0
  const BOOL status = pChannelClientData->entry(ptr.px);
1494
0
  LeaveCriticalSection(&channels->channelsLock);
1495
  /* disable MyVirtualChannelInit */
1496
0
  channels->can_call_init = FALSE;
1497
1498
0
  if (!status)
1499
0
  {
1500
0
    WLog_ERR(TAG, "error: channel export function call failed");
1501
0
    return 1;
1502
0
  }
1503
1504
0
  return 0;
1505
0
}
1506
1507
int freerdp_channels_client_load_ex(rdpChannels* channels, WINPR_ATTR_UNUSED rdpSettings* settings,
1508
                                    PVIRTUALCHANNELENTRYEX entryEx, void* data)
1509
0
{
1510
0
  WINPR_ASSERT(channels);
1511
0
  WINPR_ASSERT(channels->instance);
1512
0
  WINPR_ASSERT(channels->instance->context);
1513
0
  WINPR_ASSERT(entryEx);
1514
1515
0
  if (channels->clientDataCount + 1 > CHANNEL_MAX_COUNT)
1516
0
  {
1517
0
    WLog_ERR(TAG, "error: too many channels");
1518
0
    return 1;
1519
0
  }
1520
1521
0
  if (freerdp_channels_is_loaded_ex(channels, entryEx))
1522
0
  {
1523
0
    WLog_WARN(TAG, "Skipping, channel already loaded");
1524
0
    return 0;
1525
0
  }
1526
1527
0
  CHANNEL_CLIENT_DATA* pChannelClientData = &channels->clientDataList[channels->clientDataCount];
1528
0
  pChannelClientData->entryEx = entryEx;
1529
1530
0
  CHANNEL_INIT_DATA* pChannelInitData = &(channels->initDataList[channels->initDataCount++]);
1531
0
  void* pInitHandle = pChannelInitData;
1532
0
  pChannelInitData->channels = channels;
1533
1534
0
  CHANNEL_ENTRY_POINTS_FREERDP_EX EntryPointsEx = {
1535
0
    .cbSize = sizeof(EntryPointsEx),
1536
0
    .protocolVersion = VIRTUAL_CHANNEL_VERSION_WIN2000,
1537
0
    .pVirtualChannelInitEx = FreeRDP_VirtualChannelInitEx,
1538
0
    .pVirtualChannelOpenEx = FreeRDP_VirtualChannelOpenEx,
1539
0
    .pVirtualChannelCloseEx = FreeRDP_VirtualChannelCloseEx,
1540
0
    .pVirtualChannelWriteEx = FreeRDP_VirtualChannelWriteEx,
1541
0
    .MagicNumber = FREERDP_CHANNEL_MAGIC_NUMBER,
1542
0
    .pExtendedData = data,
1543
0
    .context = channels->instance->context
1544
0
  };
1545
1546
0
  union
1547
0
  {
1548
0
    PCHANNEL_ENTRY_POINTS_FREERDP_EX pfx;
1549
0
    PCHANNEL_ENTRY_POINTS_EX px;
1550
0
  } ptr;
1551
1552
0
  ptr.pfx = &EntryPointsEx;
1553
1554
  /* enable VirtualChannelInit */
1555
0
  channels->can_call_init = TRUE;
1556
0
  EnterCriticalSection(&channels->channelsLock);
1557
0
  const BOOL status = pChannelClientData->entryEx(ptr.px, pInitHandle);
1558
0
  LeaveCriticalSection(&channels->channelsLock);
1559
  /* disable MyVirtualChannelInit */
1560
0
  channels->can_call_init = FALSE;
1561
1562
0
  if (!status)
1563
0
  {
1564
0
    WLog_ERR(TAG, "error: channel export function call failed");
1565
0
    return 1;
1566
0
  }
1567
1568
0
  return 0;
1569
0
}
1570
1571
/**
1572
 * this is called when processing the command line parameters
1573
 * called only from main thread
1574
 */
1575
int freerdp_channels_load_plugin(rdpChannels* channels, rdpSettings* settings, const char* name,
1576
                                 void* data)
1577
0
{
1578
0
  PVIRTUALCHANNELENTRY entry =
1579
0
      freerdp_load_channel_addin_entry(name, nullptr, nullptr, FREERDP_ADDIN_CHANNEL_STATIC);
1580
1581
0
  if (!entry)
1582
0
    return 1;
1583
1584
0
  return freerdp_channels_client_load(channels, settings, entry, data);
1585
0
}