Coverage Report

Created: 2026-02-26 06:54

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/FreeRDP/channels/rdpdr/client/rdpdr_main.c
Line
Count
Source
1
/**
2
 * FreeRDP: A Remote Desktop Protocol Implementation
3
 * Device Redirection Virtual Channel
4
 *
5
 * Copyright 2010-2011 Vic Lee
6
 * Copyright 2010-2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
7
 * Copyright 2015-2016 Thincast Technologies GmbH
8
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
9
 * Copyright 2016 Armin Novak <armin.novak@thincast.com>
10
 * Copyright 2016 David PHAM-VAN <d.phamvan@inuvika.com>
11
 *
12
 * Licensed under the Apache License, Version 2.0 (the "License");
13
 * you may not use this file except in compliance with the License.
14
 * You may obtain a copy of the License at
15
 *
16
 *     http://www.apache.org/licenses/LICENSE-2.0
17
 *
18
 * Unless required by applicable law or agreed to in writing, software
19
 * distributed under the License is distributed on an "AS IS" BASIS,
20
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
21
 * See the License for the specific language governing permissions and
22
 * limitations under the License.
23
 */
24
25
#include <freerdp/config.h>
26
27
#include <stdio.h>
28
#include <stdlib.h>
29
#include <string.h>
30
#include <stdint.h>
31
32
#include <winpr/crt.h>
33
#include <winpr/sysinfo.h>
34
#include <winpr/assert.h>
35
#include <winpr/stream.h>
36
37
#include <winpr/print.h>
38
#include <winpr/sspicli.h>
39
40
#include <freerdp/types.h>
41
#include <freerdp/freerdp.h>
42
#include <freerdp/constants.h>
43
#include <freerdp/channels/log.h>
44
#include <freerdp/channels/rdpdr.h>
45
#include <freerdp/utils/rdpdr_utils.h>
46
47
#ifdef _WIN32
48
#include <windows.h>
49
#include <dbt.h>
50
#else
51
#include <sys/types.h>
52
#include <sys/stat.h>
53
#include <fcntl.h>
54
#endif
55
56
#ifdef __MACOSX__
57
#include <CoreFoundation/CoreFoundation.h>
58
#include <stdio.h>
59
#include <dirent.h>
60
#include <sys/types.h>
61
#include <sys/stat.h>
62
#include <unistd.h>
63
#endif
64
65
#include "rdpdr_capabilities.h"
66
67
#include "devman.h"
68
#include "irp.h"
69
70
#include "rdpdr_main.h"
71
72
0
#define TAG CHANNELS_TAG("rdpdr.client")
73
74
/* IMPORTANT: Keep in sync with DRIVE_DEVICE */
75
typedef struct
76
{
77
  DEVICE device;
78
  WCHAR* path;
79
  BOOL automount;
80
} DEVICE_DRIVE_EXT;
81
82
static const char* rdpdr_state_str(enum RDPDR_CHANNEL_STATE state)
83
0
{
84
0
  switch (state)
85
0
  {
86
0
    case RDPDR_CHANNEL_STATE_INITIAL:
87
0
      return "RDPDR_CHANNEL_STATE_INITIAL";
88
0
    case RDPDR_CHANNEL_STATE_ANNOUNCE:
89
0
      return "RDPDR_CHANNEL_STATE_ANNOUNCE";
90
0
    case RDPDR_CHANNEL_STATE_ANNOUNCE_REPLY:
91
0
      return "RDPDR_CHANNEL_STATE_ANNOUNCE_REPLY";
92
0
    case RDPDR_CHANNEL_STATE_NAME_REQUEST:
93
0
      return "RDPDR_CHANNEL_STATE_NAME_REQUEST";
94
0
    case RDPDR_CHANNEL_STATE_SERVER_CAPS:
95
0
      return "RDPDR_CHANNEL_STATE_SERVER_CAPS";
96
0
    case RDPDR_CHANNEL_STATE_CLIENT_CAPS:
97
0
      return "RDPDR_CHANNEL_STATE_CLIENT_CAPS";
98
0
    case RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM:
99
0
      return "RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM";
100
0
    case RDPDR_CHANNEL_STATE_READY:
101
0
      return "RDPDR_CHANNEL_STATE_READY";
102
0
    case RDPDR_CHANNEL_STATE_USER_LOGGEDON:
103
0
      return "RDPDR_CHANNEL_STATE_USER_LOGGEDON";
104
0
    default:
105
0
      return "RDPDR_CHANNEL_STATE_UNKNOWN";
106
0
  }
107
0
}
108
109
static const char* support_str(BOOL val)
110
0
{
111
0
  if (val)
112
0
    return "supported";
113
0
  return "not found";
114
0
}
115
116
static const char* rdpdr_caps_pdu_str(UINT32 flag)
117
0
{
118
0
  switch (flag)
119
0
  {
120
0
    case RDPDR_DEVICE_REMOVE_PDUS:
121
0
      return "RDPDR_USER_LOGGEDON_PDU";
122
0
    case RDPDR_CLIENT_DISPLAY_NAME_PDU:
123
0
      return "RDPDR_CLIENT_DISPLAY_NAME_PDU";
124
0
    case RDPDR_USER_LOGGEDON_PDU:
125
0
      return "RDPDR_USER_LOGGEDON_PDU";
126
0
    default:
127
0
      return "RDPDR_UNKNONW";
128
0
  }
129
0
}
130
131
static BOOL rdpdr_check_extended_pdu_flag(rdpdrPlugin* rdpdr, UINT32 flag)
132
0
{
133
0
  WINPR_ASSERT(rdpdr);
134
135
0
  const BOOL client = (rdpdr->clientExtendedPDU & flag) != 0;
136
0
  const BOOL server = (rdpdr->serverExtendedPDU & flag) != 0;
137
138
0
  if (!client || !server)
139
0
  {
140
0
    WLog_Print(rdpdr->log, WLOG_WARN, "Checking ExtendedPDU::%s, client %s, server %s",
141
0
               rdpdr_caps_pdu_str(flag), support_str(client), support_str(server));
142
0
    return FALSE;
143
0
  }
144
0
  return TRUE;
145
0
}
146
147
BOOL rdpdr_state_advance(rdpdrPlugin* rdpdr, enum RDPDR_CHANNEL_STATE next)
148
0
{
149
0
  WINPR_ASSERT(rdpdr);
150
151
0
  if (next != rdpdr->state)
152
0
    WLog_Print(rdpdr->log, WLOG_DEBUG, "[RDPDR] transition from %s to %s",
153
0
               rdpdr_state_str(rdpdr->state), rdpdr_state_str(next));
154
0
  rdpdr->state = next;
155
0
  return TRUE;
156
0
}
157
158
static BOOL device_foreach(rdpdrPlugin* rdpdr, BOOL abortOnFail,
159
                           BOOL (*fkt)(ULONG_PTR key, void* element, void* data), void* data)
160
0
{
161
0
  BOOL rc = TRUE;
162
0
  ULONG_PTR* keys = NULL;
163
164
0
  ListDictionary_Lock(rdpdr->devman->devices);
165
0
  const size_t count = ListDictionary_GetKeys(rdpdr->devman->devices, &keys);
166
0
  for (size_t x = 0; x < count; x++)
167
0
  {
168
0
    void* element = ListDictionary_GetItemValue(rdpdr->devman->devices, (void*)keys[x]);
169
0
    if (!fkt(keys[x], element, data))
170
0
    {
171
0
      rc = FALSE;
172
0
      if (abortOnFail)
173
0
        break;
174
0
    }
175
0
  }
176
0
  free(keys);
177
0
  ListDictionary_Unlock(rdpdr->devman->devices);
178
0
  return rc;
179
0
}
180
181
/**
182
 * Function description
183
 *
184
 * @return 0 on success, otherwise a Win32 error code
185
 */
186
static UINT rdpdr_try_send_device_list_announce_request(rdpdrPlugin* rdpdr);
187
188
static BOOL rdpdr_load_drive(rdpdrPlugin* rdpdr, const char* name, const char* path, BOOL automount)
189
0
{
190
0
  UINT rc = ERROR_INTERNAL_ERROR;
191
0
  union
192
0
  {
193
0
    RDPDR_DRIVE* drive;
194
0
    RDPDR_DEVICE* device;
195
0
  } drive;
196
0
  const char* args[] = { name, path, automount ? NULL : name };
197
198
0
  drive.device = freerdp_device_new(RDPDR_DTYP_FILESYSTEM, ARRAYSIZE(args), args);
199
0
  if (!drive.device)
200
0
    goto fail;
201
202
0
  WINPR_ASSERT(rdpdr->context.RdpdrRegisterDevice);
203
0
  rc = rdpdr->context.RdpdrRegisterDevice(&rdpdr->context, drive.device, &drive.device->Id);
204
0
  if (rc != CHANNEL_RC_OK)
205
0
    goto fail;
206
207
0
fail:
208
0
  freerdp_device_free(drive.device);
209
0
  return rc == CHANNEL_RC_OK;
210
0
}
211
212
/**
213
 * Function description
214
 *
215
 * @return 0 on success, otherwise a Win32 error code
216
 */
217
static UINT rdpdr_send_device_list_remove_request(rdpdrPlugin* rdpdr, UINT32 count,
218
                                                  const UINT32 ids[])
219
0
{
220
0
  wStream* s = NULL;
221
222
0
  WINPR_ASSERT(rdpdr);
223
0
  WINPR_ASSERT(ids || (count == 0));
224
225
0
  if (count == 0)
226
0
    return CHANNEL_RC_OK;
227
228
0
  if (!rdpdr_check_extended_pdu_flag(rdpdr, RDPDR_DEVICE_REMOVE_PDUS))
229
0
    return CHANNEL_RC_OK;
230
231
0
  s = StreamPool_Take(rdpdr->pool, count * sizeof(UINT32) + 8);
232
233
0
  if (!s)
234
0
  {
235
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "Stream_New failed!");
236
0
    return CHANNEL_RC_NO_MEMORY;
237
0
  }
238
239
0
  Stream_Write_UINT16(s, RDPDR_CTYP_CORE);
240
0
  Stream_Write_UINT16(s, PAKID_CORE_DEVICELIST_REMOVE);
241
0
  Stream_Write_UINT32(s, count);
242
243
0
  for (UINT32 i = 0; i < count; i++)
244
0
    Stream_Write_UINT32(s, ids[i]);
245
246
0
  Stream_SealLength(s);
247
0
  return rdpdr_send(rdpdr, s);
248
0
}
249
250
#if defined(_UWP) || defined(__IOS__)
251
252
static UINT handle_hotplug(WINPR_ATTR_UNUSED RdpdrClientContext* context,
253
                           WINPR_ATTR_UNUSED RdpdrHotplugEventType type)
254
{
255
  return ERROR_CALL_NOT_IMPLEMENTED;
256
}
257
258
static void first_hotplug(WINPR_ATTR_UNUSED rdpdrPlugin* rdpdr)
259
{
260
}
261
262
static DWORD WINAPI drive_hotplug_thread_func(WINPR_ATTR_UNUSED LPVOID arg)
263
{
264
  return CHANNEL_RC_OK;
265
}
266
267
static UINT drive_hotplug_thread_terminate(WINPR_ATTR_UNUSED rdpdrPlugin* rdpdr)
268
{
269
  return CHANNEL_RC_OK;
270
}
271
272
#elif defined(_WIN32)
273
274
static UINT handle_hotplug(WINPR_ATTR_UNUSED RdpdrClientContext* context,
275
                           WINPR_ATTR_UNUSED RdpdrHotplugEventType type)
