25#include <freerdp/config.h>
33#include <winpr/sysinfo.h>
34#include <winpr/assert.h>
35#include <winpr/stream.h>
37#include <winpr/print.h>
38#include <winpr/sspicli.h>
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>
57#include <CoreFoundation/CoreFoundation.h>
65#include "rdpdr_capabilities.h"
70#include "rdpdr_main.h"
72#define TAG CHANNELS_TAG("rdpdr.client")
82static const char* rdpdr_state_str(
enum RDPDR_CHANNEL_STATE state)
86 case RDPDR_CHANNEL_STATE_INITIAL:
87 return "RDPDR_CHANNEL_STATE_INITIAL";
88 case RDPDR_CHANNEL_STATE_ANNOUNCE:
89 return "RDPDR_CHANNEL_STATE_ANNOUNCE";
90 case RDPDR_CHANNEL_STATE_ANNOUNCE_REPLY:
91 return "RDPDR_CHANNEL_STATE_ANNOUNCE_REPLY";
92 case RDPDR_CHANNEL_STATE_NAME_REQUEST:
93 return "RDPDR_CHANNEL_STATE_NAME_REQUEST";
94 case RDPDR_CHANNEL_STATE_SERVER_CAPS:
95 return "RDPDR_CHANNEL_STATE_SERVER_CAPS";
96 case RDPDR_CHANNEL_STATE_CLIENT_CAPS:
97 return "RDPDR_CHANNEL_STATE_CLIENT_CAPS";
98 case RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM:
99 return "RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM";
100 case RDPDR_CHANNEL_STATE_READY:
101 return "RDPDR_CHANNEL_STATE_READY";
102 case RDPDR_CHANNEL_STATE_USER_LOGGEDON:
103 return "RDPDR_CHANNEL_STATE_USER_LOGGEDON";
105 return "RDPDR_CHANNEL_STATE_UNKNOWN";
109static const char* support_str(BOOL val)
116static const char* rdpdr_caps_pdu_str(UINT32 flag)
120 case RDPDR_DEVICE_REMOVE_PDUS:
121 return "RDPDR_USER_LOGGEDON_PDU";
122 case RDPDR_CLIENT_DISPLAY_NAME_PDU:
123 return "RDPDR_CLIENT_DISPLAY_NAME_PDU";
124 case RDPDR_USER_LOGGEDON_PDU:
125 return "RDPDR_USER_LOGGEDON_PDU";
127 return "RDPDR_UNKNONW";
131static BOOL rdpdr_check_extended_pdu_flag(
rdpdrPlugin* rdpdr, UINT32 flag)
135 const BOOL client = (rdpdr->clientExtendedPDU & flag) != 0;
136 const BOOL server = (rdpdr->serverExtendedPDU & flag) != 0;
138 if (!client || !server)
140 WLog_Print(rdpdr->log, WLOG_WARN,
"Checking ExtendedPDU::%s, client %s, server %s",
141 rdpdr_caps_pdu_str(flag), support_str(client), support_str(server));
147BOOL rdpdr_state_advance(
rdpdrPlugin* rdpdr,
enum RDPDR_CHANNEL_STATE next)
151 if (next != rdpdr->state)
152 WLog_Print(rdpdr->log, WLOG_DEBUG,
"[RDPDR] transition from %s to %s",
153 rdpdr_state_str(rdpdr->state), rdpdr_state_str(next));
158static BOOL device_foreach(
rdpdrPlugin* rdpdr, BOOL abortOnFail,
159 BOOL (*fkt)(ULONG_PTR key,
void* element,
void* data),
void* data)
162 ULONG_PTR* keys =
nullptr;
164 ListDictionary_Lock(rdpdr->devman->devices);
165 const size_t count = ListDictionary_GetKeys(rdpdr->devman->devices, &keys);
166 for (
size_t x = 0; x < count; x++)
168 void* element = ListDictionary_GetItemValue(rdpdr->devman->devices, (
void*)keys[x]);
169 if (!fkt(keys[x], element, data))
177 ListDictionary_Unlock(rdpdr->devman->devices);
186static UINT rdpdr_try_send_device_list_announce_request(
rdpdrPlugin* rdpdr);
188static BOOL rdpdr_load_drive(
rdpdrPlugin* rdpdr,
const char* name,
const char* path, BOOL automount)
190 UINT rc = ERROR_INTERNAL_ERROR;
196 const char* args[] = { name, path, automount ? nullptr : name };
198 drive.device = freerdp_device_new(RDPDR_DTYP_FILESYSTEM, ARRAYSIZE(args), args);
202 WINPR_ASSERT(rdpdr->context.RdpdrRegisterDevice);
203 rc = rdpdr->context.RdpdrRegisterDevice(&rdpdr->context, drive.device, &drive.device->Id);
204 if (rc != CHANNEL_RC_OK)
208 freerdp_device_free(drive.device);
209 return rc == CHANNEL_RC_OK;
217static UINT rdpdr_send_device_list_remove_request(
rdpdrPlugin* rdpdr, UINT32 count,
223 WINPR_ASSERT(ids || (count == 0));
226 return CHANNEL_RC_OK;
228 if (!rdpdr_check_extended_pdu_flag(rdpdr, RDPDR_DEVICE_REMOVE_PDUS))
229 return CHANNEL_RC_OK;
231 s = StreamPool_Take(rdpdr->pool, count *
sizeof(UINT32) + 8);
235 WLog_Print(rdpdr->log, WLOG_ERROR,
"Stream_New failed!");
236 return CHANNEL_RC_NO_MEMORY;
239 Stream_Write_UINT16(s, RDPDR_CTYP_CORE);
240 Stream_Write_UINT16(s, PAKID_CORE_DEVICELIST_REMOVE);
241 Stream_Write_UINT32(s, count);
243 for (UINT32 i = 0; i < count; i++)
244 Stream_Write_UINT32(s, ids[i]);
246 Stream_SealLength(s);
247 return rdpdr_send(rdpdr, s);
250#if defined(_UWP) || defined(__IOS__)
252static UINT handle_hotplug(WINPR_ATTR_UNUSED RdpdrClientContext* context,
253 WINPR_ATTR_UNUSED RdpdrHotplugEventType type)
255 return ERROR_CALL_NOT_IMPLEMENTED;
258static void first_hotplug(WINPR_ATTR_UNUSED
rdpdrPlugin* rdpdr)
262static DWORD WINAPI drive_hotplug_thread_func(WINPR_ATTR_UNUSED LPVOID arg)
264 return CHANNEL_RC_OK;
267static UINT drive_hotplug_thread_terminate(WINPR_ATTR_UNUSED
rdpdrPlugin* rdpdr)
269 return CHANNEL_RC_OK;
274static UINT handle_hotplug(WINPR_ATTR_UNUSED RdpdrClientContext* context,
275 WINPR_ATTR_UNUSED RdpdrHotplugEventType type)
277 return CHANNEL_RC_OK;
280static BOOL check_path(
const char* path)
282 UINT type = GetDriveTypeA(path);
284 if (!(type == DRIVE_FIXED || type == DRIVE_REMOVABLE || type == DRIVE_CDROM ||
285 type == DRIVE_REMOTE))
288 return GetVolumeInformationA(path,
nullptr, 0,
nullptr,
nullptr,
nullptr,
nullptr, 0);
293 DWORD unitmask = GetLogicalDrives();
295 for (
size_t i = 0; i < 26; i++)
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;
304 if (check_path(drive_path))
306 rdpdr_load_drive(rdpdr, drive_name, drive_path, TRUE);
310 unitmask = unitmask >> 1;
314static LRESULT CALLBACK hotplug_proc(HWND hWnd, UINT Msg, WPARAM wParam, LPARAM lParam)
317 PDEV_BROADCAST_HDR lpdb = (PDEV_BROADCAST_HDR)lParam;
319 rdpdr = (
rdpdrPlugin*)GetWindowLongPtr(hWnd, GWLP_USERDATA);
323 case WM_DEVICECHANGE:
326 case DBT_DEVICEARRIVAL:
327 if (lpdb->dbch_devicetype == DBT_DEVTYP_VOLUME)
329 PDEV_BROADCAST_VOLUME lpdbv = (PDEV_BROADCAST_VOLUME)lpdb;
330 DWORD unitmask = lpdbv->dbcv_unitmask;
332 for (
int i = 0; i < 26; i++)
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;
341 if (check_path(drive_path))
343 rdpdr_load_drive(rdpdr, drive_name, drive_path, TRUE);
347 unitmask = unitmask >> 1;
353 case DBT_DEVICEREMOVECOMPLETE:
354 if (lpdb->dbch_devicetype == DBT_DEVTYP_VOLUME)
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 =
nullptr;
360 DEVICE_DRIVE_EXT* device_ext;
362 for (
int i = 0; i < 26; i++)
366 drive_name_upper =
'A' + i;
367 drive_name_lower =
'a' + i;
369 ListDictionary_GetKeys(rdpdr->devman->devices, &keys);
371 for (
size_t j = 0; j < count; j++)
373 device_ext = (DEVICE_DRIVE_EXT*)ListDictionary_GetItemValue(
374 rdpdr->devman->devices, (
void*)keys[j]);
376 if (device_ext->device.type != RDPDR_DTYP_FILESYSTEM)
379 if (device_ext->path[0] == drive_name_upper ||
380 device_ext->path[0] == drive_name_lower)
382 if (device_ext->automount)
384 const uint32_t ids[] = { keys[j] };
385 WINPR_ASSERT(rdpdr->context.RdpdrUnregisterDevice);
386 error = rdpdr->context.RdpdrUnregisterDevice(
387 &rdpdr->context, ARRAYSIZE(ids), ids);
392 rdpdr->log, WLOG_ERROR,
393 "rdpdr_send_device_list_remove_request failed "
394 "with error %" PRIu32
"!",
406 unitmask = unitmask >> 1;
419 return DefWindowProc(hWnd, Msg, wParam, lParam);
422 return DefWindowProc(hWnd, Msg, wParam, lParam);
425static DWORD WINAPI drive_hotplug_thread_func(LPVOID arg)
432 DEV_BROADCAST_HANDLE NotificationFilter;
433 HDEVNOTIFY hDevNotify;
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(
nullptr, IDI_APPLICATION);
442 wnd_cls.hCursor =
nullptr;
443 wnd_cls.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
444 wnd_cls.lpszMenuName =
nullptr;
445 wnd_cls.lpszClassName = L
"DRIVE_HOTPLUG";
446 wnd_cls.hInstance =
nullptr;
447 wnd_cls.hIconSm = LoadIcon(
nullptr, IDI_APPLICATION);
448 RegisterClassEx(&wnd_cls);
450 hwnd = CreateWindowEx(0, L
"DRIVE_HOTPLUG",
nullptr, 0, 0, 0, 0, 0,
nullptr,
nullptr,
nullptr,
452 SetWindowLongPtr(hwnd, GWLP_USERDATA, (LONG_PTR)rdpdr);
453 rdpdr->hotplug_wnd = hwnd;
455 NotificationFilter.dbch_size =
sizeof(DEV_BROADCAST_HANDLE);
456 NotificationFilter.dbch_devicetype = DBT_DEVTYP_HANDLE;
457 hDevNotify = RegisterDeviceNotification(hwnd, &NotificationFilter, DEVICE_NOTIFY_WINDOW_HANDLE);
460 while ((bRet = GetMessage(&msg, 0, 0, 0)) != 0)
468 TranslateMessage(&msg);
469 DispatchMessage(&msg);
473 UnregisterDeviceNotification(hDevNotify);
474 return CHANNEL_RC_OK;
482static UINT drive_hotplug_thread_terminate(
rdpdrPlugin* rdpdr)
484 UINT error = CHANNEL_RC_OK;
486 if (rdpdr->hotplug_wnd && !PostMessage(rdpdr->hotplug_wnd, WM_QUIT, 0, 0))
488 error = GetLastError();
489 WLog_Print(rdpdr->log, WLOG_ERROR,
"PostMessage failed with error %" PRIu32
"", error);
495#elif defined(__MACOSX__)
497#define MAX_USB_DEVICES 100
510static UINT handle_hotplug(WINPR_ATTR_UNUSED RdpdrClientContext* context,
511 WINPR_ATTR_UNUSED RdpdrHotplugEventType type)
513 WINPR_ASSERT(context);
516 struct dirent* pDirent =
nullptr;
517 char fullpath[PATH_MAX] = WINPR_C_ARRAY_INIT;
518 char* szdir = (
char*)
"/Volumes";
519 struct stat buf = WINPR_C_ARRAY_INIT;
520 hotplug_dev dev_array[MAX_USB_DEVICES] = WINPR_C_ARRAY_INIT;
522 DEVICE_DRIVE_EXT* device_ext =
nullptr;
523 ULONG_PTR* keys =
nullptr;
525 UINT error = ERROR_INTERNAL_ERROR;
528 DIR* pDir = opendir(szdir);
532 printf(
"Cannot open directory\n");
533 return ERROR_OPEN_FAILED;
536 while ((pDirent = readdir(pDir)) !=
nullptr)
538 if (pDirent->d_name[0] !=
'.')
540 (void)sprintf_s(fullpath, ARRAYSIZE(fullpath),
"%s/%s", szdir, pDirent->d_name);
541 if (stat(fullpath, &buf) != 0)
544 if (S_ISDIR(buf.st_mode))
546 dev_array[size].path = _strdup(fullpath);
548 if (!dev_array[size].path)
551 error = CHANNEL_RC_NO_MEMORY;
555 dev_array[size++].to_add = TRUE;
562 count = ListDictionary_GetKeys(rdpdr->devman->devices, &keys);
564 for (
size_t j = 0; j < count; j++)
566 char* path =
nullptr;
567 BOOL dev_found = FALSE;
569 (DEVICE_DRIVE_EXT*)ListDictionary_GetItemValue(rdpdr->devman->devices, (
void*)keys[j]);
571 if (!device_ext || !device_ext->automount)
574 if (device_ext->device.type != RDPDR_DTYP_FILESYSTEM)
577 if (device_ext->path ==
nullptr)
580 path = ConvertWCharToUtf8Alloc(device_ext->path,
nullptr);
585 if (strstr(path,
"/Volumes/") ==
nullptr)
591 for (
size_t i = 0; i < size; i++)
593 if (strstr(path, dev_array[i].path) !=
nullptr)
596 dev_array[i].to_add = FALSE;
605 const uint32_t ids[] = { keys[j] };
606 WINPR_ASSERT(rdpdr->context.RdpdrUnregisterDevice);
607 error = rdpdr->context.RdpdrUnregisterDevice(&rdpdr->context, ARRAYSIZE(ids), ids);
610 WLog_Print(rdpdr->log, WLOG_ERROR,
611 "rdpdr_send_device_list_remove_request failed with error %" PRIu32
"!",
619 for (
size_t i = 0; i < size; i++)
621 const hotplug_dev* dev = &dev_array[i];
624 const char* path = dev->path;
625 const char* name = strrchr(path,
'/') + 1;
626 error = rdpdr_load_drive(rdpdr, name, path, TRUE);
635 for (
size_t i = 0; i < size; i++)
636 free(dev_array[i].path);
641static void drive_hotplug_fsevent_callback(ConstFSEventStreamRef streamRef,
642 void* clientCallBackInfo,
size_t numEvents,
644 const FSEventStreamEventFlags eventFlags[],
645 const FSEventStreamEventId eventIds[])
649 char** paths = (
char**)eventPaths;
652 for (
size_t i = 0; i < numEvents; i++)
654 if (strcmp(paths[i],
"/Volumes/") == 0)
656 UINT error = ERROR_CALL_NOT_IMPLEMENTED;
657 if (rdpdr->context.RdpdrHotplugDevice)
658 error = rdpdr->context.RdpdrHotplugDevice(&rdpdr->context,
659 RDPDR_HOTPLUG_CHECK_FOR_CHANGES);
662 case ERROR_DISK_CHANGE:
665 case ERROR_CALL_NOT_IMPLEMENTED:
668 WLog_Print(rdpdr->log, WLOG_ERROR,
669 "handle_hotplug failed with error %" PRIu32
"!", error);
679 UINT error = ERROR_CALL_NOT_IMPLEMENTED;
680 if (rdpdr->context.RdpdrHotplugDevice)
681 error = rdpdr->context.RdpdrHotplugDevice(&rdpdr->context, RDPDR_HOTPLUG_FIRST_CHECK);
685 case ERROR_DISK_CHANGE:
687 case ERROR_CALL_NOT_IMPLEMENTED:
690 WLog_Print(rdpdr->log, WLOG_ERROR,
"handle_hotplug failed with error %" PRIu32
"!",
696static DWORD WINAPI drive_hotplug_thread_func(LPVOID arg)
700 WINPR_ASSERT(rdpdr->stopEvent);
702 CFStringRef path = CFSTR(
"/Volumes/");
703 CFArrayRef pathsToWatch = CFArrayCreate(kCFAllocatorMalloc, (
const void**)&path, 1,
nullptr);
704 FSEventStreamContext ctx = {
705 .copyDescription =
nullptr, .info = arg, .release =
nullptr, .retain =
nullptr, .version = 0
707 FSEventStreamRef fsev =
708 FSEventStreamCreate(kCFAllocatorMalloc, drive_hotplug_fsevent_callback, &ctx, pathsToWatch,
709 kFSEventStreamEventIdSinceNow, 1, kFSEventStreamCreateFlagNone);
711 dispatch_queue_t queue = dispatch_queue_create(TAG,
nullptr);
712 FSEventStreamSetDispatchQueue(fsev, queue);
713 FSEventStreamStart(fsev);
714 WLog_Print(rdpdr->log, WLOG_DEBUG,
"Started hotplug watcher");
715 HANDLE handles[] = { rdpdr->stopEvent, freerdp_abort_event(rdpdr->rdpcontext) };
716 const DWORD status = WaitForMultipleObjects(ARRAYSIZE(handles), handles, FALSE, INFINITE);
717 WLog_Print(rdpdr->log, WLOG_DEBUG,
"Stopped hotplug watcher");
718 FSEventStreamStop(fsev);
719 FSEventStreamRelease(fsev);
720 dispatch_release(queue);
722 UINT error = CHANNEL_RC_OK;
723 if (status > WAIT_OBJECT_0 + ARRAYSIZE(handles))
724 error = ERROR_INTERNAL_ERROR;
731static const char* automountLocations[] = {
"/run/user/%lu/gvfs",
"/run/media/%s",
"/media/%s",
734static BOOL isAutomountLocation(
const char* path)
736 const size_t nrLocations =
sizeof(automountLocations) /
sizeof(automountLocations[0]);
737 char buffer[MAX_PATH] = WINPR_C_ARRAY_INIT;
738 uid_t uid = getuid();
739 char uname[MAX_PATH] = WINPR_C_ARRAY_INIT;
740 ULONG size =
sizeof(uname) - 1;
742 if (!GetUserNameExA(NameSamCompatible, uname, &size))
748 for (
size_t x = 0; x < nrLocations; x++)
750 const char* location = automountLocations[x];
753 WINPR_PRAGMA_DIAG_PUSH
754 WINPR_PRAGMA_DIAG_IGNORED_FORMAT_NONLITERAL
755 if (strstr(location,
"%lu"))
756 (
void)snprintf(buffer,
sizeof(buffer), location, (
unsigned long)uid);
757 else if (strstr(location,
"%s"))
758 (void)snprintf(buffer,
sizeof(buffer), location, uname);
760 (
void)snprintf(buffer,
sizeof(buffer),
"%s", location);
761 WINPR_PRAGMA_DIAG_POP
763 length = strnlen(buffer,
sizeof(buffer));
765 if (strncmp(buffer, path, length) == 0)
767 const char* rest = &path[length];
773 else if (*rest ==
'/')
775 const char* token = strstr(&rest[1],
"/");
777 if (!token || (token[1] ==
'\0'))
786#define MAX_USB_DEVICES 100
794static void handle_mountpoint(hotplug_dev* dev_array,
size_t* size,
const char* mountpoint)
799 if (isAutomountLocation(mountpoint) && (*size < MAX_USB_DEVICES))
801 dev_array[*size].path = _strdup(mountpoint);
802 dev_array[*size].to_add = TRUE;
808#include <sys/mnttab.h>
809static UINT handle_platform_mounts_sun(wLog* log, hotplug_dev* dev_array,
size_t* size)
813 f = winpr_fopen(
"/etc/mnttab",
"r");
816 WLog_Print(log, WLOG_ERROR,
"fopen failed!");
817 return ERROR_OPEN_FAILED;
819 while (getmntent(f, &ent) == 0)
821 handle_mountpoint(dev_array, size, ent.mnt_mountp);
824 return ERROR_SUCCESS;
828#if defined(__FreeBSD__) || defined(__OpenBSD__)
829#include <sys/mount.h>
830static UINT handle_platform_mounts_bsd(wLog* log, hotplug_dev* dev_array,
size_t* size)
833 struct statfs* mntbuf =
nullptr;
835 mntsize = getmntinfo(&mntbuf, MNT_NOWAIT);
839 WLog_Print(log, WLOG_ERROR,
"getmntinfo failed!");
840 return ERROR_OPEN_FAILED;
842 for (
size_t idx = 0; idx < (size_t)mntsize; idx++)
844 handle_mountpoint(dev_array, size, mntbuf[idx].f_mntonname);
847 return ERROR_SUCCESS;
851#if defined(__LINUX__) || defined(__linux__)
853static struct mntent* getmntent_x(FILE* f,
struct mntent* buffer,
char* pathbuffer,
854 size_t pathbuffersize)
856#if defined(FREERDP_HAVE_GETMNTENT_R)
857 WINPR_ASSERT(pathbuffersize <= INT32_MAX);
858 return getmntent_r(f, buffer, pathbuffer, (
int)pathbuffersize);
862 (void)pathbuffersize;
867static UINT handle_platform_mounts_linux(wLog* log, hotplug_dev* dev_array,
size_t* size)
870 struct mntent mnt = WINPR_C_ARRAY_INIT;
871 char pathbuffer[PATH_MAX] = WINPR_C_ARRAY_INIT;
872 struct mntent* ent =
nullptr;
873 f = winpr_fopen(
"/proc/mounts",
"r");
876 WLog_Print(log, WLOG_ERROR,
"fopen failed!");
877 return ERROR_OPEN_FAILED;
879 while ((ent = getmntent_x(f, &mnt, pathbuffer,
sizeof(pathbuffer))) !=
nullptr)
881 handle_mountpoint(dev_array, size, ent->mnt_dir);
884 return ERROR_SUCCESS;
888static UINT handle_platform_mounts(wLog* log, hotplug_dev* dev_array,
size_t* size)
891 return handle_platform_mounts_sun(log, dev_array, size);
892#elif defined(__FreeBSD__) || defined(__OpenBSD__)
893 return handle_platform_mounts_bsd(log, dev_array, size);
894#elif defined(__LINUX__) || defined(__linux__)
895 return handle_platform_mounts_linux(log, dev_array, size);
897 return ERROR_CALL_NOT_IMPLEMENTED;
900static BOOL device_not_plugged(ULONG_PTR key,
void* element,
void* data)
902 const WCHAR* path = (
const WCHAR*)data;
903 DEVICE_DRIVE_EXT* device_ext = (DEVICE_DRIVE_EXT*)element;
908 if (!device_ext || (device_ext->device.type != RDPDR_DTYP_FILESYSTEM) || !device_ext->path)
910 if (_wcscmp(device_ext->path, path) != 0)
915static BOOL device_already_plugged(
rdpdrPlugin* rdpdr,
const hotplug_dev* device)
918 WCHAR* path =
nullptr;
920 if (!rdpdr || !device)
925 WINPR_ASSERT(rdpdr->devman);
926 WINPR_ASSERT(device->path);
928 path = ConvertUtf8ToWCharAlloc(device->path,
nullptr);
932 rc = device_foreach(rdpdr, TRUE, device_not_plugged, path);
937struct hotplug_delete_arg
939 hotplug_dev* dev_array;
940 size_t dev_array_size;
944static BOOL hotplug_delete_foreach(ULONG_PTR key,
void* element,
void* data)
946 char* path =
nullptr;
947 BOOL dev_found = FALSE;
948 struct hotplug_delete_arg* arg = (
struct hotplug_delete_arg*)data;
949 DEVICE_DRIVE_EXT* device_ext = (DEVICE_DRIVE_EXT*)element;
952 WINPR_ASSERT(arg->rdpdr);
953 WINPR_ASSERT(arg->dev_array || (arg->dev_array_size == 0));
954 WINPR_ASSERT(key <= UINT32_MAX);
956 if (!device_ext || (device_ext->device.type != RDPDR_DTYP_FILESYSTEM) || !device_ext->path ||
957 !device_ext->automount)
960 WINPR_ASSERT(device_ext->path);
961 path = ConvertWCharToUtf8Alloc(device_ext->path,
nullptr);
966 if (isAutomountLocation(path))
968 for (
size_t i = 0; i < arg->dev_array_size; i++)
970 hotplug_dev* cur = &arg->dev_array[i];
971 if (cur->path && strstr(path, cur->path) !=
nullptr)
984 const UINT32 ids[1] = { (UINT32)key };
985 WINPR_ASSERT(arg->rdpdr->context.RdpdrUnregisterDevice);
987 arg->rdpdr->context.RdpdrUnregisterDevice(&arg->rdpdr->context, ARRAYSIZE(ids), ids);
991 WLog_Print(arg->rdpdr->log, WLOG_ERROR,
992 "rdpdr_send_device_list_remove_request failed with error %" PRIu32
"!",
1001static UINT handle_hotplug(RdpdrClientContext* context,
1002 WINPR_ATTR_UNUSED RdpdrHotplugEventType type)
1004 WINPR_ASSERT(context);
1007 hotplug_dev dev_array[MAX_USB_DEVICES] = WINPR_C_ARRAY_INIT;
1009 UINT error = ERROR_SUCCESS;
1010 struct hotplug_delete_arg arg = { dev_array, ARRAYSIZE(dev_array), rdpdr };
1012 WINPR_ASSERT(rdpdr);
1013 WINPR_ASSERT(rdpdr->devman);
1015 error = handle_platform_mounts(rdpdr->log, dev_array, &size);
1018 device_foreach(rdpdr, FALSE, hotplug_delete_foreach, &arg);
1021 for (
size_t i = 0; i < size; i++)
1023 hotplug_dev* cur = &dev_array[i];
1024 if (!device_already_plugged(rdpdr, cur))
1026 const char* path = cur->path;
1027 const char* name = strrchr(path,
'/') + 1;
1029 rdpdr_load_drive(rdpdr, name, path, TRUE);
1030 error = ERROR_DISK_CHANGE;
1034 for (
size_t i = 0; i < size; i++)
1035 free(dev_array[i].path);
1042 UINT error = ERROR_CALL_NOT_IMPLEMENTED;
1044 WINPR_ASSERT(rdpdr);
1045 if (rdpdr->context.RdpdrHotplugDevice)
1046 error = rdpdr->context.RdpdrHotplugDevice(&rdpdr->context, RDPDR_HOTPLUG_FIRST_CHECK);
1050 case ERROR_DISK_CHANGE:
1052 case ERROR_OPEN_FAILED:
1053 case ERROR_CALL_NOT_IMPLEMENTED:
1056 WLog_Print(rdpdr->log, WLOG_ERROR,
"handle_hotplug failed with error %" PRIu32
"!",
1062static DWORD WINAPI drive_hotplug_thread_func(LPVOID arg)
1066 WINPR_ASSERT(rdpdr);
1067 WINPR_ASSERT(rdpdr->stopEvent);
1069 while (WaitForSingleObject(rdpdr->stopEvent, 1000) == WAIT_TIMEOUT)
1071 UINT error = ERROR_CALL_NOT_IMPLEMENTED;
1072 if (rdpdr->context.RdpdrHotplugDevice)
1074 rdpdr->context.RdpdrHotplugDevice(&rdpdr->context, RDPDR_HOTPLUG_CHECK_FOR_CHANGES);
1077 case ERROR_DISK_CHANGE:
1080 case ERROR_OPEN_FAILED:
1081 case ERROR_CALL_NOT_IMPLEMENTED:
1084 WLog_Print(rdpdr->log, WLOG_ERROR,
"handle_hotplug failed with error %" PRIu32
"!",
1092 const UINT error = GetLastError();
1093 if (error && rdpdr->rdpcontext)
1094 setChannelError(rdpdr->rdpcontext, error,
"reported an error");
1103#if !defined(_WIN32) && !defined(__IOS__)
1109static UINT drive_hotplug_thread_terminate(
rdpdrPlugin* rdpdr)
1113 WINPR_ASSERT(rdpdr);
1115 if (rdpdr->hotplugThread)
1118 if (rdpdr->stopEvent)
1119 (void)SetEvent(rdpdr->stopEvent);
1122 if (WaitForSingleObject(rdpdr->hotplugThread, INFINITE) == WAIT_FAILED)
1124 error = GetLastError();
1125 WLog_Print(rdpdr->log, WLOG_ERROR,
"WaitForSingleObject failed with error %" PRIu32
"!",
1130 (void)CloseHandle(rdpdr->hotplugThread);
1131 rdpdr->hotplugThread =
nullptr;
1134 return CHANNEL_RC_OK;
1141 WINPR_ASSERT(rdpdr);
1142 WINPR_ASSERT(rdpdr->rdpcontext);
1144 rdpSettings* settings = rdpdr->rdpcontext->settings;
1145 WINPR_ASSERT(settings);
1151 freerdp_settings_get_pointer_array_writable(settings, FreeRDP_DeviceArray, index);
1152 WINPR_ASSERT(device);
1154 if (device->Type == RDPDR_DTYP_FILESYSTEM)
1156 const char DynamicDrives[] =
"DynamicDrives";
1161 const char wildcard[] =
"*";
1162 BOOL hotplugAll = strncmp(drive->Path, wildcard,
sizeof(wildcard)) == 0;
1163 BOOL hotplugLater = strncmp(drive->Path, DynamicDrives,
sizeof(DynamicDrives)) == 0;
1165 if (hotplugAll || hotplugLater)
1169 WLog_Print(rdpdr->log, WLOG_WARN,
1170 "Drive hotplug is not supported in synchronous mode!");
1175 first_hotplug(rdpdr);
1180 if (!rdpdr->hotplugThread)
1182 rdpdr->hotplugThread =
1183 CreateThread(
nullptr, 0, drive_hotplug_thread_func, rdpdr, 0,
nullptr);
1184 if (!rdpdr->hotplugThread)
1186 WLog_Print(rdpdr->log, WLOG_ERROR,
"CreateThread failed!");
1187 return ERROR_INTERNAL_ERROR;
1195 const UINT error = devman_load_device_service(rdpdr->devman, device, rdpdr->rdpcontext);
1198 WLog_Print(rdpdr->log, WLOG_ERROR,
1199 "devman_load_device_service failed with error %" PRIu32
"!", error);
1203 return CHANNEL_RC_OK;
1211static UINT rdpdr_process_connect(
rdpdrPlugin* rdpdr)
1213 WINPR_ASSERT(rdpdr);
1215 rdpdr->devman = devman_new(rdpdr);
1219 WLog_Print(rdpdr->log, WLOG_ERROR,
"devman_new failed!");
1220 return CHANNEL_RC_NO_MEMORY;
1223 WINPR_ASSERT(rdpdr->rdpcontext);
1225 rdpSettings* settings = rdpdr->rdpcontext->settings;
1226 WINPR_ASSERT(settings);
1235 DWORD size = ARRAYSIZE(rdpdr->computerName);
1236 if (!GetComputerNameExA(ComputerNameNetBIOS, rdpdr->computerName, &size))
1237 return ERROR_INTERNAL_ERROR;
1240 strncpy(rdpdr->computerName, name, strnlen(name,
sizeof(rdpdr->computerName)));
1242 return rdpdr_add_devices(rdpdr);
1247 WINPR_ASSERT(rdpdr);
1250 if (!Stream_CheckAndLogRequiredLengthWLog(rdpdr->log, s, 8))
1251 return ERROR_INVALID_DATA;
1253 Stream_Read_UINT16(s, rdpdr->serverVersionMajor);
1254 Stream_Read_UINT16(s, rdpdr->serverVersionMinor);
1255 Stream_Read_UINT32(s, rdpdr->clientID);
1256 rdpdr->sequenceId++;
1258 rdpdr->clientVersionMajor = MIN(RDPDR_VERSION_MAJOR, rdpdr->serverVersionMajor);
1259 rdpdr->clientVersionMinor = MIN(RDPDR_VERSION_MINOR_RDP10X, rdpdr->serverVersionMinor);
1260 WLog_Print(rdpdr->log, WLOG_DEBUG,
1261 "[rdpdr] server announces version %" PRIu32
".%" PRIu32
", client uses %" PRIu32
1263 rdpdr->serverVersionMajor, rdpdr->serverVersionMinor, rdpdr->clientVersionMajor,
1264 rdpdr->clientVersionMinor);
1265 return CHANNEL_RC_OK;
1273static UINT rdpdr_send_client_announce_reply(
rdpdrPlugin* rdpdr)
1275 WINPR_ASSERT(rdpdr);
1276 WINPR_ASSERT(rdpdr->state == RDPDR_CHANNEL_STATE_ANNOUNCE);
1277 if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_ANNOUNCE_REPLY))
1278 return ERROR_INVALID_STATE;
1280 wStream* s = StreamPool_Take(rdpdr->pool, 12);
1284 WLog_Print(rdpdr->log, WLOG_ERROR,
"Stream_New failed!");
1285 return CHANNEL_RC_NO_MEMORY;
1288 Stream_Write_UINT16(s, RDPDR_CTYP_CORE);
1289 Stream_Write_UINT16(s, PAKID_CORE_CLIENTID_CONFIRM);
1290 Stream_Write_UINT16(s, rdpdr->clientVersionMajor);
1291 Stream_Write_UINT16(s, rdpdr->clientVersionMinor);
1292 Stream_Write_UINT32(s, rdpdr->clientID);
1293 return rdpdr_send(rdpdr, s);
1301static UINT rdpdr_send_client_name_request(
rdpdrPlugin* rdpdr)
1304 WCHAR* computerNameW =
nullptr;
1305 size_t computerNameLenW = 0;
1307 WINPR_ASSERT(rdpdr);
1308 WINPR_ASSERT(rdpdr->state == RDPDR_CHANNEL_STATE_ANNOUNCE_REPLY);
1309 if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_NAME_REQUEST))
1310 return ERROR_INVALID_STATE;
1312 const size_t len = strnlen(rdpdr->computerName,
sizeof(rdpdr->computerName));
1314 return ERROR_INTERNAL_ERROR;
1316 WINPR_ASSERT(rdpdr->computerName);
1317 computerNameW = ConvertUtf8NToWCharAlloc(rdpdr->computerName, len, &computerNameLenW);
1318 computerNameLenW *=
sizeof(WCHAR);
1320 if (computerNameLenW > 0)
1321 computerNameLenW +=
sizeof(WCHAR);
1323 s = StreamPool_Take(rdpdr->pool, 16U + computerNameLenW);
1327 free(computerNameW);
1328 WLog_Print(rdpdr->log, WLOG_ERROR,
"Stream_New failed!");
1329 return CHANNEL_RC_NO_MEMORY;
1332 Stream_Write_UINT16(s, RDPDR_CTYP_CORE);
1333 Stream_Write_UINT16(s, PAKID_CORE_CLIENT_NAME);
1334 Stream_Write_UINT32(s, 1);
1335 Stream_Write_UINT32(s, 0);
1336 Stream_Write_UINT32(s,
1337 (UINT32)computerNameLenW);
1338 Stream_Write(s, computerNameW, computerNameLenW);
1339 free(computerNameW);
1340 return rdpdr_send(rdpdr, s);
1345 UINT16 versionMajor = 0;
1346 UINT16 versionMinor = 0;
1347 UINT32 clientID = 0;
1349 WINPR_ASSERT(rdpdr);
1352 if (!Stream_CheckAndLogRequiredLengthWLog(rdpdr->log, s, 8))
1353 return ERROR_INVALID_DATA;
1355 Stream_Read_UINT16(s, versionMajor);
1356 Stream_Read_UINT16(s, versionMinor);
1357 Stream_Read_UINT32(s, clientID);
1359 if (versionMajor != rdpdr->clientVersionMajor || versionMinor != rdpdr->clientVersionMinor)
1361 WLog_Print(rdpdr->log, WLOG_WARN,
1362 "[rdpdr] server announced version %" PRIu32
".%" PRIu32
", client uses %" PRIu32
1363 ".%" PRIu32
" but clientid confirm requests version %" PRIu32
".%" PRIu32,
1364 rdpdr->serverVersionMajor, rdpdr->serverVersionMinor, rdpdr->clientVersionMajor,
1365 rdpdr->clientVersionMinor, versionMajor, versionMinor);
1366 rdpdr->clientVersionMajor = versionMajor;
1367 rdpdr->clientVersionMinor = versionMinor;
1370 if (clientID != rdpdr->clientID)
1371 rdpdr->clientID = clientID;
1373 return CHANNEL_RC_OK;
1376struct device_announce_arg
1384static BOOL device_announce(ULONG_PTR key,
void* element,
void* data)
1386 struct device_announce_arg* arg = data;
1388 DEVICE* device = (DEVICE*)element;
1393 WINPR_ASSERT(device);
1394 WINPR_ASSERT(arg->rdpdr);
1395 WINPR_ASSERT(arg->s);
1406 if ((rdpdr->clientVersionMinor == RDPDR_VERSION_MINOR_RDP51) ||
1407 (device->type == RDPDR_DTYP_SMARTCARD) || arg->userLoggedOn)
1409 size_t data_len = (device->data ==
nullptr ? 0 : Stream_GetPosition(device->data));
1411 if (!Stream_EnsureRemainingCapacity(arg->s, 20 + data_len))
1413 Stream_Release(arg->s);
1414 WLog_Print(rdpdr->log, WLOG_ERROR,
"Stream_EnsureRemainingCapacity failed!");
1418 Stream_Write_UINT32(arg->s, device->type);
1419 Stream_Write_UINT32(arg->s, device->id);
1420 strncpy(Stream_Pointer(arg->s), device->name, 8);
1422 for (
size_t i = 0; i < 8; i++)
1425 Stream_Peek_UINT8(arg->s, c);
1428 Stream_Write_UINT8(arg->s,
'_');
1430 Stream_Seek_UINT8(arg->s);
1433 WINPR_ASSERT(data_len <= UINT32_MAX);
1434 Stream_Write_UINT32(arg->s, (UINT32)data_len);
1437 Stream_Write(arg->s, Stream_Buffer(device->data), data_len);
1440 WLog_Print(rdpdr->log, WLOG_INFO,
1441 "registered [%9s] device #%" PRIu32
": %5s (type=%2" PRIu32
" id=%2" PRIu32
")",
1442 rdpdr_device_type_string(device->type), arg->count, device->name, device->type,
1448static UINT rdpdr_send_device_list_announce_request(
rdpdrPlugin* rdpdr, BOOL userLoggedOn)
1452 size_t count_pos = 0;
1453 struct device_announce_arg arg = WINPR_C_ARRAY_INIT;
1455 WINPR_ASSERT(rdpdr);
1456 WINPR_ASSERT(rdpdr->devman);
1460 rdpdr->userLoggedOn = TRUE;
1463 s = StreamPool_Take(rdpdr->pool, 256);
1467 WLog_Print(rdpdr->log, WLOG_ERROR,
"Stream_New failed!");
1468 return CHANNEL_RC_NO_MEMORY;
1471 Stream_Write_UINT16(s, RDPDR_CTYP_CORE);
1472 Stream_Write_UINT16(s, PAKID_CORE_DEVICELIST_ANNOUNCE);
1473 count_pos = Stream_GetPosition(s);
1474 Stream_Seek_UINT32(s);
1477 arg.userLoggedOn = userLoggedOn;
1479 if (!device_foreach(rdpdr, TRUE, device_announce, &arg))
1480 return ERROR_INVALID_DATA;
1485 return CHANNEL_RC_OK;
1487 pos = Stream_GetPosition(s);
1488 if (!Stream_SetPosition(s, count_pos))
1491 return ERROR_INVALID_DATA;
1493 Stream_Write_UINT32(s, arg.count);
1494 if (!Stream_SetPosition(s, pos))
1497 return ERROR_INVALID_DATA;
1499 Stream_SealLength(s);
1500 return rdpdr_send(rdpdr, s);
1503UINT rdpdr_try_send_device_list_announce_request(
rdpdrPlugin* rdpdr)
1505 WINPR_ASSERT(rdpdr);
1506 if (rdpdr->state != RDPDR_CHANNEL_STATE_READY)
1508 WLog_Print(rdpdr->log, WLOG_DEBUG,
1509 "hotplug event received, but channel [RDPDR] is not ready (state %s), ignoring.",
1510 rdpdr_state_str(rdpdr->state));
1511 return CHANNEL_RC_OK;
1513 return rdpdr_send_device_list_announce_request(rdpdr, rdpdr->userLoggedOn);
1518 WINPR_ASSERT(rdpdr);
1521 wStream* output = StreamPool_Take(rdpdr->pool, 256);
1524 WLog_Print(rdpdr->log, WLOG_ERROR,
"Stream_New failed!");
1525 return CHANNEL_RC_NO_MEMORY;
1528 if (!Stream_SetPosition(s, 4))
1530 Stream_Release(output);
1531 return ERROR_INVALID_DATA;
1534 const uint32_t DeviceId = Stream_Get_UINT32(s);
1535 const uint32_t FileId = Stream_Get_UINT32(s);
1536 const uint32_t CompletionId = Stream_Get_UINT32(s);
1538 WLog_Print(rdpdr->log, WLOG_WARN,
1539 "Dummy response {DeviceId=%" PRIu32
", FileId=%" PRIu32
", CompletionId=%" PRIu32
1541 DeviceId, FileId, CompletionId);
1542 if (!rdpdr_write_iocompletion_header(output, DeviceId, CompletionId, STATUS_UNSUCCESSFUL))
1543 return CHANNEL_RC_NO_MEMORY;
1545 return rdpdr_send(rdpdr, output);
1555 UINT error = CHANNEL_RC_OK;
1557 WINPR_ASSERT(rdpdr);
1560 IRP* irp = irp_new(rdpdr->devman, rdpdr->pool, s, rdpdr->log, &error);
1564 if ((error == CHANNEL_RC_OK) ||
1565 (error == ERROR_DEV_NOT_EXIST && rdpdr->ignoreInvalidDevices))
1567 return dummy_irp_response(rdpdr, s);
1570 WLog_Print(rdpdr->log, WLOG_ERROR,
"irp_new failed with %" PRIu32
"!", error);
1574 if (irp->device->IRPRequest)
1575 error = irp->device->IRPRequest(irp->device, irp);
1577 error = irp->Discard(irp);
1579 if (error != CHANNEL_RC_OK)
1581 WLog_Print(rdpdr->log, WLOG_ERROR,
"device->IRPRequest failed with error %" PRIu32
"",
1588static UINT rdpdr_process_component(
rdpdrPlugin* rdpdr, UINT16 component, UINT16 packetId,
1592 DEVICE* device =
nullptr;
1594 WINPR_ASSERT(rdpdr);
1599 case RDPDR_CTYP_PRN:
1600 type = RDPDR_DTYP_PRINT;
1604 return ERROR_INVALID_DATA;
1607 device = devman_get_device_by_type(rdpdr->devman, type);
1610 return ERROR_DEV_NOT_EXIST;
1612 return IFCALLRESULT(ERROR_INVALID_PARAMETER, device->CustomComponentRequest, device, component,
1621static BOOL device_init(ULONG_PTR key,
void* element,
void* data)
1624 UINT error = CHANNEL_RC_OK;
1625 DEVICE* device = element;
1630 IFCALLRET(device->Init, error, device);
1632 if (error != CHANNEL_RC_OK)
1634 WLog_Print(log, WLOG_ERROR,
"Device init failed with %s", WTSErrorToString(error));
1642 WINPR_ASSERT(rdpdr);
1643 WINPR_ASSERT(rdpdr->devman);
1645 rdpdr->userLoggedOn = FALSE;
1646 if (!device_foreach(rdpdr, TRUE, device_init, rdpdr->log))
1647 return ERROR_INTERNAL_ERROR;
1648 return CHANNEL_RC_OK;
1651static BOOL state_match(
enum RDPDR_CHANNEL_STATE state,
size_t count, va_list ap)
1653 for (
size_t x = 0; x < count; x++)
1655 enum RDPDR_CHANNEL_STATE cur = va_arg(ap,
enum RDPDR_CHANNEL_STATE);
1662static const char* state_str(
size_t count, va_list ap,
char* buffer,
size_t size)
1664 for (
size_t x = 0; x < count; x++)
1666 enum RDPDR_CHANNEL_STATE cur = va_arg(ap,
enum RDPDR_CHANNEL_STATE);
1667 const char* curstr = rdpdr_state_str(cur);
1668 winpr_str_append(curstr, buffer, size,
"|");
1673static BOOL rdpdr_state_check(
rdpdrPlugin* rdpdr, UINT16 packetid,
enum RDPDR_CHANNEL_STATE next,
1676 va_list ap = WINPR_C_ARRAY_INIT;
1677 WINPR_ASSERT(rdpdr);
1679 va_start(ap, count);
1680 BOOL rc = state_match(rdpdr->state, count, ap);
1685 const char* strstate = rdpdr_state_str(rdpdr->state);
1686 char buffer[256] = WINPR_C_ARRAY_INIT;
1688 va_start(ap, count);
1689 state_str(count, ap, buffer,
sizeof(buffer));
1692 WLog_Print(rdpdr->log, WLOG_ERROR,
1693 "channel [RDPDR] received %s, expected states [%s] but have state %s, aborting.",
1694 rdpdr_packetid_string(packetid), buffer, strstate);
1696 if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_INITIAL))
1700 return rdpdr_state_advance(rdpdr, next);
1703static BOOL rdpdr_check_channel_state(
rdpdrPlugin* rdpdr, UINT16 packetid)
1705 WINPR_ASSERT(rdpdr);
1709 case PAKID_CORE_SERVER_ANNOUNCE:
1714 if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_INITIAL))
1716 return rdpdr_state_check(rdpdr, packetid, RDPDR_CHANNEL_STATE_ANNOUNCE, 1,
1717 RDPDR_CHANNEL_STATE_INITIAL);
1718 case PAKID_CORE_SERVER_CAPABILITY:
1719 return rdpdr_state_check(
1720 rdpdr, packetid, RDPDR_CHANNEL_STATE_SERVER_CAPS, 6,
1721 RDPDR_CHANNEL_STATE_NAME_REQUEST, RDPDR_CHANNEL_STATE_SERVER_CAPS,
1722 RDPDR_CHANNEL_STATE_READY, RDPDR_CHANNEL_STATE_CLIENT_CAPS,
1723 RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM, RDPDR_CHANNEL_STATE_USER_LOGGEDON);
1724 case PAKID_CORE_CLIENTID_CONFIRM:
1725 return rdpdr_state_check(rdpdr, packetid, RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM, 5,
1726 RDPDR_CHANNEL_STATE_NAME_REQUEST,
1727 RDPDR_CHANNEL_STATE_SERVER_CAPS,
1728 RDPDR_CHANNEL_STATE_CLIENT_CAPS, RDPDR_CHANNEL_STATE_READY,
1729 RDPDR_CHANNEL_STATE_USER_LOGGEDON);
1730 case PAKID_CORE_USER_LOGGEDON:
1731 if (!rdpdr_check_extended_pdu_flag(rdpdr, RDPDR_USER_LOGGEDON_PDU))
1734 return rdpdr_state_check(
1735 rdpdr, packetid, RDPDR_CHANNEL_STATE_USER_LOGGEDON, 4,
1736 RDPDR_CHANNEL_STATE_NAME_REQUEST, RDPDR_CHANNEL_STATE_CLIENT_CAPS,
1737 RDPDR_CHANNEL_STATE_CLIENTID_CONFIRM, RDPDR_CHANNEL_STATE_READY);
1740 enum RDPDR_CHANNEL_STATE state = RDPDR_CHANNEL_STATE_READY;
1741 return rdpdr_state_check(rdpdr, packetid, state, 1, state);
1748 if (rdpdr->haveClientId && rdpdr->haveServerCaps)
1750 const UINT error = rdpdr_send_device_list_announce_request(rdpdr, FALSE);
1753 WLog_Print(rdpdr->log, WLOG_ERROR,
1754 "rdpdr_send_device_list_announce_request failed with error %" PRIu32
"",
1758 if (!rdpdr_state_advance(rdpdr, RDPDR_CHANNEL_STATE_READY))
1771 UINT16 component = 0;
1772 UINT16 packetId = 0;
1773 UINT32 deviceId = 0;
1775 UINT error = ERROR_INVALID_DATA;
1778 return CHANNEL_RC_NULL_DATA;
1780 rdpdr_dump_received_packet(rdpdr->log, WLOG_TRACE, s,
"[rdpdr-channel] receive");
1781 if (Stream_GetRemainingLength(s) >= 4)
1783 Stream_Read_UINT16(s, component);
1784 Stream_Read_UINT16(s, packetId);
1786 if (component == RDPDR_CTYP_CORE)
1788 if (!rdpdr_check_channel_state(rdpdr, packetId))
1789 return CHANNEL_RC_OK;
1793 case PAKID_CORE_SERVER_ANNOUNCE:
1794 rdpdr->haveClientId = FALSE;
1795 rdpdr->haveServerCaps = FALSE;
1796 if ((error = rdpdr_process_server_announce_request(rdpdr, s)))
1799 else if ((error = rdpdr_send_client_announce_reply(rdpdr)))
1801 WLog_Print(rdpdr->log, WLOG_ERROR,
1802 "rdpdr_send_client_announce_reply failed with error %" PRIu32
"",
1805 else if ((error = rdpdr_send_client_name_request(rdpdr)))
1807 WLog_Print(rdpdr->log, WLOG_ERROR,
1808 "rdpdr_send_client_name_request failed with error %" PRIu32
"",
1811 else if ((error = rdpdr_process_init(rdpdr)))
1813 WLog_Print(rdpdr->log, WLOG_ERROR,
1814 "rdpdr_process_init failed with error %" PRIu32
"", error);
1819 case PAKID_CORE_SERVER_CAPABILITY:
1820 if ((error = rdpdr_process_capability_request(rdpdr, s)))
1823 else if ((error = rdpdr_send_capability_response(rdpdr)))
1825 WLog_Print(rdpdr->log, WLOG_ERROR,
1826 "rdpdr_send_capability_response failed with error %" PRIu32
"",
1831 rdpdr->haveServerCaps = TRUE;
1832 if (!tryAdvance(rdpdr))
1833 error = ERROR_INTERNAL_ERROR;
1838 case PAKID_CORE_CLIENTID_CONFIRM:
1839 if ((error = rdpdr_process_server_clientid_confirm(rdpdr, s)))
1844 rdpdr->haveClientId = TRUE;
1845 if (!tryAdvance(rdpdr))
1846 error = ERROR_INTERNAL_ERROR;
1850 case PAKID_CORE_USER_LOGGEDON:
1851 if (!rdpdr->haveServerCaps)
1853 WLog_Print(rdpdr->log, WLOG_ERROR,
1854 "Wrong state %s for %s. [serverCaps=%d, clientId=%d]",
1855 rdpdr_state_str(rdpdr->state), rdpdr_packetid_string(packetId),
1856 rdpdr->haveServerCaps, rdpdr->haveClientId);
1857 error = ERROR_INTERNAL_ERROR;
1859 else if ((error = rdpdr_send_device_list_announce_request(rdpdr, TRUE)))
1862 rdpdr->log, WLOG_ERROR,
1863 "rdpdr_send_device_list_announce_request failed with error %" PRIu32
"",
1866 else if (!tryAdvance(rdpdr))
1868 error = ERROR_INTERNAL_ERROR;
1873 case PAKID_CORE_DEVICE_REPLY:
1876 if (Stream_GetRemainingLength(s) >= 8)
1878 Stream_Read_UINT32(s, deviceId);
1879 Stream_Read_UINT32(s, status);
1882 devman_unregister_device(rdpdr->devman, (
void*)((
size_t)deviceId));
1883 error = CHANNEL_RC_OK;
1888 case PAKID_CORE_DEVICE_IOREQUEST:
1889 if ((error = rdpdr_process_irp(rdpdr, s)))
1891 WLog_Print(rdpdr->log, WLOG_ERROR,
1892 "rdpdr_process_irp failed with error %" PRIu32
"", error);
1901 WLog_Print(rdpdr->log, WLOG_ERROR,
1902 "RDPDR_CTYP_CORE unknown PacketId: 0x%04" PRIX16
"", packetId);
1903 error = ERROR_INVALID_DATA;
1909 error = rdpdr_process_component(rdpdr, component, packetId, s);
1911 if (error != CHANNEL_RC_OK)
1913 DWORD level = WLOG_ERROR;
1914 if (rdpdr->ignoreInvalidDevices)
1916 if (error == ERROR_DEV_NOT_EXIST)
1919 error = CHANNEL_RC_OK;
1922 WLog_Print(rdpdr->log, level,
1923 "Unknown message: Component: %s [0x%04" PRIX16
1924 "] PacketId: %s [0x%04" PRIX16
"]",
1925 rdpdr_component_string(component), component,
1926 rdpdr_packetid_string(packetId), packetId);
1946 return CHANNEL_RC_NULL_DATA;
1952 return CHANNEL_RC_BAD_INIT_HANDLE;
1955 const size_t pos = Stream_GetPosition(s);
1956 UINT status = ERROR_INTERNAL_ERROR;
1957 if (pos <= UINT32_MAX)
1959 rdpdr_dump_send_packet(rdpdr->log, WLOG_TRACE, s,
"[rdpdr-channel] send");
1960 status = plugin->channelEntryPoints.pVirtualChannelWriteEx(
1961 plugin->InitHandle, plugin->OpenHandle, Stream_Buffer(s), (UINT32)pos, s);
1964 if (status != CHANNEL_RC_OK)
1967 WLog_Print(rdpdr->log, WLOG_ERROR,
"pVirtualChannelWriteEx failed with %s [%08" PRIX32
"]",
1968 WTSErrorToString(status), status);
1979static UINT rdpdr_virtual_channel_event_data_received(
rdpdrPlugin* rdpdr,
void* pData,
1980 UINT32 dataLength, UINT32 totalLength,
1985 WINPR_ASSERT(rdpdr);
1986 WINPR_ASSERT(pData || (dataLength == 0));
1988 if ((dataFlags & CHANNEL_FLAG_SUSPEND) || (dataFlags & CHANNEL_FLAG_RESUME))
1996 return CHANNEL_RC_OK;
1999 if (dataFlags & CHANNEL_FLAG_FIRST)
2001 if (rdpdr->data_in !=
nullptr)
2002 Stream_Release(rdpdr->data_in);
2004 rdpdr->data_in = StreamPool_Take(rdpdr->pool, totalLength);
2006 if (!rdpdr->data_in)
2008 WLog_Print(rdpdr->log, WLOG_ERROR,
"Stream_New failed!");
2009 return CHANNEL_RC_NO_MEMORY;
2013 data_in = rdpdr->data_in;
2015 if (!Stream_EnsureRemainingCapacity(data_in, dataLength))
2017 WLog_Print(rdpdr->log, WLOG_ERROR,
"Stream_EnsureRemainingCapacity failed!");
2018 return ERROR_INVALID_DATA;
2021 Stream_Write(data_in, pData, dataLength);
2023 if (dataFlags & CHANNEL_FLAG_LAST)
2025 const size_t pos = Stream_GetPosition(data_in);
2026 const size_t cap = Stream_Capacity(data_in);
2029 WLog_Print(rdpdr->log, WLOG_ERROR,
2030 "rdpdr_virtual_channel_event_data_received: read error");
2031 return ERROR_INTERNAL_ERROR;
2034 Stream_SealLength(data_in);
2035 Stream_ResetPosition(data_in);
2039 if (!MessageQueue_Post(rdpdr->queue,
nullptr, 0, (
void*)data_in,
nullptr))
2041 WLog_Print(rdpdr->log, WLOG_ERROR,
"MessageQueue_Post failed!");
2042 return ERROR_INTERNAL_ERROR;
2044 rdpdr->data_in =
nullptr;
2048 UINT error = rdpdr_process_receive(rdpdr, data_in);
2049 Stream_Release(data_in);
2050 rdpdr->data_in =
nullptr;
2056 return CHANNEL_RC_OK;
2059static VOID VCAPITYPE rdpdr_virtual_channel_open_event_ex(LPVOID lpUserParam, DWORD openHandle,
2060 UINT event, LPVOID pData,
2061 UINT32 dataLength, UINT32 totalLength,
2064 UINT error = CHANNEL_RC_OK;
2067 WINPR_ASSERT(rdpdr);
2070 case CHANNEL_EVENT_DATA_RECEIVED:
2071 if (!rdpdr || !pData || (rdpdr->OpenHandle != openHandle))
2073 WLog_Print(rdpdr->log, WLOG_ERROR,
"error no match");
2076 if ((error = rdpdr_virtual_channel_event_data_received(rdpdr, pData, dataLength,
2077 totalLength, dataFlags)))
2078 WLog_Print(rdpdr->log, WLOG_ERROR,
2079 "rdpdr_virtual_channel_event_data_received failed with error %" PRIu32
2085 case CHANNEL_EVENT_WRITE_CANCELLED:
2086 case CHANNEL_EVENT_WRITE_COMPLETE:
2093 case CHANNEL_EVENT_USER:
2099 if (error && rdpdr && rdpdr->rdpcontext)
2100 setChannelError(rdpdr->rdpcontext, error,
2101 "rdpdr_virtual_channel_open_event_ex reported an error");
2104static DWORD WINAPI rdpdr_virtual_channel_client_thread(LPVOID arg)
2111 ExitThread((DWORD)CHANNEL_RC_NULL_DATA);
2112 return CHANNEL_RC_NULL_DATA;
2115 if ((error = rdpdr_process_connect(rdpdr)))
2117 WLog_Print(rdpdr->log, WLOG_ERROR,
"rdpdr_process_connect failed with error %" PRIu32
"!",
2120 if (rdpdr->rdpcontext)
2121 setChannelError(rdpdr->rdpcontext, error,
2122 "rdpdr_virtual_channel_client_thread reported an error");
2130 wMessage message = WINPR_C_ARRAY_INIT;
2131 WINPR_ASSERT(rdpdr);
2133 if (!MessageQueue_Wait(rdpdr->queue))
2136 if (MessageQueue_Peek(rdpdr->queue, &message, TRUE))
2138 if (message.id == WMQ_QUIT)
2141 if (message.id == 0)
2145 error = rdpdr_process_receive(rdpdr, data);
2147 Stream_Release(data);
2150 WLog_Print(rdpdr->log, WLOG_ERROR,
2151 "rdpdr_process_receive failed with error %" PRIu32
"!", error);
2153 if (rdpdr->rdpcontext)
2154 setChannelError(rdpdr->rdpcontext, error,
2155 "rdpdr_virtual_channel_client_thread reported an error");
2164 if ((error = drive_hotplug_thread_terminate(rdpdr)))
2165 WLog_Print(rdpdr->log, WLOG_ERROR,
2166 "drive_hotplug_thread_terminate failed with error %" PRIu32
"!", error);
2172static void queue_free(
void* obj)
2175 wMessage* msg = (wMessage*)obj;
2177 if (!msg || (msg->id != 0))
2190static UINT rdpdr_virtual_channel_event_connected(
rdpdrPlugin* rdpdr, LPVOID pData,
2195 WINPR_ASSERT(rdpdr);
2196 WINPR_UNUSED(pData);
2197 WINPR_UNUSED(dataLength);
2201 rdpdr->queue = MessageQueue_New(
nullptr);
2205 WLog_Print(rdpdr->log, WLOG_ERROR,
"MessageQueue_New failed!");
2206 return CHANNEL_RC_NO_MEMORY;
2209 obj = MessageQueue_Object(rdpdr->queue);
2212 if (!(rdpdr->thread = CreateThread(
nullptr, 0, rdpdr_virtual_channel_client_thread,
2213 (
void*)rdpdr, 0,
nullptr)))
2215 WLog_Print(rdpdr->log, WLOG_ERROR,
"CreateThread failed!");
2216 return ERROR_INTERNAL_ERROR;
2221 UINT error = rdpdr_process_connect(rdpdr);
2224 WLog_Print(rdpdr->log, WLOG_ERROR,
2225 "rdpdr_process_connect failed with error %" PRIu32
"!", error);
2230 return rdpdr->channelEntryPoints.pVirtualChannelOpenEx(rdpdr->InitHandle, &rdpdr->OpenHandle,
2231 rdpdr->channelDef.name,
2232 rdpdr_virtual_channel_open_event_ex);
2240static UINT rdpdr_virtual_channel_event_disconnected(
rdpdrPlugin* rdpdr)
2244 WINPR_ASSERT(rdpdr);
2246 if (rdpdr->OpenHandle == 0)
2247 return CHANNEL_RC_OK;
2249 if (rdpdr->queue && rdpdr->thread)
2251 if (MessageQueue_PostQuit(rdpdr->queue, 0) &&
2252 (WaitForSingleObject(rdpdr->thread, INFINITE) == WAIT_FAILED))
2254 error = GetLastError();
2255 WLog_Print(rdpdr->log, WLOG_ERROR,
"WaitForSingleObject failed with error %" PRIu32
"!",
2262 (void)CloseHandle(rdpdr->thread);
2263 MessageQueue_Free(rdpdr->queue);
2264 rdpdr->queue =
nullptr;
2265 rdpdr->thread =
nullptr;
2267 WINPR_ASSERT(rdpdr->channelEntryPoints.pVirtualChannelCloseEx);
2268 error = rdpdr->channelEntryPoints.pVirtualChannelCloseEx(rdpdr->InitHandle, rdpdr->OpenHandle);
2270 if (CHANNEL_RC_OK != error)
2272 WLog_Print(rdpdr->log, WLOG_ERROR,
"pVirtualChannelCloseEx failed with %s [%08" PRIX32
"]",
2273 WTSErrorToString(error), error);
2276 rdpdr->OpenHandle = 0;
2280 Stream_Release(rdpdr->data_in);
2281 rdpdr->data_in =
nullptr;
2286 devman_free(rdpdr->devman);
2287 rdpdr->devman =
nullptr;
2293static void rdpdr_virtual_channel_event_terminated(
rdpdrPlugin* rdpdr)
2295 WINPR_ASSERT(rdpdr);
2297 if (rdpdr->stopEvent)
2299 (void)CloseHandle(rdpdr->stopEvent);
2300 rdpdr->stopEvent =
nullptr;
2303 rdpdr->InitHandle =
nullptr;
2304 StreamPool_Free(rdpdr->pool);
2308static UINT rdpdr_register_device(RdpdrClientContext* context,
const RDPDR_DEVICE* device,
2311 WINPR_ASSERT(context);
2312 WINPR_ASSERT(device);
2316 WINPR_ASSERT(rdpdr);
2320 return ERROR_INVALID_DATA;
2321 UINT rc = devman_load_device_service(rdpdr->devman, copy, rdpdr->rdpcontext);
2323 freerdp_device_free(copy);
2324 if (rc == CHANNEL_RC_OK)
2325 rc = rdpdr_try_send_device_list_announce_request(rdpdr);
2329static UINT rdpdr_unregister_device(RdpdrClientContext* context,
size_t count,
const uint32_t ids[])
2331 WINPR_ASSERT(context);
2334 WINPR_ASSERT(rdpdr);
2336 for (
size_t x = 0; x < count; x++)
2338 const uintptr_t
id = ids[x];
2339 devman_unregister_device(rdpdr->devman, (
void*)
id);
2341 return rdpdr_send_device_list_remove_request(rdpdr, WINPR_ASSERTING_INT_CAST(uint32_t, count),
2345static UINT rdpdr_virtual_channel_event_initialized(
rdpdrPlugin* rdpdr,
2346 WINPR_ATTR_UNUSED LPVOID pData,
2347 WINPR_ATTR_UNUSED UINT32 dataLength)
2349 WINPR_ASSERT(rdpdr);
2351 WINPR_ASSERT(!rdpdr->stopEvent);
2352 rdpdr->stopEvent = CreateEvent(
nullptr, TRUE, FALSE,
nullptr);
2353 WINPR_ASSERT(rdpdr->stopEvent);
2356 rdpdr->context.handle = rdpdr;
2357 rdpdr->context.RdpdrHotplugDevice = handle_hotplug;
2358 rdpdr->context.RdpdrRegisterDevice = rdpdr_register_device;
2359 rdpdr->context.RdpdrUnregisterDevice = rdpdr_unregister_device;
2360 return CHANNEL_RC_OK;
2363static VOID VCAPITYPE rdpdr_virtual_channel_init_event_ex(LPVOID lpUserParam, LPVOID pInitHandle,
2364 UINT event, LPVOID pData, UINT dataLength)
2366 UINT error = CHANNEL_RC_OK;
2369 if (!rdpdr || (rdpdr->InitHandle != pInitHandle))
2371 WLog_ERR(TAG,
"error no match");
2375 WINPR_ASSERT(pData || (dataLength == 0));
2379 case CHANNEL_EVENT_INITIALIZED:
2380 error = rdpdr_virtual_channel_event_initialized(rdpdr, pData, dataLength);
2383 case CHANNEL_EVENT_CONNECTED:
2384 if ((error = rdpdr_virtual_channel_event_connected(rdpdr, pData, dataLength)))
2385 WLog_Print(rdpdr->log, WLOG_ERROR,
2386 "rdpdr_virtual_channel_event_connected failed with error %" PRIu32
"!",
2391 case CHANNEL_EVENT_DISCONNECTED:
2392 if ((error = rdpdr_virtual_channel_event_disconnected(rdpdr)))
2393 WLog_Print(rdpdr->log, WLOG_ERROR,
2394 "rdpdr_virtual_channel_event_disconnected failed with error %" PRIu32
2400 case CHANNEL_EVENT_TERMINATED:
2401 rdpdr_virtual_channel_event_terminated(rdpdr);
2405 case CHANNEL_EVENT_ATTACHED:
2406 case CHANNEL_EVENT_DETACHED:
2408 WLog_Print(rdpdr->log, WLOG_ERROR,
"unknown event %" PRIu32
"!", event);
2412 if (error && rdpdr && rdpdr->rdpcontext)
2413 setChannelError(rdpdr->rdpcontext, error,
2414 "rdpdr_virtual_channel_init_event_ex reported an error");
2418#define VirtualChannelEntryEx rdpdr_VirtualChannelEntryEx
2420FREERDP_ENTRY_POINT(BOOL VCAPITYPE VirtualChannelEntryEx(PCHANNEL_ENTRY_POINTS_EX pEntryPoints,
2423 WINPR_ASSERT(pEntryPoints);
2424 WINPR_ASSERT(pInitHandle);
2430 WLog_ERR(TAG,
"calloc failed!");
2433 rdpdr->log = WLog_Get(TAG);
2435 rdpdr->clientExtendedPDU =
2436 RDPDR_DEVICE_REMOVE_PDUS | RDPDR_CLIENT_DISPLAY_NAME_PDU | RDPDR_USER_LOGGEDON_PDU;
2437 rdpdr->clientIOCode1 =
2438 RDPDR_IRP_MJ_CREATE | RDPDR_IRP_MJ_CLEANUP | RDPDR_IRP_MJ_CLOSE | RDPDR_IRP_MJ_READ |
2439 RDPDR_IRP_MJ_WRITE | RDPDR_IRP_MJ_FLUSH_BUFFERS | RDPDR_IRP_MJ_SHUTDOWN |
2440 RDPDR_IRP_MJ_DEVICE_CONTROL | RDPDR_IRP_MJ_QUERY_VOLUME_INFORMATION |
2441 RDPDR_IRP_MJ_SET_VOLUME_INFORMATION | RDPDR_IRP_MJ_QUERY_INFORMATION |
2442 RDPDR_IRP_MJ_SET_INFORMATION | RDPDR_IRP_MJ_DIRECTORY_CONTROL | RDPDR_IRP_MJ_LOCK_CONTROL |
2443 RDPDR_IRP_MJ_QUERY_SECURITY | RDPDR_IRP_MJ_SET_SECURITY;
2445 rdpdr->clientExtraFlags1 = ENABLE_ASYNCIO;
2447 rdpdr->pool = StreamPool_New(TRUE, 1024);
2454 rdpdr->channelDef.options =
2455 CHANNEL_OPTION_INITIALIZED | CHANNEL_OPTION_ENCRYPT_RDP | CHANNEL_OPTION_COMPRESS_RDP;
2456 (void)sprintf_s(rdpdr->channelDef.name, ARRAYSIZE(rdpdr->channelDef.name),
2457 RDPDR_SVC_CHANNEL_NAME);
2458 rdpdr->sequenceId = 0;
2463 (pEntryPointsEx->MagicNumber == FREERDP_CHANNEL_MAGIC_NUMBER))
2465 rdpdr->rdpcontext = pEntryPointsEx->context;
2467 FreeRDP_SynchronousStaticChannels))
2468 rdpdr->async = TRUE;
2472 rdpdr->InitHandle = pInitHandle;
2473 const UINT rc = rdpdr->channelEntryPoints.pVirtualChannelInitEx(
2474 rdpdr, &rdpdr->context, pInitHandle, &rdpdr->channelDef, 1, VIRTUAL_CHANNEL_VERSION_WIN2000,
2475 rdpdr_virtual_channel_init_event_ex);
2477 if (CHANNEL_RC_OK != rc)
2479 WLog_Print(rdpdr->log, WLOG_ERROR,
"pVirtualChannelInitEx failed with %s [%08" PRIX32
"]",
2480 WTSErrorToString(rc), rc);
WINPR_ATTR_NODISCARD FREERDP_API const char * freerdp_settings_get_string(const rdpSettings *settings, FreeRDP_Settings_Keys_String id)
Returns a immutable string settings value.
WINPR_ATTR_NODISCARD FREERDP_API UINT32 freerdp_settings_get_uint32(const rdpSettings *settings, FreeRDP_Settings_Keys_UInt32 id)
Returns a UINT32 settings value.
WINPR_ATTR_NODISCARD FREERDP_API BOOL freerdp_settings_get_bool(const rdpSettings *settings, FreeRDP_Settings_Keys_Bool id)
Returns a boolean settings value.
This struct contains function pointer to initialize/free objects.
OBJECT_FREE_FN fnObjectFree