276
{
277
  return CHANNEL_RC_OK;
278
}
279
280
static BOOL check_path(const char* path)
281
{
282
  UINT type = GetDriveTypeA(path);
283
284
  if (!(type == DRIVE_FIXED || type == DRIVE_REMOVABLE || type == DRIVE_CDROM ||
285
        type == DRIVE_REMOTE))
286
    return FALSE;
287
288
  return GetVolumeInformationA(path, NULL, 0, NULL, NULL, NULL, NULL, 0);
289
}
290
291
static void first_hotplug(rdpdrPlugin* rdpdr)
292
{
293
  DWORD unitmask = GetLogicalDrives();
294
295
  for (size_t i = 0; i < 26; i++)
296
  {
297
    if (unitmask & 0x01)
298
    {
299
      char drive_path[] = { 'c', ':', '\\', '\0' };
300
      char drive_name[] = { 'c', '\0' };
301
      drive_path[0] = 'A' + (char)i;
302
      drive_name[0] = 'A' + (char)i;
303
304
      if (check_path(drive_path))
305
      {
306
        rdpdr_load_drive(rdpdr, drive_name, drive_path, TRUE);
307
      }
308
    }
309
310
    unitmask = unitmask >> 1;
311
  }
312
}
313
314
static LRESULT CALLBACK hotplug_proc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
315
{
316
  rdpdrPlugin* rdpdr;
317
  PDEV_BROADCAST_HDR lpdb = (PDEV_BROADCAST_HDR)lParam;
318
  UINT error;
319
  rdpdr = (rdpdrPlugin*)GetWindowLongPtr(hWnd, GWLP_USERDATA);
320
321
  switch (Msg)
322
  {
323
    case WM_DEVICECHANGE:
324
      switch (wParam)
325
      {
326
        case DBT_DEVICEARRIVAL:
327
          if (lpdb->dbch_devicetype == DBT_DEVTYP_VOLUME)
328
          {
329
            PDEV_BROADCAST_VOLUME lpdbv = (PDEV_BROADCAST_VOLUME)lpdb;
330
            DWORD unitmask = lpdbv->dbcv_unitmask;
331
332
            for (int i = 0; i < 26; i++)
333
            {
334
              if (unitmask & 0x01)
335
              {
336
                char drive_path[] = { 'c', ':', '/', '\0' };
337
                char drive_name[] = { 'c', '\0' };
338
                drive_path[0] = 'A' + (char)i;
339
                drive_name[0] = 'A' + (char)i;
340
341
                if (check_path(drive_path))
342
                {
343
                  rdpdr_load_drive(rdpdr, drive_name, drive_path, TRUE);
344
                }
345
              }
346
347
              unitmask = unitmask >> 1;
348
            }
349
          }
350
351
          break;
352
353
        case DBT_DEVICEREMOVECOMPLETE:
354
          if (lpdb->dbch_devicetype == DBT_DEVTYP_VOLUME)
355
          {
356
            PDEV_BROADCAST_VOLUME lpdbv = (PDEV_BROADCAST_VOLUME)lpdb;
357
            DWORD unitmask = lpdbv->dbcv_unitmask;
358
            char drive_name_upper, drive_name_lower;
359
            ULONG_PTR* keys = NULL;
360
            DEVICE_DRIVE_EXT* device_ext;
361
362
            for (int i = 0; i < 26; i++)
363
            {
364
              if (unitmask & 0x01)
365
              {
366
                drive_name_upper = 'A' + i;
367
                drive_name_lower = 'a' + i;
368
                const size_t count =
369
                    ListDictionary_GetKeys(rdpdr->devman->devices, &keys);
370
371
                for (size_t j = 0; j < count; j++)
372
                {
373
                  device_ext = (DEVICE_DRIVE_EXT*)ListDictionary_GetItemValue(
374
                      rdpdr->devman->devices, (void*)keys[j]);
375
376
                  if (device_ext->device.type != RDPDR_DTYP_FILESYSTEM)
377
                    continue;
378
379
                  if (device_ext->path[0] == drive_name_upper ||
380
                      device_ext->path[0] == drive_name_lower)
381
                  {
382
                    if (device_ext->automount)
383
                    {
384
                      const uint32_t ids[] = { keys[j] };
385
                      WINPR_ASSERT(rdpdr->context.RdpdrUnregisterDevice);
386
                      error = rdpdr->context.RdpdrUnregisterDevice(
387
                          &rdpdr->context, ARRAYSIZE(ids), ids);
388
                      if (error)
389
                      {
390
                        // don't end on error, just report ?
391
                        WLog_Print(
392
                            rdpdr->log, WLOG_ERROR,
393
                            "rdpdr_send_device_list_remove_request failed "
394
                            "with error %" PRIu32 "!",
395
                            error);
396
                      }
397
398
                      break;
399
                    }
400
                  }
401
                }
402
403
                free(keys);
404
              }
405
406
              unitmask = unitmask >> 1;
407
            }
408
          }
409
410
          break;
411
412
        default:
413
          break;
414
      }
415
416
      break;
417
418
    default:
419
      return DefWindowProc(hWnd, Msg, wParam, lParam);
420
  }
421
422
  return DefWindowProc(hWnd, Msg, wParam, lParam);
423
}
424
425
static DWORD WINAPI drive_hotplug_thread_func(LPVOID arg)
426
{
427
  rdpdrPlugin* rdpdr;
428
  WNDCLASSEX wnd_cls;
429
  HWND hwnd;
430
  MSG msg;
431
  BOOL bRet;
432
  DEV_BROADCAST_HANDLE NotificationFilter;
433
  HDEVNOTIFY hDevNotify;
434
  rdpdr = (rdpdrPlugin*)arg;
435
  /* init windows class */
436
  wnd_cls.cbSize = sizeof(WNDCLASSEX);
437
  wnd_cls.style = CS_HREDRAW | CS_VREDRAW;
438
  wnd_cls.lpfnWndProc = hotplug_proc;
439
  wnd_cls.cbClsExtra = 0;
440
  wnd_cls.cbWndExtra = 0;
441
  wnd_cls.hIcon = LoadIcon(NULL, IDI_APPLICATION);
442
  wnd_cls.hCursor = NULL;
443
  wnd_cls.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
444
  wnd_cls.lpszMenuName = NULL;
445
  wnd_cls.lpszClassName = L"DRIVE_HOTPLUG";
446
  wnd_cls.hInstance = NULL;
447
  wnd_cls.hIconSm = LoadIcon(NULL, IDI_APPLICATION);
448
  RegisterClassEx(&wnd_cls);
449
  /* create window */
450
  hwnd = CreateWindowEx(0, L"DRIVE_HOTPLUG", NULL, 0, 0, 0, 0, 0, NULL, NULL, NULL, NULL);
451
  SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)rdpdr);
452
  rdpdr->hotplug_wnd = hwnd;
453
  /* register device interface to hwnd */
454
  NotificationFilter.dbch_size = sizeof(DEV_BROADCAST_HANDLE);
455
  NotificationFilter.dbch_devicetype = DBT_DEVTYP_HANDLE;
456
  hDevNotify = RegisterDeviceNotification(hwnd, &NotificationFilter, DEVICE_NOTIFY_WINDOW_HANDLE);
457
458
  /* message loop */
459
  while ((bRet = GetMessage(&msg, 0, 0, 0)) != 0)
460
  {
461
    if (bRet == -1)
462
    {
463
      break;
464
    }
465
    else
466
    {
467
      TranslateMessage(&msg);
468
      DispatchMessage(&msg);
469
    }
470
  }
471
472
  UnregisterDeviceNotification(hDevNotify);
473
  return CHANNEL_RC_OK;
474
}
475
476
/**
477
 * Function description
478
 *
479
 * @return 0 on success, otherwise a Win32 error code
480
 */
481
static UINT drive_hotplug_thread_terminate(rdpdrPlugin* rdpdr)
482
{
483
  UINT error = CHANNEL_RC_OK;
484
485
  if (rdpdr->hotplug_wnd && !PostMessage(rdpdr->hotplug_wnd, WM_QUIT, 0, 0))
486
  {
487
    error = GetLastError();
488
    WLog_Print(rdpdr->log, WLOG_ERROR, "PostMessage failed with error %" PRIu32 "", error);
489
  }
490
491
  return error;
492
}
493
494
#elif defined(__MACOSX__)
495
496
#define MAX_USB_DEVICES 100
497
498
typedef struct
499
{
500
  char* path;
501
  BOOL to_add;
502
} hotplug_dev;
503
504
/**
505
 * Function description
506
 *
507
 * @return 0 on success, otherwise a Win32 error code
508
 */
509
static UINT handle_hotplug(WINPR_ATTR_UNUSED RdpdrClientContext* context,
510
                           WINPR_ATTR_UNUSED RdpdrHotplugEventType type)
511
{
512
  WINPR_ASSERT(context);
513
  rdpdrPlugin* rdpdr = context->handle;
514
515
  struct dirent* pDirent = NULL;
516
  char fullpath[PATH_MAX] = WINPR_C_ARRAY_INIT;
517
  char* szdir = (char*)"/Volumes";
518
  struct stat buf = WINPR_C_ARRAY_INIT;
519
  hotplug_dev dev_array[MAX_USB_DEVICES] = WINPR_C_ARRAY_INIT;
520
  int count = 0;
521
  DEVICE_DRIVE_EXT* device_ext = NULL;
522
  ULONG_PTR* keys = NULL;
523
  int size = 0;
524
  UINT error = ERROR_INTERNAL_ERROR;
525
  UINT32 ids[1];
526
527
  DIR* pDir = opendir(szdir);
528
529
  if (pDir == NULL)
530
  {
531
    printf("Cannot open directory\n");
532
    return ERROR_OPEN_FAILED;
533
  }
534
535
  while ((pDirent = readdir(pDir)) != NULL)
536
  {
537
    if (pDirent->d_name[0] != '.')
538
    {
539
      (void)sprintf_s(fullpath, ARRAYSIZE(fullpath), "%s/%s", szdir, pDirent->d_name);
540
      if (stat(fullpath, &buf) != 0)
541
        continue;
542
543
      if (S_ISDIR(buf.st_mode))
544
      {
545
        dev_array[size].path = _strdup(fullpath);
546
547
        if (!dev_array[size].path)
548
        {
549
          closedir(pDir);
550
          error = CHANNEL_RC_NO_MEMORY;
551
          goto cleanup;
552
        }
553
554
        dev_array[size++].to_add = TRUE;
555
      }
556
    }
557
  }
558
559
  closedir(pDir);
560
  /* delete removed devices */
561
  count = ListDictionary_GetKeys(rdpdr->devman->devices, &keys);
562
563
  for (size_t j = 0; j < count; j++)
564
  {
565
    char* path = NULL;
566
    BOOL dev_found = FALSE;
567
    device_ext =
568
        (DEVICE_DRIVE_EXT*)ListDictionary_GetItemValue(rdpdr->devman->devices, (void*)keys[j]);
569
570
    if (!device_ext || !device_ext->automount)
571
      continue;
572
573
    if (device_ext->device.type != RDPDR_DTYP_FILESYSTEM)
574
      continue;
575
576
    if (device_ext->path == NULL)
577
      continue;
578
579
    path = ConvertWCharToUtf8Alloc(device_ext->path, NULL);
580
    if (!path)
581
      continue;
582
583
    /* not pluggable device */
584
    if (strstr(path, "/Volumes/") == NULL)
585
    {
586
      free(path);
587
      continue;
588
    }
589
590
    for (size_t i = 0; i < size; i++)
591
    {
592
      if (strstr(path, dev_array[i].path) != NULL)
593
      {
594
        dev_found = TRUE;
595
        dev_array[i].to_add = FALSE;
596
        break;
597
      }
598
    }
599
600
    free(path);
601
602
    if (!dev_found)
603
    {
604
      const uint32_t ids[] = { keys[j] };
605
      WINPR_ASSERT(rdpdr->context.RdpdrUnregisterDevice);
606
      error = rdpdr->context.RdpdrUnregisterDevice(&rdpdr->context, ARRAYSIZE(ids), ids);
607
      if (error)
608
      {
609
        WLog_Print(rdpdr->log, WLOG_ERROR,
610
                   "rdpdr_send_device_list_remove_request failed with error %" PRIu32 "!",
611
                   error);
612
        goto cleanup;
613
      }
614
    }
615
  }
616
617
  /* add new devices */
618
  for (size_t i = 0; i < size; i++)
619
  {
620
    const hotplug_dev* dev = &dev_array[i];
621
    if (dev->to_add)
622
    {
623
      const char* path = dev->path;
624
      const char* name = strrchr(path, '/') + 1;
625
      error = rdpdr_load_drive(rdpdr, name, path, TRUE);
626
      if (error)
627
        goto cleanup;
628
    }
629
  }
630
631
cleanup:
632
  free(keys);
633
634
  for (size_t i = 0; i < size; i++)
635
    free(dev_array[i].path);
636
637
  return error;
638
}
639
640
static void drive_hotplug_fsevent_callback(ConstFSEventStreamRef streamRef,
641
                                           void* clientCallBackInfo, size_t numEvents,
642
                                           void* eventPaths,
643
                                           const FSEventStreamEventFlags eventFlags[],
644
                                           const FSEventStreamEventId eventIds[])
645
{
646
  rdpdrPlugin* rdpdr;
647
  UINT error;
648
  char** paths = (char**)eventPaths;
649
  rdpdr = (rdpdrPlugin*)clientCallBackInfo;
650
651
  for (size_t i = 0; i < numEvents; i++)
652
  {
653
    if (strcmp(paths[i], "/Volumes/") == 0)
654
    {
655
      UINT error = ERROR_CALL_NOT_IMPLEMENTED;
656
      if (rdpdr->context.RdpdrHotplugDevice)
657
        error = rdpdr->context.RdpdrHotplugDevice(&rdpdr->context,
658
                                                  RDPDR_HOTPLUG_CHECK_FOR_CHANGES);
659
      switch (error)
660
      {
661
        case ERROR_DISK_CHANGE:
662
        case CHANNEL_RC_OK:
663
          break;
664
        case ERROR_CALL_NOT_IMPLEMENTED:
665
          break;
666
        default:
667
          WLog_Print(rdpdr->log, WLOG_ERROR,
668
                     "handle_hotplug failed with error %" PRIu32 "!", error);
669
          break;
670
      }
671
    }
672
  }
673
}
674
675
static void first_hotplug(rdpdrPlugin* rdpdr)
676
{
677
  WINPR_ASSERT(rdpdr);
678
  UINT error = ERROR_CALL_NOT_IMPLEMENTED;
679
  if (rdpdr->context.RdpdrHotplugDevice)
680
    error = rdpdr->context.RdpdrHotplugDevice(&rdpdr->context, RDPDR_HOTPLUG_FIRST_CHECK);
681
682
  switch (error)
683
  {
684
    case ERROR_DISK_CHANGE:
685
    case CHANNEL_RC_OK:
686
    case ERROR_CALL_NOT_IMPLEMENTED:
687
      break;
688
    default:
689
      WLog_Print(rdpdr->log, WLOG_ERROR, "handle_hotplug failed with error %" PRIu32 "!",
690
                 error);
691
      break;
692
  }
693
}
694
695
static DWORD WINAPI drive_hotplug_thread_func(LPVOID arg)
696
{
697
  rdpdrPlugin* rdpdr = (rdpdrPlugin*)arg;
698
  WINPR_ASSERT(rdpdr);
699
  WINPR_ASSERT(rdpdr->stopEvent);
700
701
  CFStringRef path = CFSTR("/Volumes/");
702
  CFArrayRef pathsToWatch = CFArrayCreate(kCFAllocatorMalloc, (const void**)&path, 1, NULL);
703
  FSEventStreamContext ctx = {
704
    .copyDescription = NULL, .info = arg, .release = NULL, .retain = NULL, .version = 0
705
  };
706
  FSEventStreamRef fsev =
707
      FSEventStreamCreate(kCFAllocatorMalloc, drive_hotplug_fsevent_callback, &ctx, pathsToWatch,
708
                          kFSEventStreamEventIdSinceNow, 1, kFSEventStreamCreateFlagNone);
709
710
  dispatch_queue_t queue = dispatch_queue_create(TAG, NULL);
711
  FSEventStreamSetDispatchQueue(fsev, queue);
712
  FSEventStreamStart(fsev);
713
  WLog_Print(rdpdr->log, WLOG_DEBUG, "Started hotplug watcher");
714
  HANDLE handles[] = { rdpdr->stopEvent, freerdp_abort_event(rdpdr->rdpcontext) };
715
  WaitForMultipleObjects(ARRAYSIZE(handles), handles, FALSE, INFINITE);
716
  WLog_Print(rdpdr->log, WLOG_DEBUG, "Stopped hotplug watcher");
717
  FSEventStreamStop(fsev);
718
  FSEventStreamRelease(fsev);
719
  dispatch_release(queue);
720
out:
721
  ExitThread(CHANNEL_RC_OK);
722
  return CHANNEL_RC_OK;
723
}
724
725
#else
726
727
static const char* automountLocations[] = { "/run/user/%lu/gvfs", "/run/media/%s", "/media/%s",
728
                                          "/media", "/mnt" };
729
730
static BOOL isAutomountLocation(const char* path)
731
0
{
732
0
  const size_t nrLocations = sizeof(automountLocations) / sizeof(automountLocations[0]);
733
0
  char buffer[MAX_PATH] = WINPR_C_ARRAY_INIT;
734
0
  uid_t uid = getuid();
735
0
  char uname[MAX_PATH] = WINPR_C_ARRAY_INIT;
736
0
  ULONG size = sizeof(uname) - 1;
737
738
0
  if (!GetUserNameExA(NameSamCompatible, uname, &size))
739
0
    return FALSE;
740
741
0
  if (!path)
742
0
    return FALSE;
743
744
0
  for (size_t x = 0; x < nrLocations; x++)
745
0
  {
746
0
    const char* location = automountLocations[x];
747
0
    size_t length = 0;
748
749
0
    WINPR_PRAGMA_DIAG_PUSH
750
0
    WINPR_PRAGMA_DIAG_IGNORED_FORMAT_NONLITERAL
751
0
    if (strstr(location, "%lu"))
752
0
      (void)snprintf(buffer, sizeof(buffer), location, (unsigned long)uid);
753
0
    else if (strstr(location, "%s"))
754
0
      (void)snprintf(buffer, sizeof(buffer), location, uname);
755
0
    else
756
0
      (void)snprintf(buffer, sizeof(buffer), "%s", location);
757
0
    WINPR_PRAGMA_DIAG_POP
758
759
0
    length = strnlen(buffer, sizeof(buffer));
760
761
0
    if (strncmp(buffer, path, length) == 0)
762
0
    {
763
0
      const char* rest = &path[length];
764
765
      /* Only consider mount locations with max depth of 1 below the
766
       * base path or the base path itself. */
767
0
      if (*rest == '\0')
768
0
        return TRUE;
769
0
      else if (*rest == '/')
770
0
      {
771
0
        const char* token = strstr(&rest[1], "/");
772
773
0
        if (!token || (token[1] == '\0'))
774
0
          return TRUE;
775
0
      }
776
0
    }
777
0
  }
778
779
0
  return FALSE;
780
0
}
781
782
0
#define MAX_USB_DEVICES 100
783
784
typedef struct
785
{
786
  char* path;
787
  BOOL to_add;
788
} hotplug_dev;
789
790
static void handle_mountpoint(hotplug_dev* dev_array, size_t* size, const char* mountpoint)
791
0
{
792
0
  if (!mountpoint)
793
0
    return;
794
  /* copy hotpluged device mount point to the dev_array */
795
0
  if (isAutomountLocation(mountpoint) && (*size < MAX_USB_DEVICES))
796
0
  {
797
0
    dev_array[*size].path = _strdup(mountpoint);
798
0
    dev_array[*size].to_add = TRUE;
799
0
    (*size)++;
800
0
  }
801
0
}
802
803
#ifdef __sun
804
#include <sys/mnttab.h>
805
static UINT handle_platform_mounts_sun(wLog* log, hotplug_dev* dev_array, size_t* size)
806
{
807
  FILE* f;
808
  struct mnttab ent;
809
  f = winpr_fopen("/etc/mnttab", "r");
810
  if (f == NULL)
811
  {
812
    WLog_Print(log, WLOG_ERROR, "fopen failed!");
813
    return ERROR_OPEN_FAILED;
814
  }
815
  while (getmntent(f, &ent) == 0)
816
  {
817
    handle_mountpoint(dev_array, size, ent.mnt_mountp);
818
  }
819
  fclose(f);
820
  return ERROR_SUCCESS;
821
}
822
#endif
823
824
#if defined(__FreeBSD__) || defined(__OpenBSD__)
825
#include <sys/mount.h>
826
static UINT handle_platform_mounts_bsd(wLog* log, hotplug_dev* dev_array, size_t* size)
827
{
828
  int mntsize;
829
  struct statfs* mntbuf = NULL;
830
831
  mntsize = getmntinfo(&mntbuf, MNT_NOWAIT);
832
  if (!mntsize)
833
  {
834
    /* TODO: handle 'errno' */
835
    WLog_Print(log, WLOG_ERROR, "getmntinfo failed!");
836
    return ERROR_OPEN_FAILED;
837
  }
838
  for (size_t idx = 0; idx < (size_t)mntsize; idx++)
839
  {
840
    handle_mountpoint(dev_array, size, mntbuf[idx].f_mntonname);
841
  }
842
  free(mntbuf);
843
  return ERROR_SUCCESS;
844
}
845
#endif
846
847
#if defined(__LINUX__) || defined(__linux__)
848
#include <mntent.h>
849
static struct mntent* getmntent_x(FILE* f, struct mntent* buffer, char* pathbuffer,
850
                                  size_t pathbuffersize)
851
0
{
852
0
#if defined(FREERDP_HAVE_GETMNTENT_R)
853
0
  WINPR_ASSERT(pathbuffersize <= INT32_MAX);
854
0
  return getmntent_r(f, buffer, pathbuffer, (int)pathbuffersize);
855
#else
856
  (void)buffer;
857
  (void)pathbuffer;
858
  (void)pathbuffersize;
859
  return getmntent(f);
860
#endif
861
0
}
862
863
static UINT handle_platform_mounts_linux(wLog* log, hotplug_dev* dev_array, size_t* size)
864
0
{
865
0
  FILE* f = NULL;
866
0
  struct mntent mnt = WINPR_C_ARRAY_INIT;
867
0
  char pathbuffer[PATH_MAX] = WINPR_C_ARRAY_INIT;
868
0
  struct mntent* ent = NULL;
869
0
  f = winpr_fopen("/proc/mounts", "r");
870
0
  if (f == NULL)
871
0
  {
872
0
    WLog_Print(log, WLOG_ERROR, "fopen failed!");
873
0
    return ERROR_OPEN_FAILED;
874
0
  }
875
0
  while ((ent = getmntent_x(f, &mnt, pathbuffer, sizeof(pathbuffer))) != NULL)
876
0
  {
877
0
    handle_mountpoint(dev_array, size, ent->mnt_dir);
878
0
  }
879
0
  (void)fclose(f);
880
0
  return ERROR_SUCCESS;
881
0
}
882
#endif
883
884
static UINT handle_platform_mounts(wLog* log, hotplug_dev* dev_array, size_t* size)
885
0
{
886
#ifdef __sun
887
  return handle_platform_mounts_sun(log, dev_array, size);
888
#elif defined(__FreeBSD__) || defined(__OpenBSD__)
889
  return handle_platform_mounts_bsd(log, dev_array, size);
890
#elif defined(__LINUX__) || defined(__linux__)
891
  return handle_platform_mounts_linux(log, dev_array, size);
892
0
#endif
893
0
  return ERROR_CALL_NOT_IMPLEMENTED;
894
0
}
895
896
static BOOL device_not_plugged(ULONG_PTR key, void* element, void* data)
897
0
{
898
0
  const WCHAR* path = (const WCHAR*)data;
899
0
  DEVICE_DRIVE_EXT* device_ext = (DEVICE_DRIVE_EXT*)element;
900
901
0
  WINPR_UNUSED(key);
902
0
  WINPR_ASSERT(path);
903
904
0
  if (!device_ext || (device_ext->device.type != RDPDR_DTYP_FILESYSTEM) || !device_ext->path)
905
0
    return TRUE;
906
0
  if (_wcscmp(device_ext->path, path) != 0)
907
0
    return TRUE;
908
0
  return FALSE;
909
0
}
910
911
static BOOL device_already_plugged(rdpdrPlugin* rdpdr, const hotplug_dev* device)
912
0
{
913
0
  BOOL rc = FALSE;
914
0
  WCHAR* path = NULL;
915
916
0
  if (!rdpdr || !device)
917
0
    return TRUE;
918
0
  if (!device->to_add)
919
0
    return TRUE;
920
921
0
  WINPR_ASSERT(rdpdr->devman);
922
0
  WINPR_ASSERT(device->path);
923
924
0
  path = ConvertUtf8ToWCharAlloc(device->path, NULL);
925
0
  if (!path)
926
0
    return TRUE;
927
928
0
  rc = device_foreach(rdpdr, TRUE, device_not_plugged, path);
929
0
  free(path);
930
0
  return !rc;
931
0
}
932
933
struct hotplug_delete_arg
934
{
935
  hotplug_dev* dev_array;
936
  size_t dev_array_size;
937
  rdpdrPlugin* rdpdr;
938
};
939
940
static BOOL hotplug_delete_foreach(ULONG_PTR key, void* element, void* data)
941
0
{
942
0
  char* path = NULL;
943
0
  BOOL dev_found = FALSE;
944
0
  struct hotplug_delete_arg* arg = (struct hotplug_delete_arg*)data;
945
0
  DEVICE_DRIVE_EXT* device_ext = (DEVICE_DRIVE_EXT*)element;
946
947
0
  WINPR_ASSERT(arg);
948
0
  WINPR_ASSERT(arg->rdpdr);
949
0
  WINPR_ASSERT(arg->dev_array || (arg->dev_array_size == 0));
950
0
  WINPR_ASSERT(key <= UINT32_MAX);
951
952
0
  if (!device_ext || (device_ext->device.type != RDPDR_DTYP_FILESYSTEM) || !device_ext->path ||
953
0
      !device_ext->automount)
954
0
    return TRUE;
955
956
0
  WINPR_ASSERT(device_ext->path);
957
0
  path = ConvertWCharToUtf8Alloc(device_ext->path, NULL);
958
0
  if (!path)
959
0
    return FALSE;
960
961
  /* not pluggable device */
962
0
  if (isAutomountLocation(path))
963
0
  {
964
0
    for (size_t i = 0; i < arg->dev_array_size; i++)
965
0
    {
966
0
      hotplug_dev* cur = &arg->dev_array[i];
967
0
      if (cur->path && strstr(path, cur->path) != NULL)
968
0
      {
969
0
        dev_found = TRUE;
970
0
        cur->to_add = FALSE;
971
0
        break;
972
0
      }
973
0
    }
974
0
  }
975
976
0
  free(path);
977
978
0
  if (!dev_found)
979
0
  {
980
0
    const UINT32 ids[1] = { (UINT32)key };
981
0
    WINPR_ASSERT(arg->rdpdr->context.RdpdrUnregisterDevice);
982
0
    const UINT error =
983
0
        arg->rdpdr->context.RdpdrUnregisterDevice(&arg->rdpdr->context, ARRAYSIZE(ids), ids);
984
985
0
    if (error)
986
0
    {
987
0
      WLog_Print(arg->rdpdr->log, WLOG_ERROR,
988
0
                 "rdpdr_send_device_list_remove_request failed with error %" PRIu32 "!",
989
0
                 error);
990
0
      return FALSE;
991
0
    }
992
0
  }
993
994
0
  return TRUE;
995
0
}
996
997
static UINT handle_hotplug(RdpdrClientContext* context,
998
                           WINPR_ATTR_UNUSED RdpdrHotplugEventType type)
999
0
{
1000
0
  WINPR_ASSERT(context);
1001
0
  rdpdrPlugin* rdpdr = context->handle;
1002
1003
0
  hotplug_dev dev_array[MAX_USB_DEVICES] = WINPR_C_ARRAY_INIT;
1004
0
  size_t size = 0;
1005
0
  UINT error = ERROR_SUCCESS;
1006
0
  struct hotplug_delete_arg arg = { dev_array, ARRAYSIZE(dev_array), rdpdr };
1007
1008
0
  WINPR_ASSERT(rdpdr);
1009
0
  WINPR_ASSERT(rdpdr->devman);
1010
1011
0
  error = handle_platform_mounts(rdpdr->log, dev_array, &size);
1012
1013
  /* delete removed devices */
1014
0
  /* Ignore result */ device_foreach(rdpdr, FALSE, hotplug_delete_foreach, &arg);
1015
1016
  /* add new devices */
1017
0
  for (size_t i = 0; i < size; i++)
1018
0
  {
1019
0
    hotplug_dev* cur = &dev_array[i];
1020
0
    if (!device_already_plugged(rdpdr, cur))
1021
0
    {
1022
0
      const char* path = cur->path;
1023
0
      const char* name = strrchr(path, '/') + 1;
1024
1025
0
      rdpdr_load_drive(rdpdr, name, path, TRUE);
1026
0
      error = ERROR_DISK_CHANGE;
1027
0
    }
1028
0
  }
1029
1030
0
  for (size_t i = 0; i < size; i++)
1031
0
    free(dev_array[i].path);
1032
1033
0
  return error;
1034
0
}
1035
1036
static void first_hotplug(rdpdrPlugin* rdpdr)
1037
0
{
1038
0
  UINT error = ERROR_CALL_NOT_IMPLEMENTED;
1039
1040
0
  WINPR_ASSERT(rdpdr);
1041
0
  if (rdpdr->context.RdpdrHotplugDevice)
1042
0
    error = rdpdr->context.RdpdrHotplugDevice(&rdpdr->context, RDPDR_HOTPLUG_FIRST_CHECK);
1043
1044
0
  switch (error)
1045
0
  {
1046
0
    case ERROR_DISK_CHANGE:
1047
0
    case CHANNEL_RC_OK:
1048
0
    case ERROR_OPEN_FAILED:
1049
0
    case ERROR_CALL_NOT_IMPLEMENTED:
1050
0
      break;
1051
0
    default:
1052
0
      WLog_Print(rdpdr->log, WLOG_ERROR, "handle_hotplug failed with error %" PRIu32 "!",
1053
0
                 error);
1054
0
      break;
1055
0
  }
1056
0
}
1057
1058
static DWORD WINAPI drive_hotplug_thread_func(LPVOID arg)
1059
0
{
1060
0
  rdpdrPlugin* rdpdr = (rdpdrPlugin*)arg;
1061
1062
0
  WINPR_ASSERT(rdpdr);
1063
0
  WINPR_ASSERT(rdpdr->stopEvent);
1064
1065
0
  while (WaitForSingleObject(rdpdr->stopEvent, 1000) == WAIT_TIMEOUT)
1066
0
  {
1067
0
    UINT error = ERROR_CALL_NOT_IMPLEMENTED;
1068
0
    if (rdpdr->context.RdpdrHotplugDevice)
1069
0
      error =
1070
0
          rdpdr->context.RdpdrHotplugDevice(&rdpdr->context, RDPDR_HOTPLUG_CHECK_FOR_CHANGES);
1071
0
    switch (error)
1072
0
    {
1073
0
      case ERROR_DISK_CHANGE:
1074
0
        break;
1075
0
      case CHANNEL_RC_OK:
1076
0
      case ERROR_OPEN_FAILED:
1077
0
      case ERROR_CALL_NOT_IMPLEMENTED:
1078
0
        break;
1079
0
      default:
1080
0
        WLog_Print(rdpdr->log, WLOG_ERROR, "handle_hotplug failed with error %" PRIu32 "!",
1081
0
                   error);
1082
0
        goto out;
1083
0
    }
1084
0
  }
1085
1086
0
out:
1087
0
{
1088
0
  const UINT error = GetLastError();
1089
0
  if (error && rdpdr->rdpcontext)
1090
0
    setChannelError(rdpdr->rdpcontext, error, "reported an error");
1091
1092
0
  ExitThread(error);
1093
0
  return error;
1094
0
}
1095
0
}
1096
1097
#endif
1098
1099
#if !defined(_WIN32) && !defined(__IOS__)
1100
/**
1101
 * Function description
1102
 *
1103
 * @return 0 on success, otherwise a Win32 error code
1104
 */
1105
static UINT drive_hotplug_thread_terminate(rdpdrPlugin* rdpdr)
1106
0
{
1107
0
  UINT error = 0;
1108
1109
0
  WINPR_ASSERT(rdpdr);
1110
1111
0
  if (rdpdr->hotplugThread)
1112
0
  {
1113
0
#if !defined(_WIN32)
1114
0
    if (rdpdr->stopEvent)
1115
0
      (void)SetEvent(rdpdr->stopEvent);
1116
0
#endif
1117
1118
0
    if (WaitForSingleObject(rdpdr->hotplugThread, INFINITE) == WAIT_FAILED)
1119
0
    {
1120
0
      error = GetLastError();
1121
0
      WLog_Print(rdpdr->log, WLOG_ERROR, "WaitForSingleObject failed with error %" PRIu32 "!",
1122
0
                 error);
1123
0
      return error;
1124
0
    }
1125
1126
0
    (void)CloseHandle(rdpdr->hotplugThread);
1127
0
    rdpdr->hotplugThread = NULL;
1128
0
  }
1129
1130
0
  return CHANNEL_RC_OK;
1131
0
}
1132
1133
#endif
1134
1135
static UINT rdpdr_add_devices(rdpdrPlugin* rdpdr)
1136
0
{
1137
0
  WINPR_ASSERT(rdpdr);
1138
0
  WINPR_ASSERT(rdpdr->rdpcontext);
1139
1140
0
  rdpSettings* settings = rdpdr->rdpcontext->settings;
1141
0
  WINPR_ASSERT(settings);
1142
1143
0
  for (UINT32 index = 0; index < freerdp_settings_get_uint32(settings, FreeRDP_DeviceCount);
1144
0
       index++)
1145
0
  {
1146
0
    RDPDR_DEVICE* device =
1147
0
        freerdp_settings_get_pointer_array_writable(settings, FreeRDP_DeviceArray, index);
1148
0
    WINPR_ASSERT(device);
1149
1150
0
    if (device->Type == RDPDR_DTYP_FILESYSTEM)
1151
0
    {
1152
0
      const char DynamicDrives[] = "DynamicDrives";
1153
0
      const RDPDR_DRIVE* drive = (const RDPDR_DRIVE*)device;
1154
0
      if (!drive->Path)
1155
0
        continue;
1156
1157
0
      const char wildcard[] = "*";
1158
0
      BOOL hotplugAll = strncmp(drive->Path, wildcard, sizeof(wildcard)) == 0;
1159
0
      BOOL hotplugLater = strncmp(drive->Path, DynamicDrives, sizeof(DynamicDrives)) == 0;
1160
1161
0
      if (hotplugAll || hotplugLater)
1162
0
      {
1163
0
        if (!rdpdr->async)
1164
0
        {
1165
0
          WLog_Print(rdpdr->log, WLOG_WARN,
1166
0
                     "Drive hotplug is not supported in synchronous mode!");
1167
0
          continue;
1168
0
        }
1169
1170
0
        if (hotplugAll)
1171
0
          first_hotplug(rdpdr);
1172
1173
        /* There might be multiple hotplug related device entries.
1174
         * Ensure the thread is only started once
1175
         */
1176
0
        if (!rdpdr->hotplugThread)
1177
0
        {
1178
0
          rdpdr->hotplugThread =
1179
0
              CreateThread(NULL, 0, drive_hotplug_thread_func, rdpdr, 0, NULL);
1180
0
          if (!rdpdr->hotplugThread)
1181
0
          {
1182
0
            WLog_Print(rdpdr->log, WLOG_ERROR, "CreateThread failed!");
1183
0
            return ERROR_INTERNAL_ERROR;
1184
0
          }
1185
0
        }
1186
1187
0
        continue;
1188
0
      }
1189
0
    }
1190
1191
0
    const UINT error = devman_load_device_service(rdpdr->devman, device, rdpdr->rdpcontext);
1192
0
    if (error)
1193
0
    {
1194
0
      WLog_Print(rdpdr->log, WLOG_ERROR,
1195
0
                 "devman_load_device_service failed with error %" PRIu32 "!", error);
1196
0
      return error;
1197
0
    }
1198
0
  }
1199
0
  return CHANNEL_RC_OK;
1200
0
}
1201
1202
/**
1203
 * Function description
1204
 *
1205
 * @return 0 on success, otherwise a Win32 error code
1206
 */
1207
static UINT rdpdr_process_connect(rdpdrPlugin* rdpdr)
1208
0
{
1209
0
  WINPR_ASSERT(rdpdr);
1210
1211
0
  rdpdr->devman = devman_new(rdpdr);
1212
1213
0
  if (!rdpdr->devman)
1214
0
  {
1215
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "devman_new failed!");
1216
0
    return CHANNEL_RC_NO_MEMORY;
1217
0
  }
1218
1219
0
  WINPR_ASSERT(rdpdr->rdpcontext);
1220
1221
0
  rdpSettings* settings = rdpdr->rdpcontext->settings;
1222
0
  WINPR_ASSERT(settings);
1223
1224
0
  rdpdr->ignoreInvalidDevices = freerdp_settings_get_bool(settings, FreeRDP_IgnoreInvalidDevices);
1225
1226
0
  const char* name = freerdp_settings_get_string(settings, FreeRDP_ClientHostname);
1227
0
  if (!name)
1228
0
    name = freerdp_settings_get_string(settings, FreeRDP_ComputerName);
1229
0
  if (!name)
1230
0
  {
1231
0
    DWORD size = ARRAYSIZE(rdpdr->computerName);
1232
0
    if (!GetComputerNameExA(ComputerNameNetBIOS, rdpdr->computerName, &size))
1233
0
      return ERROR_INTERNAL_ERROR;
1234
0
  }
1235
0
  else
1236
0
    strncpy(rdpdr->computerName, name, strnlen(name, sizeof(rdpdr->computerName)));
1237
1238
0
  return rdpdr_add_devices(rdpdr);
1239
0
}
1240
1241
static UINT rdpdr_process_server_announce_request(rdpdrPlugin* rdpdr, wStream* s)
1242
0
{
1243
0
  WINPR_ASSERT(rdpdr);
1244
0
  WINPR_ASSERT(s);
1245
1246
0
  if (!Stream_CheckAndLogRequiredLengthWLog(rdpdr->log, s, 8))
1247
0
    return ERROR_INVALID_DATA;
1248
1249
0
  Stream_Read_UINT16(s, rdpdr->serverVersionMajor);
1250
0
  Stream_Read_UINT16(s, rdpdr->serverVersionMinor);
1251
0
  Stream_Read_UINT32(s, rdpdr->clientID);
1252
0
  rdpdr->sequenceId++;
1253
1254
0
  rdpdr->clientVersionMajor = MIN(RDPDR_VERSION_MAJOR, rdpdr->serverVersionMajor);
1255
0
  rdpdr->clientVersionMinor = MIN(RDPDR_VERSION_MINOR_RDP10X, rdpdr->serverVersionMinor);
1256
0
  WLog_Print(rdpdr->log, WLOG_DEBUG,
1257
0
             "[rdpdr] server announces version %" PRIu32 ".%" PRIu32 ", client uses %" PRIu32
1258
0
             ".%" PRIu32,
1259
0
             rdpdr->serverVersionMajor, rdpdr->serverVersionMinor, rdpdr->clientVersionMajor,
1260
0
             rdpdr->clientVersionMinor);
1261
0
  return CHANNEL_RC_OK;
1262
0
}
1263
1264
/**
1265
 * Function description
1266
 *
1267
 * @return 0 on success, otherwise a Win32 error code
1268
 */
1269
static UINT rdpdr_send_client_announce_reply(rdpdrPlugin* rdpdr)
1270
0
{
1271
0
  WINPR_ASSERT(rdpdr);
1272
0
  WINPR_ASSERT(rdpdr->state == RDPDR_CHANNEL_STATE_ANNOUNCE);
1273
0
  if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_ANNOUNCE_REPLY))
1274
0
    return ERROR_INVALID_STATE;
1275
1276
0
  wStream* s = StreamPool_Take(rdpdr->pool, 12);
1277
1278
0
  if (!s)
1279
0
  {
1280
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "Stream_New failed!");
1281
0
    return CHANNEL_RC_NO_MEMORY;
1282
0
  }
1283
1284
0
  Stream_Write_UINT16(s, RDPDR_CTYP_CORE);             /* Component (2 bytes) */
1285
0
  Stream_Write_UINT16(s, PAKID_CORE_CLIENTID_CONFIRM); /* PacketId (2 bytes) */
1286
0
  Stream_Write_UINT16(s, rdpdr->clientVersionMajor);
1287
0
  Stream_Write_UINT16(s, rdpdr->clientVersionMinor);
1288
0
  Stream_Write_UINT32(s, rdpdr->clientID);
1289
0
  return rdpdr_send(rdpdr, s);
1290
0
}
1291
1292
/**
1293
 * Function description
1294
 *
1295
 * @return 0 on success, otherwise a Win32 error code
1296
 */
1297
static UINT rdpdr_send_client_name_request(rdpdrPlugin* rdpdr)
1298
0
{
1299
0
  wStream* s = NULL;
1300
0
  WCHAR* computerNameW = NULL;
1301
0
  size_t computerNameLenW = 0;
1302
1303
0
  WINPR_ASSERT(rdpdr);
1304
0
  WINPR_ASSERT(rdpdr->state == RDPDR_CHANNEL_STATE_ANNOUNCE_REPLY);
1305
0
  if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_NAME_REQUEST))
1306
0
    return ERROR_INVALID_STATE;
1307
1308
0
  const size_t len = strnlen(rdpdr->computerName, sizeof(rdpdr->computerName));
1309
0
  if (len == 0)
1310
0
    return ERROR_INTERNAL_ERROR;
1311
1312
0
  WINPR_ASSERT(rdpdr->computerName);
1313
0
  computerNameW = ConvertUtf8NToWCharAlloc(rdpdr->computerName, len, &computerNameLenW);
1314
0
  computerNameLenW *= sizeof(WCHAR);
1315
1316
0
  if (computerNameLenW > 0)
1317
0
    computerNameLenW += sizeof(WCHAR); // also write '\0'
1318
1319
0
  s = StreamPool_Take(rdpdr->pool, 16U + computerNameLenW);
1320
1321
0
  if (!s)
1322
0
  {
1323
0
    free(computerNameW);
1324
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "Stream_New failed!");
1325
0
    return CHANNEL_RC_NO_MEMORY;
1326
0
  }
1327
1328
0
  Stream_Write_UINT16(s, RDPDR_CTYP_CORE);        /* Component (2 bytes) */
1329
0
  Stream_Write_UINT16(s, PAKID_CORE_CLIENT_NAME); /* PacketId (2 bytes) */
1330
0
  Stream_Write_UINT32(s, 1);                      /* unicodeFlag, 0 for ASCII and 1 for Unicode */
1331
0
  Stream_Write_UINT32(s, 0);                      /* codePage, must be set to zero */
1332
0
  Stream_Write_UINT32(s,
1333
0
                      (UINT32)computerNameLenW); /* computerNameLen, including null terminator */
1334
0
  Stream_Write(s, computerNameW, computerNameLenW);
1335
0
  free(computerNameW);
1336
0
  return rdpdr_send(rdpdr, s);
1337
0
}
1338
1339
static UINT rdpdr_process_server_clientid_confirm(rdpdrPlugin* rdpdr, wStream* s)
1340
0
{
1341
0
  UINT16 versionMajor = 0;
1342
0
  UINT16 versionMinor = 0;
1343
0
  UINT32 clientID = 0;
1344
1345
0
  WINPR_ASSERT(rdpdr);
1346
0
  WINPR_ASSERT(s);
1347
1348
0
  if (!Stream_CheckAndLogRequiredLengthWLog(rdpdr->log, s, 8))
1349
0
    return ERROR_INVALID_DATA;
1350
1351
0
  Stream_Read_UINT16(s, versionMajor);
1352
0
  Stream_Read_UINT16(s, versionMinor);
1353
0
  Stream_Read_UINT32(s, clientID);
1354
1355
0
  if (versionMajor != rdpdr->clientVersionMajor || versionMinor != rdpdr->clientVersionMinor)
1356
0
  {
1357
0
    WLog_Print(rdpdr->log, WLOG_WARN,
1358
0
               "[rdpdr] server announced version %" PRIu32 ".%" PRIu32 ", client uses %" PRIu32
1359
0
               ".%" PRIu32 " but clientid confirm requests version %" PRIu32 ".%" PRIu32,
1360
0
               rdpdr->serverVersionMajor, rdpdr->serverVersionMinor, rdpdr->clientVersionMajor,
1361
0
               rdpdr->clientVersionMinor, versionMajor, versionMinor);
1362
0
    rdpdr->clientVersionMajor = versionMajor;
1363
0
    rdpdr->clientVersionMinor = versionMinor;
1364
0
  }
1365
1366
0
  if (clientID != rdpdr->clientID)
1367
0
    rdpdr->clientID = clientID;
1368
1369
0
  return CHANNEL_RC_OK;
1370
0
}
1371
1372
struct device_announce_arg
1373
{
1374
  rdpdrPlugin* rdpdr;
1375
  wStream* s;
1376
  BOOL userLoggedOn;
1377
  UINT32 count;
1378
};
1379
1380
static BOOL device_announce(ULONG_PTR key, void* element, void* data)
1381
0
{
1382
0
  struct device_announce_arg* arg = data;
1383
0
  rdpdrPlugin* rdpdr = NULL;
1384
0
  DEVICE* device = (DEVICE*)element;
1385
1386
0
  WINPR_UNUSED(key);
1387
1388
0
  WINPR_ASSERT(arg);
1389
0
  WINPR_ASSERT(device);
1390
0
  WINPR_ASSERT(arg->rdpdr);
1391
0
  WINPR_ASSERT(arg->s);
1392
1393
0
  rdpdr = arg->rdpdr;
1394
1395
  /**
1396
   * 1. versionMinor 0x0005 doesn't send PAKID_CORE_USER_LOGGEDON
1397
   *    so all devices should be sent regardless of user_loggedon
1398
   * 2. smartcard devices should be always sent
1399
   * 3. other devices are sent only after user_loggedon
1400
   */
1401
1402
0
  if ((rdpdr->clientVersionMinor == RDPDR_VERSION_MINOR_RDP51) ||
1403
0
      (device->type == RDPDR_DTYP_SMARTCARD) || arg->userLoggedOn)
1404
0
  {
1405
0
    size_t data_len = (device->data == NULL ? 0 : Stream_GetPosition(device->data));
1406
1407
0
    if (!Stream_EnsureRemainingCapacity(arg->s, 20 + data_len))
1408
0
    {
1409
0
      Stream_Release(arg->s);
1410
0
      WLog_Print(rdpdr->log, WLOG_ERROR, "Stream_EnsureRemainingCapacity failed!");
1411
0
      return FALSE;
1412
0
    }
1413
1414
0
    Stream_Write_UINT32(arg->s, device->type); /* deviceType */
1415
0
    Stream_Write_UINT32(arg->s, device->id);   /* deviceID */
1416
0
    strncpy(Stream_Pointer(arg->s), device->name, 8);
1417
1418
0
    for (size_t i = 0; i < 8; i++)
1419
0
    {
1420
0
      BYTE c = 0;
1421
0
      Stream_Peek_UINT8(arg->s, c);
1422
1423
0
      if (c > 0x7F)
1424
0
        Stream_Write_UINT8(arg->s, '_');
1425
0
      else
1426
0
        Stream_Seek_UINT8(arg->s);
1427
0
    }
1428
1429
0
    WINPR_ASSERT(data_len <= UINT32_MAX);
1430
0
    Stream_Write_UINT32(arg->s, (UINT32)data_len);
1431
1432
0
    if (data_len > 0)
1433
0
      Stream_Write(arg->s, Stream_Buffer(device->data), data_len);
1434
1435
0
    arg->count++;
1436
0
    WLog_Print(rdpdr->log, WLOG_INFO,
1437
0
               "registered [%9s] device #%" PRIu32 ": %5s (type=%2" PRIu32 " id=%2" PRIu32 ")",
1438
0
               rdpdr_device_type_string(device->type), arg->count, device->name, device->type,
1439
0
               device->id);
1440
0
  }
1441
0
  return TRUE;
1442
0
}
1443
1444
static UINT rdpdr_send_device_list_announce_request(rdpdrPlugin* rdpdr, BOOL userLoggedOn)
1445
0
{
1446
0
  size_t pos = 0;
1447
0
  wStream* s = NULL;
1448
0
  size_t count_pos = 0;
1449
0
  struct device_announce_arg arg = WINPR_C_ARRAY_INIT;
1450
1451
0
  WINPR_ASSERT(rdpdr);
1452
0
  WINPR_ASSERT(rdpdr->devman);
1453
1454
0
  if (userLoggedOn)
1455
0
  {
1456
0
    rdpdr->userLoggedOn = TRUE;
1457
0
  }
1458
1459
0
  s = StreamPool_Take(rdpdr->pool, 256);
1460
1461
0
  if (!s)
1462
0
  {
1463
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "Stream_New failed!");
1464
0
    return CHANNEL_RC_NO_MEMORY;
1465
0
  }
1466
1467
0
  Stream_Write_UINT16(s, RDPDR_CTYP_CORE);                /* Component (2 bytes) */
1468
0
  Stream_Write_UINT16(s, PAKID_CORE_DEVICELIST_ANNOUNCE); /* PacketId (2 bytes) */
1469
0
  count_pos = Stream_GetPosition(s);
1470
0
  Stream_Seek_UINT32(s); /* deviceCount */
1471
1472
0
  arg.rdpdr = rdpdr;
1473
0
  arg.userLoggedOn = userLoggedOn;
1474
0
  arg.s = s;
1475
0
  if (!device_foreach(rdpdr, TRUE, device_announce, &arg))
1476
0
    return ERROR_INVALID_DATA;
1477
1478
0
  if (arg.count == 0)
1479
0
  {
1480
0
    Stream_Release(s);
1481
0
    return CHANNEL_RC_OK;
1482
0
  }
1483
0
  pos = Stream_GetPosition(s);
1484
0
  Stream_SetPosition(s, count_pos);
1485
0
  Stream_Write_UINT32(s, arg.count);
1486
0
  Stream_SetPosition(s, pos);
1487
0
  Stream_SealLength(s);
1488
0
  return rdpdr_send(rdpdr, s);
1489
0
}
1490
1491
UINT rdpdr_try_send_device_list_announce_request(rdpdrPlugin* rdpdr)
1492
0
{
1493
0
  WINPR_ASSERT(rdpdr);
1494
0
  if (rdpdr->state != RDPDR_CHANNEL_STATE_READY)
1495
0
  {
1496
0
    WLog_Print(rdpdr->log, WLOG_DEBUG,
1497
0
               "hotplug event received, but channel [RDPDR] is not ready (state %s), ignoring.",
1498
0
               rdpdr_state_str(rdpdr->state));
1499
0
    return CHANNEL_RC_OK;
1500
0
  }
1501
0
  return rdpdr_send_device_list_announce_request(rdpdr, rdpdr->userLoggedOn);
1502
0
}
1503
1504
static UINT dummy_irp_response(rdpdrPlugin* rdpdr, wStream* s)
1505
0
{
1506
0
  WINPR_ASSERT(rdpdr);
1507
0
  WINPR_ASSERT(s);
1508
1509
0
  wStream* output = StreamPool_Take(rdpdr->pool, 256); // RDPDR_DEVICE_IO_RESPONSE_LENGTH
1510
0
  if (!output)
1511
0
  {
1512
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "Stream_New failed!");
1513
0
    return CHANNEL_RC_NO_MEMORY;
1514
0
  }
1515
1516
0
  Stream_SetPosition(s, 4); /* see "rdpdr_process_receive" */
1517
1518
0
  const uint32_t DeviceId = Stream_Get_UINT32(s);     /* DeviceId (4 bytes) */
1519
0
  const uint32_t FileId = Stream_Get_UINT32(s);       /* FileId (4 bytes) */
1520
0
  const uint32_t CompletionId = Stream_Get_UINT32(s); /* CompletionId (4 bytes) */
1521
1522
0
  WLog_Print(rdpdr->log, WLOG_WARN,
1523
0
             "Dummy response {DeviceId=%" PRIu32 ", FileId=%" PRIu32 ", CompletionId=%" PRIu32
1524
0
             "}",
1525
0
             DeviceId, FileId, CompletionId);
1526
0
  if (!rdpdr_write_iocompletion_header(output, DeviceId, CompletionId, STATUS_UNSUCCESSFUL))
1527
0
    return CHANNEL_RC_NO_MEMORY;
1528
1529
0
  return rdpdr_send(rdpdr, output);
1530
0
}
1531
1532
/**
1533
 * Function description
1534
 *
1535
 * @return 0 on success, otherwise a Win32 error code
1536
 */
1537
static UINT rdpdr_process_irp(rdpdrPlugin* rdpdr, wStream* s)
1538
0
{
1539
0
  UINT error = CHANNEL_RC_OK;
1540
1541
0
  WINPR_ASSERT(rdpdr);
1542
0
  WINPR_ASSERT(s);
1543
1544
0
  IRP* irp = irp_new(rdpdr->devman, rdpdr->pool, s, rdpdr->log, &error);
1545
1546
0
  if (!irp)
1547
0
  {
1548
0
    if ((error == CHANNEL_RC_OK) ||
1549
0
        (error == ERROR_DEV_NOT_EXIST && rdpdr->ignoreInvalidDevices))
1550
0
    {
1551
0
      return dummy_irp_response(rdpdr, s);
1552
0
    }
1553
1554
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "irp_new failed with %" PRIu32 "!", error);
1555
0
    return error;
1556
0
  }
1557
1558
0
  if (irp->device->IRPRequest)
1559
0
    error = irp->device->IRPRequest(irp->device, irp);
1560
0
  else
1561
0
    error = irp->Discard(irp);
1562
1563
0
  if (error != CHANNEL_RC_OK)
1564
0
  {
1565
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "device->IRPRequest failed with error %" PRIu32 "",
1566
0
               error);
1567
0
  }
1568
1569
0
  return error;
1570
0
}
1571
1572
static UINT rdpdr_process_component(rdpdrPlugin* rdpdr, UINT16 component, UINT16 packetId,
1573
                                    wStream* s)
1574
0
{
1575
0
  UINT32 type = 0;
1576
0
  DEVICE* device = NULL;
1577
1578
0
  WINPR_ASSERT(rdpdr);
1579
0
  WINPR_ASSERT(s);
1580
1581
0
  switch (component)
1582
0
  {
1583
0
    case RDPDR_CTYP_PRN:
1584
0
      type = RDPDR_DTYP_PRINT;
1585
0
      break;
1586
1587
0
    default:
1588
0
      return ERROR_INVALID_DATA;
1589
0
  }
1590
1591
0
  device = devman_get_device_by_type(rdpdr->devman, type);
1592
1593
0
  if (!device)
1594
0
    return ERROR_DEV_NOT_EXIST;
1595
1596
0
  return IFCALLRESULT(ERROR_INVALID_PARAMETER, device->CustomComponentRequest, device, component,
1597
0
                      packetId, s);
1598
0
}
1599
1600
/**
1601
 * Function description
1602
 *
1603
 * @return 0 on success, otherwise a Win32 error code
1604
 */
1605
static BOOL device_init(ULONG_PTR key, void* element, void* data)
1606
0
{
1607
0
  wLog* log = data;
1608
0
  UINT error = CHANNEL_RC_OK;
1609
0
  DEVICE* device = element;
1610
1611
0
  WINPR_UNUSED(key);
1612
0
  WINPR_UNUSED(data);
1613
1614
0
  IFCALLRET(device->Init, error, device);
1615
1616
0
  if (error != CHANNEL_RC_OK)
1617
0
  {
1618
0
    WLog_Print(log, WLOG_ERROR, "Device init failed with %s", WTSErrorToString(error));
1619
0
    return FALSE;
1620
0
  }
1621
0
  return TRUE;
1622
0
}
1623
1624
static UINT rdpdr_process_init(rdpdrPlugin* rdpdr)
1625
0
{
1626
0
  WINPR_ASSERT(rdpdr);
1627
0
  WINPR_ASSERT(rdpdr->devman);
1628
1629
0
  rdpdr->userLoggedOn = FALSE; /* reset possible received state */
1630
0
  if (!device_foreach(rdpdr, TRUE, device_init, rdpdr->log))
1631
0
    return ERROR_INTERNAL_ERROR;
1632
0
  return CHANNEL_RC_OK;
1633
0
}
1634
1635
static BOOL state_match(enum RDPDR_CHANNEL_STATE state, size_t count, va_list ap)
1636
0
{
1637
0
  for (size_t x = 0; x < count; x++)
1638
0
  {
1639
0
    enum RDPDR_CHANNEL_STATE cur = va_arg(ap, enum RDPDR_CHANNEL_STATE);
1640
0
    if (state == cur)
1641
0
      return TRUE;
1642
0
  }
1643
0
  return FALSE;
1644
0
}
1645
1646
static const char* state_str(size_t count, va_list ap, char* buffer, size_t size)
1647
0
{
1648
0
  for (size_t x = 0; x < count; x++)
1649
0
  {
1650
0
    enum RDPDR_CHANNEL_STATE cur = va_arg(ap, enum RDPDR_CHANNEL_STATE);
1651
0
    const char* curstr = rdpdr_state_str(cur);
1652
0
    winpr_str_append(curstr, buffer, size, "|");
1653
0
  }
1654
0
  return buffer;
1655
0
}
1656
1657
static BOOL rdpdr_state_check(rdpdrPlugin* rdpdr, UINT16 packetid, enum RDPDR_CHANNEL_STATE next,
1658
                              size_t count, ...)
1659
0
{
1660
0
  va_list ap = WINPR_C_ARRAY_INIT;
1661
0
  WINPR_ASSERT(rdpdr);
1662
1663
0
  va_start(ap, count);
1664
0
  BOOL rc = state_match(rdpdr->state, count, ap);
1665
0
  va_end(ap);
1666
1667
0
  if (!rc)
1668
0
  {
1669
0
    const char* strstate = rdpdr_state_str(rdpdr->state);
1670
0
    char buffer[256] = WINPR_C_ARRAY_INIT;
1671
1672
0
    va_start(ap, count);
1673
0
    state_str(count, ap, buffer, sizeof(buffer));
1674
0
    va_end(ap);
1675
1676
0
    WLog_Print(rdpdr->log, WLOG_ERROR,
1677
0
               "channel [RDPDR] received %s, expected states [%s] but have state %s, aborting.",
1678
0
               rdpdr_packetid_string(packetid), buffer, strstate);
1679
1680
0
    if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_INITIAL))
1681
0
      return FALSE;
1682
0
    return FALSE;
1683
0
  }
1684
0
  return rdpdr_state_advance(rdpdr, next);
1685
0
}
1686
1687
static BOOL rdpdr_check_channel_state(rdpdrPlugin* rdpdr, UINT16 packetid)
1688
0
{
1689
0
  WINPR_ASSERT(rdpdr);
1690
1691
0
  switch (packetid)
1692
0
  {
1693
0
    case PAKID_CORE_SERVER_ANNOUNCE:
1694
      /* windows servers sometimes send this message.
1695
       * it seems related to session login (e.g. first initialization for RDP/TLS style login,
1696
       * then reinitialize the channel after login successful
1697
       */
1698
0
      if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_INITIAL))
1699
0
        return FALSE;
1700
0
      return rdpdr_state_check(rdpdr, packetid, RDPDR_CHANNEL_STATE_ANNOUNCE, 1,
1701
0
                               RDPDR_CHANNEL_STATE_INITIAL);
1702
0
    case PAKID_CORE_SERVER_CAPABILITY:
1703
0
      return rdpdr_state_check(
1704
0
          rdpdr, packetid, RDPDR_CHANNEL_STATE_SERVER_CAPS, 6,
1705
0
          RDPDR_CHANNEL_STATE_NAME_REQUEST, RDPDR_CHANNEL_STATE_SERVER_CAPS,
1706
0
          RDPDR_CHANNEL_STATE_READY, RDPDR_CHANNEL_STATE_CLIENT_CAPS,
1707
0
          RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM, RDPDR_CHANNEL_STATE_USER_LOGGEDON);
1708
0
    case PAKID_CORE_CLIENTID_CONFIRM:
1709
0
      return rdpdr_state_check(rdpdr, packetid, RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM, 5,
1710
0
                               RDPDR_CHANNEL_STATE_NAME_REQUEST,
1711
0
                               RDPDR_CHANNEL_STATE_SERVER_CAPS,
1712
0
                               RDPDR_CHANNEL_STATE_CLIENT_CAPS, RDPDR_CHANNEL_STATE_READY,
1713
0
                               RDPDR_CHANNEL_STATE_USER_LOGGEDON);
1714
0
    case PAKID_CORE_USER_LOGGEDON:
1715
0
      if (!rdpdr_check_extended_pdu_flag(rdpdr, RDPDR_USER_LOGGEDON_PDU))
1716
0
        return FALSE;
1717
1718
0
      return rdpdr_state_check(
1719
0
          rdpdr, packetid, RDPDR_CHANNEL_STATE_USER_LOGGEDON, 4,
1720
0
          RDPDR_CHANNEL_STATE_NAME_REQUEST, RDPDR_CHANNEL_STATE_CLIENT_CAPS,
1721
0
          RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM, RDPDR_CHANNEL_STATE_READY);
1722
0
    default:
1723
0
    {
1724
0
      enum RDPDR_CHANNEL_STATE state = RDPDR_CHANNEL_STATE_READY;
1725
0
      return rdpdr_state_check(rdpdr, packetid, state, 1, state);
1726
0
    }
1727
0
  }
1728
0
}
1729
1730
static BOOL tryAdvance(rdpdrPlugin* rdpdr)
1731
0
{
1732
0
  if (rdpdr->haveClientId && rdpdr->haveServerCaps)
1733
0
  {
1734
0
    const UINT error = rdpdr_send_device_list_announce_request(rdpdr, FALSE);
1735
0
    if (error)
1736
0
    {
1737
0
      WLog_Print(rdpdr->log, WLOG_ERROR,
1738
0
                 "rdpdr_send_device_list_announce_request failed with error %" PRIu32 "",
1739
0
                 error);
1740
0
      return FALSE;
1741
0
    }
1742
0
    if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_READY))
1743
0
      return FALSE;
1744
0
  }
1745
0
  return TRUE;
1746
0
}
1747
1748
/**
1749
 * Function description
1750
 *
1751
 * @return 0 on success, otherwise a Win32 error code
1752
 */
1753
static UINT rdpdr_process_receive(rdpdrPlugin* rdpdr, wStream* s)
1754
0
{
1755
0
  UINT16 component = 0;
1756
0
  UINT16 packetId = 0;
1757
0
  UINT32 deviceId = 0;
1758
0
  UINT32 status = 0;
1759
0
  UINT error = ERROR_INVALID_DATA;
1760
1761
0
  if (!rdpdr || !s)
1762
0
    return CHANNEL_RC_NULL_DATA;
1763
1764
0
  rdpdr_dump_received_packet(rdpdr->log, WLOG_TRACE, s, "[rdpdr-channel] receive");
1765
0
  if (Stream_GetRemainingLength(s) >= 4)
1766
0
  {
1767
0
    Stream_Read_UINT16(s, component); /* Component (2 bytes) */
1768
0
    Stream_Read_UINT16(s, packetId);  /* PacketId (2 bytes) */
1769
1770
0
    if (component == RDPDR_CTYP_CORE)
1771
0
    {
1772
0
      if (!rdpdr_check_channel_state(rdpdr, packetId))
1773
0
        return CHANNEL_RC_OK;
1774
1775
0
      switch (packetId)
1776
0
      {
1777
0
        case PAKID_CORE_SERVER_ANNOUNCE:
1778
0
          rdpdr->haveClientId = FALSE;
1779
0
          rdpdr->haveServerCaps = FALSE;
1780
0
          if ((error = rdpdr_process_server_announce_request(rdpdr, s)))
1781
0
          {
1782
0
          }
1783
0
          else if ((error = rdpdr_send_client_announce_reply(rdpdr)))
1784
0
          {
1785
0
            WLog_Print(rdpdr->log, WLOG_ERROR,
1786
0
                       "rdpdr_send_client_announce_reply failed with error %" PRIu32 "",
1787
0
                       error);
1788
0
          }
1789
0
          else if ((error = rdpdr_send_client_name_request(rdpdr)))
1790
0
          {
1791
0
            WLog_Print(rdpdr->log, WLOG_ERROR,
1792
0
                       "rdpdr_send_client_name_request failed with error %" PRIu32 "",
1793
0
                       error);
1794
0
          }
1795
0
          else if ((error = rdpdr_process_init(rdpdr)))
1796
0
          {
1797
0
            WLog_Print(rdpdr->log, WLOG_ERROR,
1798
0
                       "rdpdr_process_init failed with error %" PRIu32 "", error);
1799
0
          }
1800
1801
0
          break;
1802
1803
0
        case PAKID_CORE_SERVER_CAPABILITY:
1804
0
          if ((error = rdpdr_process_capability_request(rdpdr, s)))
1805
0
          {
1806
0
          }
1807
0
          else if ((error = rdpdr_send_capability_response(rdpdr)))
1808
0
          {
1809
0
            WLog_Print(rdpdr->log, WLOG_ERROR,
1810
0
                       "rdpdr_send_capability_response failed with error %" PRIu32 "",
1811
0
                       error);
1812
0
          }
1813
0
          else
1814
0
          {
1815
0
            rdpdr->haveServerCaps = TRUE;
1816
0
            if (!tryAdvance(rdpdr))
1817
0
              error = ERROR_INTERNAL_ERROR;
1818
0
          }
1819
1820
0
          break;
1821
1822
0
        case PAKID_CORE_CLIENTID_CONFIRM:
1823
0
          if ((error = rdpdr_process_server_clientid_confirm(rdpdr, s)))
1824
0
          {
1825
0
          }
1826
0
          else
1827
0
          {
1828
0
            rdpdr->haveClientId = TRUE;
1829
0
            if (!tryAdvance(rdpdr))
1830
0
              error = ERROR_INTERNAL_ERROR;
1831
0
          }
1832
0
          break;
1833
1834
0
        case PAKID_CORE_USER_LOGGEDON:
1835
0
          if (!rdpdr->haveServerCaps)
1836
0
          {
1837
0
            WLog_Print(rdpdr->log, WLOG_ERROR,
1838
0
                       "Wrong state %s for %s. [serverCaps=%d, clientId=%d]",
1839
0
                       rdpdr_state_str(rdpdr->state), rdpdr_packetid_string(packetId),
1840
0
                       rdpdr->haveServerCaps, rdpdr->haveClientId);
1841
0
            error = ERROR_INTERNAL_ERROR;
1842
0
          }
1843
0
          else if ((error = rdpdr_send_device_list_announce_request(rdpdr, TRUE)))
1844
0
          {
1845
0
            WLog_Print(
1846
0
                rdpdr->log, WLOG_ERROR,
1847
0
                "rdpdr_send_device_list_announce_request failed with error %" PRIu32 "",
1848
0
                error);
1849
0
          }
1850
0
          else if (!tryAdvance(rdpdr))
1851
0
          {
1852
0
            error = ERROR_INTERNAL_ERROR;
1853
0
          }
1854
1855
0
          break;
1856
1857
0
        case PAKID_CORE_DEVICE_REPLY:
1858
1859
          /* connect to a specific resource */
1860
0
          if (Stream_GetRemainingLength(s) >= 8)
1861
0
          {
1862
0
            Stream_Read_UINT32(s, deviceId);
1863
0
            Stream_Read_UINT32(s, status);
1864
1865
0
            if (status != 0)
1866
0
              devman_unregister_device(rdpdr->devman, (void*)((size_t)deviceId));
1867
0
            error = CHANNEL_RC_OK;
1868
0
          }
1869
1870
0
          break;
1871
1872
0
        case PAKID_CORE_DEVICE_IOREQUEST:
1873
0
          if ((error = rdpdr_process_irp(rdpdr, s)))
1874
0
          {
1875
0
            WLog_Print(rdpdr->log, WLOG_ERROR,
1876
0
                       "rdpdr_process_irp failed with error %" PRIu32 "", error);
1877
0
            return error;
1878
0
          }
1879
0
          else
1880
0
            s = NULL;
1881
1882
0
          break;
1883
1884
0
        default:
1885
0
          WLog_Print(rdpdr->log, WLOG_ERROR,
1886
0
                     "RDPDR_CTYP_CORE unknown PacketId: 0x%04" PRIX16 "", packetId);
1887
0
          error = ERROR_INVALID_DATA;
1888
0
          break;
1889
0
      }
1890
0
    }
1891
0
    else
1892
0
    {
1893
0
      error = rdpdr_process_component(rdpdr, component, packetId, s);
1894
1895
0
      if (error != CHANNEL_RC_OK)
1896
0
      {
1897
0
        DWORD level = WLOG_ERROR;
1898
0
        if (rdpdr->ignoreInvalidDevices)
1899
0
        {
1900
0
          if (error == ERROR_DEV_NOT_EXIST)
1901
0
          {
1902
0
            level = WLOG_WARN;
1903
0
            error = CHANNEL_RC_OK;
1904
0
          }
1905
0
        }
1906
0
        WLog_Print(rdpdr->log, level,
1907
0
                   "Unknown message: Component: %s [0x%04" PRIX16
1908
0
                   "] PacketId: %s [0x%04" PRIX16 "]",
1909
0
                   rdpdr_component_string(component), component,
1910
0
                   rdpdr_packetid_string(packetId), packetId);
1911
0
      }
1912
0
    }
1913
0
  }
1914
1915
0
  return error;
1916
0
}
1917
1918
/**
1919
 * Function description
1920
 *
1921
 * @return 0 on success, otherwise a Win32 error code
1922
 */
1923
UINT rdpdr_send(rdpdrPlugin* rdpdr, wStream* s)
1924
0
{
1925
0
  rdpdrPlugin* plugin = rdpdr;
1926
1927
0
  if (!s)
1928
0
  {
1929
0
    Stream_Release(s);
1930
0
    return CHANNEL_RC_NULL_DATA;
1931
0
  }
1932
1933
0
  if (!plugin)
1934
0
  {
1935
0
    Stream_Release(s);
1936
0
    return CHANNEL_RC_BAD_INIT_HANDLE;
1937
0
  }
1938
1939
0
  const size_t pos = Stream_GetPosition(s);
1940
0
  UINT status = ERROR_INTERNAL_ERROR;
1941
0
  if (pos <= UINT32_MAX)
1942
0
  {
1943
0
    rdpdr_dump_send_packet(rdpdr->log, WLOG_TRACE, s, "[rdpdr-channel] send");
1944
0
    status = plugin->channelEntryPoints.pVirtualChannelWriteEx(
1945
0
        plugin->InitHandle, plugin->OpenHandle, Stream_Buffer(s), (UINT32)pos, s);
1946
0
  }
1947
1948
0
  if (status != CHANNEL_RC_OK)
1949
0
  {
1950
0
    Stream_Release(s);
1951
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "pVirtualChannelWriteEx failed with %s [%08" PRIX32 "]",
1952
0
               WTSErrorToString(status), status);
1953
0
  }
1954
1955
0
  return status;
1956
0
}
1957
1958
/**
1959
 * Function description
1960
 *
1961
 * @return 0 on success, otherwise a Win32 error code
1962
 */
1963
static UINT rdpdr_virtual_channel_event_data_received(rdpdrPlugin* rdpdr, void* pData,
1964
                                                      UINT32 dataLength, UINT32 totalLength,
1965
                                                      UINT32 dataFlags)
1966
0
{
1967
0
  wStream* data_in = NULL;
1968
1969
0
  WINPR_ASSERT(rdpdr);
1970
0
  WINPR_ASSERT(pData || (dataLength == 0));
1971
1972
0
  if ((dataFlags & CHANNEL_FLAG_SUSPEND) || (dataFlags & CHANNEL_FLAG_RESUME))
1973
0
  {
1974
    /*
1975
     * According to MS-RDPBCGR 2.2.6.1, "All virtual channel traffic MUST be suspended.
1976
     * This flag is only valid in server-to-client virtual channel traffic. It MUST be
1977
     * ignored in client-to-server data." Thus it would be best practice to cease data
1978
     * transmission. However, simply returning here avoids a crash.
1979
     */
1980
0
    return CHANNEL_RC_OK;
1981
0
  }
1982
1983
0
  if (dataFlags & CHANNEL_FLAG_FIRST)
1984
0
  {
1985
0
    if (rdpdr->data_in != NULL)
1986
0
      Stream_Release(rdpdr->data_in);
1987
1988
0
    rdpdr->data_in = StreamPool_Take(rdpdr->pool, totalLength);
1989
1990
0
    if (!rdpdr->data_in)
1991
0
    {
1992
0
      WLog_Print(rdpdr->log, WLOG_ERROR, "Stream_New failed!");
1993
0
      return CHANNEL_RC_NO_MEMORY;
1994
0
    }
1995
0
  }
1996
1997
0
  data_in = rdpdr->data_in;
1998
1999
0
  if (!Stream_EnsureRemainingCapacity(data_in, dataLength))
2000
0
  {
2001
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "Stream_EnsureRemainingCapacity failed!");
2002
0
    return ERROR_INVALID_DATA;
2003
0
  }
2004
2005
0
  Stream_Write(data_in, pData, dataLength);
2006
2007
0
  if (dataFlags & CHANNEL_FLAG_LAST)
2008
0
  {
2009
0
    const size_t pos = Stream_GetPosition(data_in);
2010
0
    const size_t cap = Stream_Capacity(data_in);
2011
0
    if (cap < pos)
2012
0
    {
2013
0
      WLog_Print(rdpdr->log, WLOG_ERROR,
2014
0
                 "rdpdr_virtual_channel_event_data_received: read error");
2015
0
      return ERROR_INTERNAL_ERROR;
2016
0
    }
2017
2018
0
    Stream_SealLength(data_in);
2019
0
    Stream_SetPosition(data_in, 0);
2020
2021
0
    if (rdpdr->async)
2022
0
    {
2023
0
      if (!MessageQueue_Post(rdpdr->queue, NULL, 0, (void*)data_in, NULL))
2024
0
      {
2025
0
        WLog_Print(rdpdr->log, WLOG_ERROR, "MessageQueue_Post failed!");
2026
0
        return ERROR_INTERNAL_ERROR;
2027
0
      }
2028
0
      rdpdr->data_in = NULL;
2029
0
    }
2030
0
    else
2031
0
    {
2032
0
      UINT error = rdpdr_process_receive(rdpdr, data_in);
2033
0
      Stream_Release(data_in);
2034
0
      rdpdr->data_in = NULL;
2035
0
      if (error)
2036
0
        return error;
2037
0
    }
2038
0
  }
2039
2040
0
  return CHANNEL_RC_OK;
2041
0
}
2042
2043
static VOID VCAPITYPE rdpdr_virtual_channel_open_event_ex(LPVOID lpUserParam, DWORD openHandle,
2044
                                                          UINT event, LPVOID pData,
2045
                                                          UINT32 dataLength, UINT32 totalLength,
2046
                                                          UINT32 dataFlags)
2047
0
{
2048
0
  UINT error = CHANNEL_RC_OK;
2049
0
  rdpdrPlugin* rdpdr = (rdpdrPlugin*)lpUserParam;
2050
2051
0
  WINPR_ASSERT(rdpdr);
2052
0
  switch (event)
2053
0
  {
2054
0
    case CHANNEL_EVENT_DATA_RECEIVED:
2055
0
      if (!rdpdr || !pData || (rdpdr->OpenHandle != openHandle))
2056
0
      {
2057
0
        WLog_Print(rdpdr->log, WLOG_ERROR, "error no match");
2058
0
        return;
2059
0
      }
2060
0
      if ((error = rdpdr_virtual_channel_event_data_received(rdpdr, pData, dataLength,
2061
0
                                                             totalLength, dataFlags)))
2062
0
        WLog_Print(rdpdr->log, WLOG_ERROR,
2063
0
                   "rdpdr_virtual_channel_event_data_received failed with error %" PRIu32
2064
0
                   "!",
2065
0
                   error);
2066
2067
0
      break;
2068
2069
0
    case CHANNEL_EVENT_WRITE_CANCELLED:
2070
0
    case CHANNEL_EVENT_WRITE_COMPLETE:
2071
0
    {
2072
0
      wStream* s = (wStream*)pData;
2073
0
      Stream_Release(s);
2074
0
    }
2075
0
    break;
2076
2077
0
    case CHANNEL_EVENT_USER:
2078
0
      break;
2079
0
    default:
2080
0
      break;
2081
0
  }
2082
2083
0
  if (error && rdpdr && rdpdr->rdpcontext)
2084
0
    setChannelError(rdpdr->rdpcontext, error,
2085
0
                    "rdpdr_virtual_channel_open_event_ex reported an error");
2086
0
}
2087
2088
static DWORD WINAPI rdpdr_virtual_channel_client_thread(LPVOID arg)
2089
0
{
2090
0
  rdpdrPlugin* rdpdr = (rdpdrPlugin*)arg;
2091
0
  UINT error = 0;
2092
2093
0
  if (!rdpdr)
2094
0
  {
2095
0
    ExitThread((DWORD)CHANNEL_RC_NULL_DATA);
2096
0
    return CHANNEL_RC_NULL_DATA;
2097
0
  }
2098
2099
0
  if ((error = rdpdr_process_connect(rdpdr)))
2100
0
  {
2101
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "rdpdr_process_connect failed with error %" PRIu32 "!",
2102
0
               error);
2103
2104
0
    if (rdpdr->rdpcontext)
2105
0
      setChannelError(rdpdr->rdpcontext, error,
2106
0
                      "rdpdr_virtual_channel_client_thread reported an error");
2107
2108
0
    ExitThread(error);
2109
0
    return error;
2110
0
  }
2111
2112
0
  while (1)
2113
0
  {
2114
0
    wMessage message = WINPR_C_ARRAY_INIT;
2115
0
    WINPR_ASSERT(rdpdr);
2116
2117
0
    if (!MessageQueue_Wait(rdpdr->queue))
2118
0
      break;
2119
2120
0
    if (MessageQueue_Peek(rdpdr->queue, &message, TRUE))
2121
0
    {
2122
0
      if (message.id == WMQ_QUIT)
2123
0
        break;
2124
2125
0
      if (message.id == 0)
2126
0
      {
2127
0
        wStream* data = (wStream*)message.wParam;
2128
2129
0
        error = rdpdr_process_receive(rdpdr, data);
2130
2131
0
        Stream_Release(data);
2132
0
        if (error)
2133
0
        {
2134
0
          WLog_Print(rdpdr->log, WLOG_ERROR,
2135
0
                     "rdpdr_process_receive failed with error %" PRIu32 "!", error);
2136
2137
0
          if (rdpdr->rdpcontext)
2138
0
            setChannelError(rdpdr->rdpcontext, error,
2139
0
                            "rdpdr_virtual_channel_client_thread reported an error");
2140
2141
0
          goto fail;
2142
0
        }
2143
0
      }
2144
0
    }
2145
0
  }
2146
2147
0
fail:
2148
0
  if ((error = drive_hotplug_thread_terminate(rdpdr)))
2149
0
    WLog_Print(rdpdr->log, WLOG_ERROR,
2150
0
               "drive_hotplug_thread_terminate failed with error %" PRIu32 "!", error);
2151
2152
0
  ExitThread(error);
2153
0
  return error;
2154
0
}
2155
2156
static void queue_free(void* obj)
2157
0
{
2158
0
  wStream* s = NULL;
2159
0
  wMessage* msg = (wMessage*)obj;
2160
2161
0
  if (!msg || (msg->id != 0))
2162
0
    return;
2163
2164
0
  s = (wStream*)msg->wParam;
2165
0
  WINPR_ASSERT(s);
2166
0
  Stream_Release(s);
2167
0
}
2168
2169
/**
2170
 * Function description
2171
 *
2172
 * @return 0 on success, otherwise a Win32 error code
2173
 */
2174
static UINT rdpdr_virtual_channel_event_connected(rdpdrPlugin* rdpdr, LPVOID pData,
2175
                                                  UINT32 dataLength)
2176
0
{
2177
0
  wObject* obj = NULL;
2178
2179
0
  WINPR_ASSERT(rdpdr);
2180
0
  WINPR_UNUSED(pData);
2181
0
  WINPR_UNUSED(dataLength);
2182
2183
0
  if (rdpdr->async)
2184
0
  {
2185
0
    rdpdr->queue = MessageQueue_New(NULL);
2186
2187
0
    if (!rdpdr->queue)
2188
0
    {
2189
0
      WLog_Print(rdpdr->log, WLOG_ERROR, "MessageQueue_New failed!");
2190
0
      return CHANNEL_RC_NO_MEMORY;
2191
0
    }
2192
2193
0
    obj = MessageQueue_Object(rdpdr->queue);
2194
0
    obj->fnObjectFree = queue_free;
2195
2196
0
    if (!(rdpdr->thread = CreateThread(NULL, 0, rdpdr_virtual_channel_client_thread,
2197
0
                                       (void*)rdpdr, 0, NULL)))
2198
0
    {
2199
0
      WLog_Print(rdpdr->log, WLOG_ERROR, "CreateThread failed!");
2200
0
      return ERROR_INTERNAL_ERROR;
2201
0
    }
2202
0
  }
2203
0
  else
2204
0
  {
2205
0
    UINT error = rdpdr_process_connect(rdpdr);
2206
0
    if (error)
2207
0
    {
2208
0
      WLog_Print(rdpdr->log, WLOG_ERROR,
2209
0
                 "rdpdr_process_connect failed with error %" PRIu32 "!", error);
2210
0
      return error;
2211
0
    }
2212
0
  }
2213
2214
0
  return rdpdr->channelEntryPoints.pVirtualChannelOpenEx(rdpdr->InitHandle, &rdpdr->OpenHandle,
2215
0
                                                         rdpdr->channelDef.name,
2216
0
                                                         rdpdr_virtual_channel_open_event_ex);
2217
0
}
2218
2219
/**
2220
 * Function description
2221
 *
2222
 * @return 0 on success, otherwise a Win32 error code
2223
 */
2224
static UINT rdpdr_virtual_channel_event_disconnected(rdpdrPlugin* rdpdr)
2225
0
{
2226
0
  UINT error = 0;
2227
2228
0
  WINPR_ASSERT(rdpdr);
2229
2230
0
  if (rdpdr->OpenHandle == 0)
2231
0
    return CHANNEL_RC_OK;
2232
2233
0
  if (rdpdr->queue && rdpdr->thread)
2234
0
  {
2235
0
    if (MessageQueue_PostQuit(rdpdr->queue, 0) &&
2236
0
        (WaitForSingleObject(rdpdr->thread, INFINITE) == WAIT_FAILED))
2237
0
    {
2238
0
      error = GetLastError();
2239
0
      WLog_Print(rdpdr->log, WLOG_ERROR, "WaitForSingleObject failed with error %" PRIu32 "!",
2240
0
                 error);
2241
0
      return error;
2242
0
    }
2243
0
  }
2244
2245
0
  if (rdpdr->thread)
2246
0
    (void)CloseHandle(rdpdr->thread);
2247
0
  MessageQueue_Free(rdpdr->queue);
2248
0
  rdpdr->queue = NULL;
2249
0
  rdpdr->thread = NULL;
2250
2251
0
  WINPR_ASSERT(rdpdr->channelEntryPoints.pVirtualChannelCloseEx);
2252
0
  error = rdpdr->channelEntryPoints.pVirtualChannelCloseEx(rdpdr->InitHandle, rdpdr->OpenHandle);
2253
2254
0
  if (CHANNEL_RC_OK != error)
2255
0
  {
2256
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "pVirtualChannelCloseEx failed with %s [%08" PRIX32 "]",
2257
0
               WTSErrorToString(error), error);
2258
0
  }
2259
2260
0
  rdpdr->OpenHandle = 0;
2261
2262
0
  if (rdpdr->data_in)
2263
0
  {
2264
0
    Stream_Release(rdpdr->data_in);
2265
0
    rdpdr->data_in = NULL;
2266
0
  }
2267
2268
0
  if (rdpdr->devman)
2269
0
  {
2270
0
    devman_free(rdpdr->devman);
2271
0
    rdpdr->devman = NULL;
2272
0
  }
2273
2274
0
  return error;
2275
0
}
2276
2277
static void rdpdr_virtual_channel_event_terminated(rdpdrPlugin* rdpdr)
2278
0
{
2279
0
  WINPR_ASSERT(rdpdr);
2280
0
#if !defined(_WIN32)
2281
0
  if (rdpdr->stopEvent)
2282
0
  {
2283
0
    (void)CloseHandle(rdpdr->stopEvent);
2284
0
    rdpdr->stopEvent = NULL;
2285
0
  }
2286
0
#endif
2287
0
  rdpdr->InitHandle = 0;
2288
0
  StreamPool_Free(rdpdr->pool);
2289
0
  free(rdpdr);
2290
0
}
2291
2292
static UINT rdpdr_register_device(RdpdrClientContext* context, const RDPDR_DEVICE* device,
2293
                                  uint32_t* pid)
2294
0
{
2295
0
  WINPR_ASSERT(context);
2296
0
  WINPR_ASSERT(device);
2297
0
  WINPR_ASSERT(pid);
2298
2299
0
  rdpdrPlugin* rdpdr = context->handle;
2300
0
  WINPR_ASSERT(rdpdr);
2301
2302
0
  RDPDR_DEVICE* copy = freerdp_device_clone(device);
2303
0
  if (!copy)
2304
0
    return ERROR_INVALID_DATA;
2305
0
  UINT rc = devman_load_device_service(rdpdr->devman, copy, rdpdr->rdpcontext);
2306
0
  *pid = copy->Id;
2307
0
  freerdp_device_free(copy);
2308
0
  if (rc == CHANNEL_RC_OK)
2309
0
    rc = rdpdr_try_send_device_list_announce_request(rdpdr);
2310
0
  return rc;
2311
0
}
2312
2313
static UINT rdpdr_unregister_device(RdpdrClientContext* context, size_t count, const uint32_t ids[])
2314
0
{
2315
0
  WINPR_ASSERT(context);
2316
2317
0
  rdpdrPlugin* rdpdr = context->handle;
2318
0
  WINPR_ASSERT(rdpdr);
2319
2320
0
  for (size_t x = 0; x < count; x++)
2321
0
  {
2322
0
    const uintptr_t id = ids[x];
2323
0
    devman_unregister_device(rdpdr->devman, (void*)id);
2324
0
  }
2325
0
  return rdpdr_send_device_list_remove_request(rdpdr, WINPR_ASSERTING_INT_CAST(uint32_t, count),
2326
0
                                               ids);
2327
0
}
2328
2329
static UINT rdpdr_virtual_channel_event_initialized(rdpdrPlugin* rdpdr,
2330
                                                    WINPR_ATTR_UNUSED LPVOID pData,
2331
                                                    WINPR_ATTR_UNUSED UINT32 dataLength)
2332
0
{
2333
0
  WINPR_ASSERT(rdpdr);
2334
0
#if !defined(_WIN32)
2335
0
  WINPR_ASSERT(!rdpdr->stopEvent);
2336
0
  rdpdr->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
2337
0
  WINPR_ASSERT(rdpdr->stopEvent);
2338
0
#endif
2339
2340
0
  rdpdr->context.handle = rdpdr;
2341
0
  rdpdr->context.RdpdrHotplugDevice = handle_hotplug;
2342
0
  rdpdr->context.RdpdrRegisterDevice = rdpdr_register_device;
2343
0
  rdpdr->context.RdpdrUnregisterDevice = rdpdr_unregister_device;
2344
0
  return CHANNEL_RC_OK;
2345
0
}
2346
2347
static VOID VCAPITYPE rdpdr_virtual_channel_init_event_ex(LPVOID lpUserParam, LPVOID pInitHandle,
2348
                                                          UINT event, LPVOID pData, UINT dataLength)
2349
0
{
2350
0
  UINT error = CHANNEL_RC_OK;
2351
0
  rdpdrPlugin* rdpdr = (rdpdrPlugin*)lpUserParam;
2352
2353
0
  if (!rdpdr || (rdpdr->InitHandle != pInitHandle))
2354
0
  {
2355
0
    WLog_ERR(TAG, "error no match");
2356
0
    return;
2357
0
  }
2358
2359
0
  WINPR_ASSERT(pData || (dataLength == 0));
2360
2361
0
  switch (event)
2362
0
  {
2363
0
    case CHANNEL_EVENT_INITIALIZED:
2364
0
      error = rdpdr_virtual_channel_event_initialized(rdpdr, pData, dataLength);
2365
0
      break;
2366
2367
0
    case CHANNEL_EVENT_CONNECTED:
2368
0
      if ((error = rdpdr_virtual_channel_event_connected(rdpdr, pData, dataLength)))
2369
0
        WLog_Print(rdpdr->log, WLOG_ERROR,
2370
0
                   "rdpdr_virtual_channel_event_connected failed with error %" PRIu32 "!",
2371
0
                   error);
2372
2373
0
      break;
2374
2375
0
    case CHANNEL_EVENT_DISCONNECTED:
2376
0
      if ((error = rdpdr_virtual_channel_event_disconnected(rdpdr)))
2377
0
        WLog_Print(rdpdr->log, WLOG_ERROR,
2378
0
                   "rdpdr_virtual_channel_event_disconnected failed with error %" PRIu32
2379
0
                   "!",
2380
0
                   error);
2381
2382
0
      break;
2383
2384
0
    case CHANNEL_EVENT_TERMINATED:
2385
0
      rdpdr_virtual_channel_event_terminated(rdpdr);
2386
0
      rdpdr = NULL;
2387
0
      break;
2388
2389
0
    case CHANNEL_EVENT_ATTACHED:
2390
0
    case CHANNEL_EVENT_DETACHED:
2391
0
    default:
2392
0
      WLog_Print(rdpdr->log, WLOG_ERROR, "unknown event %" PRIu32 "!", event);
2393
0
      break;
2394
0
  }
2395
2396
0
  if (error && rdpdr && rdpdr->rdpcontext)
2397
0
    setChannelError(rdpdr->rdpcontext, error,
2398
0
                    "rdpdr_virtual_channel_init_event_ex reported an error");
2399
0
}
2400
2401
/* rdpdr is always built-in */
2402
#define VirtualChannelEntryEx rdpdr_VirtualChannelEntryEx
2403
2404
FREERDP_ENTRY_POINT(BOOL VCAPITYPE VirtualChannelEntryEx(PCHANNEL_ENTRY_POINTS_EX pEntryPoints,
2405
                                                         PVOID pInitHandle))
2406
0
{
2407
0
  WINPR_ASSERT(pEntryPoints);
2408
0
  WINPR_ASSERT(pInitHandle);
2409
2410
0
  rdpdrPlugin* rdpdr = (rdpdrPlugin*)calloc(1, sizeof(rdpdrPlugin));
2411
2412
0
  if (!rdpdr)
2413
0
  {
2414
0
    WLog_ERR(TAG, "calloc failed!");
2415
0
    return FALSE;
2416
0
  }
2417
0
  rdpdr->log = WLog_Get(TAG);
2418
2419
0
  rdpdr->clientExtendedPDU =
2420
0
      RDPDR_DEVICE_REMOVE_PDUS | RDPDR_CLIENT_DISPLAY_NAME_PDU | RDPDR_USER_LOGGEDON_PDU;
2421
0
  rdpdr->clientIOCode1 =
2422
0
      RDPDR_IRP_MJ_CREATE | RDPDR_IRP_MJ_CLEANUP | RDPDR_IRP_MJ_CLOSE | RDPDR_IRP_MJ_READ |
2423
0
      RDPDR_IRP_MJ_WRITE | RDPDR_IRP_MJ_FLUSH_BUFFERS | RDPDR_IRP_MJ_SHUTDOWN |
2424
0
      RDPDR_IRP_MJ_DEVICE_CONTROL | RDPDR_IRP_MJ_QUERY_VOLUME_INFORMATION |
2425
0
      RDPDR_IRP_MJ_SET_VOLUME_INFORMATION | RDPDR_IRP_MJ_QUERY_INFORMATION |
2426
0
      RDPDR_IRP_MJ_SET_INFORMATION | RDPDR_IRP_MJ_DIRECTORY_CONTROL | RDPDR_IRP_MJ_LOCK_CONTROL |
2427
0
      RDPDR_IRP_MJ_QUERY_SECURITY | RDPDR_IRP_MJ_SET_SECURITY;
2428
2429
0
  rdpdr->clientExtraFlags1 = ENABLE_ASYNCIO;
2430
2431
0
  rdpdr->pool = StreamPool_New(TRUE, 1024);
2432
0
  if (!rdpdr->pool)
2433
0
  {
2434
0
    free(rdpdr);
2435
0
    return FALSE;
2436
0
  }
2437
2438
0
  rdpdr->channelDef.options =
2439
0
      CHANNEL_OPTION_INITIALIZED | CHANNEL_OPTION_ENCRYPT_RDP | CHANNEL_OPTION_COMPRESS_RDP;
2440
0
  (void)sprintf_s(rdpdr->channelDef.name, ARRAYSIZE(rdpdr->channelDef.name),
2441
0
                  RDPDR_SVC_CHANNEL_NAME);
2442
0
  rdpdr->sequenceId = 0;
2443
0
  CHANNEL_ENTRY_POINTS_FREERDP_EX* pEntryPointsEx =
2444
0
      (CHANNEL_ENTRY_POINTS_FREERDP_EX*)pEntryPoints;
2445
2446
0
  if ((pEntryPointsEx->cbSize >= sizeof(CHANNEL_ENTRY_POINTS_FREERDP_EX)) &&
2447
0
      (pEntryPointsEx->MagicNumber == FREERDP_CHANNEL_MAGIC_NUMBER))
2448
0
  {
2449
0
    rdpdr->rdpcontext = pEntryPointsEx->context;
2450
0
    if (!freerdp_settings_get_bool(rdpdr->rdpcontext->settings,
2451
0
                                   FreeRDP_SynchronousStaticChannels))
2452
0
      rdpdr->async = TRUE;
2453
0
  }
2454
2455
0
  CopyMemory(&(rdpdr->channelEntryPoints), pEntryPoints, sizeof(CHANNEL_ENTRY_POINTS_FREERDP_EX));
2456
0
  rdpdr->InitHandle = pInitHandle;
2457
0
  const UINT rc = rdpdr->channelEntryPoints.pVirtualChannelInitEx(
2458
0
      rdpdr, &rdpdr->context, pInitHandle, &rdpdr->channelDef, 1, VIRTUAL_CHANNEL_VERSION_WIN2000,
2459
0
      rdpdr_virtual_channel_init_event_ex);
2460
2461
0
  if (CHANNEL_RC_OK != rc)
2462
0
  {
2463
0
    WLog_Print(rdpdr->log, WLOG_ERROR, "pVirtualChannelInitEx failed with %s [%08" PRIX32 "]",
2464
0
               WTSErrorToString(rc), rc);
2465
0
    free(rdpdr);
2466
0
    return FALSE;
2467
0
  }
2468
2469
0
  return TRUE;
2470
0
}