24#include <freerdp/config.h>
32#include <winpr/assert.h>
33#include <winpr/path.h>
34#include <winpr/file.h>
35#include <winpr/string.h>
36#include <winpr/synch.h>
37#include <winpr/thread.h>
38#include <winpr/stream.h>
39#include <winpr/environment.h>
40#include <winpr/interlocked.h>
41#include <winpr/collections.h>
42#include <winpr/shell.h>
44#include <freerdp/freerdp.h>
45#include <freerdp/channels/rdpdr.h>
47#include "drive_file.h"
56 wListDictionary* files;
60 wMessageQueue* IrpQueue;
64 rdpContext* rdpcontext;
67static NTSTATUS drive_map_windows_err(DWORD fs_errno)
79 case ERROR_ACCESS_DENIED:
80 case ERROR_SHARING_VIOLATION:
81 rc = STATUS_ACCESS_DENIED;
84 case ERROR_FILE_NOT_FOUND:
85 rc = STATUS_NO_SUCH_FILE;
88 case ERROR_BUSY_DRIVE:
89 rc = STATUS_DEVICE_BUSY;
92 case ERROR_INVALID_DRIVE:
93 rc = STATUS_NO_SUCH_DEVICE;
97 rc = STATUS_NO_SUCH_DEVICE;
100 case ERROR_FILE_EXISTS:
101 case ERROR_ALREADY_EXISTS:
102 rc = STATUS_OBJECT_NAME_COLLISION;
105 case ERROR_INVALID_NAME:
106 rc = STATUS_NO_SUCH_FILE;
109 case ERROR_INVALID_HANDLE:
110 rc = STATUS_INVALID_HANDLE;
113 case ERROR_NO_MORE_FILES:
114 rc = STATUS_NO_MORE_FILES;
117 case ERROR_DIRECTORY:
118 rc = STATUS_NOT_A_DIRECTORY;
121 case ERROR_PATH_NOT_FOUND:
122 rc = STATUS_OBJECT_PATH_NOT_FOUND;
125 case ERROR_DIR_NOT_EMPTY:
126 rc = STATUS_DIRECTORY_NOT_EMPTY;
130 rc = STATUS_UNSUCCESSFUL;
131 WLog_ERR(TAG,
"Error code not found: %" PRIu32
"", fs_errno);
138static DRIVE_FILE* drive_get_file_by_id(DRIVE_DEVICE* drive, UINT32
id)
141 void* key = (
void*)(
size_t)id;
146 file = (
DRIVE_FILE*)ListDictionary_GetItemValue(drive->files, key);
155static UINT drive_process_irp_create(DRIVE_DEVICE* drive, IRP* irp)
159 BYTE Information = 0;
160 const WCHAR* path = NULL;
162 if (!drive || !irp || !irp->devman)
163 return ERROR_INVALID_PARAMETER;
165 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 6 * 4 + 8))
166 return ERROR_INVALID_DATA;
168 const uint32_t DesiredAccess = Stream_Get_UINT32(irp->input);
169 const uint64_t allocationSize = Stream_Get_UINT64(irp->input);
170 const uint32_t FileAttributes = Stream_Get_UINT32(irp->input);
171 const uint32_t SharedAccess = Stream_Get_UINT32(irp->input);
172 const uint32_t CreateDisposition = Stream_Get_UINT32(irp->input);
173 const uint32_t CreateOptions = Stream_Get_UINT32(irp->input);
174 const uint32_t PathLength = Stream_Get_UINT32(irp->input);
176 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, PathLength))
177 return ERROR_INVALID_DATA;
179 path = Stream_ConstPointer(irp->input);
180 FileId = irp->devman->id_sequence++;
181 file = drive_file_new(drive->path, path, PathLength /
sizeof(WCHAR), FileId, DesiredAccess,
182 CreateDisposition, CreateOptions, FileAttributes, SharedAccess);
186 irp->IoStatus = drive_map_windows_err(GetLastError());
192 void* key = (
void*)(
size_t)file->id;
194 if (!ListDictionary_Add(drive->files, key, file))
196 WLog_ERR(TAG,
"ListDictionary_Add failed!");
197 return ERROR_INTERNAL_ERROR;
200 switch (CreateDisposition)
206 Information = FILE_SUPERSEDED;
210 Information = FILE_OPENED;
213 case FILE_OVERWRITE_IF:
214 Information = FILE_OVERWRITTEN;
222 if (allocationSize > 0)
224 const BYTE buffer[] = {
'\0' };
225 if (!drive_file_seek(file, allocationSize -
sizeof(buffer)))
226 return ERROR_INTERNAL_ERROR;
227 if (!drive_file_write(file, buffer,
sizeof(buffer)))
228 return ERROR_INTERNAL_ERROR;
232 Stream_Write_UINT32(irp->output, FileId);
233 Stream_Write_UINT8(irp->output, Information);
235 WINPR_ASSERT(irp->Complete);
236 return irp->Complete(irp);
244static UINT drive_process_irp_close(DRIVE_DEVICE* drive, IRP* irp)
249 if (!drive || !irp || !irp->output)
250 return ERROR_INVALID_PARAMETER;
252 file = drive_get_file_by_id(drive, irp->FileId);
253 key = (
void*)(
size_t)irp->FileId;
256 irp->IoStatus = STATUS_UNSUCCESSFUL;
259 ListDictionary_Take(drive->files, key);
261 if (drive_file_free(file))
262 irp->IoStatus = STATUS_SUCCESS;
264 irp->IoStatus = drive_map_windows_err(GetLastError());
267 Stream_Zero(irp->output, 5);
269 WINPR_ASSERT(irp->Complete);
270 return irp->Complete(irp);
278static UINT drive_process_irp_read(DRIVE_DEVICE* drive, IRP* irp)
284 if (!drive || !irp || !irp->output)
285 return ERROR_INVALID_PARAMETER;
287 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 12))
288 return ERROR_INVALID_DATA;
290 Stream_Read_UINT32(irp->input, Length);
291 Stream_Read_UINT64(irp->input, Offset);
292 file = drive_get_file_by_id(drive, irp->FileId);
296 irp->IoStatus = STATUS_UNSUCCESSFUL;
299 else if (!drive_file_seek(file, Offset))
301 irp->IoStatus = drive_map_windows_err(GetLastError());
305 if (!Stream_EnsureRemainingCapacity(irp->output, Length + 4))
307 WLog_ERR(TAG,
"Stream_EnsureRemainingCapacity failed!");
308 return ERROR_INTERNAL_ERROR;
310 else if (Length == 0)
311 Stream_Write_UINT32(irp->output, 0);
314 BYTE* buffer = Stream_PointerAs(irp->output, BYTE) +
sizeof(UINT32);
316 if (!drive_file_read(file, buffer, &Length))
318 irp->IoStatus = drive_map_windows_err(GetLastError());
319 Stream_Write_UINT32(irp->output, 0);
323 Stream_Write_UINT32(irp->output, Length);
324 Stream_Seek(irp->output, Length);
328 WINPR_ASSERT(irp->Complete);
329 return irp->Complete(irp);
337static UINT drive_process_irp_write(DRIVE_DEVICE* drive, IRP* irp)
343 if (!drive || !irp || !irp->input || !irp->output)
344 return ERROR_INVALID_PARAMETER;
346 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 32))
347 return ERROR_INVALID_DATA;
349 Stream_Read_UINT32(irp->input, Length);
350 Stream_Read_UINT64(irp->input, Offset);
351 Stream_Seek(irp->input, 20);
352 const void* ptr = Stream_ConstPointer(irp->input);
353 if (!Stream_SafeSeek(irp->input, Length))
354 return ERROR_INVALID_DATA;
355 file = drive_get_file_by_id(drive, irp->FileId);
359 irp->IoStatus = STATUS_UNSUCCESSFUL;
362 else if (!drive_file_seek(file, Offset))
364 irp->IoStatus = drive_map_windows_err(GetLastError());
367 else if (!drive_file_write(file, ptr, Length))
369 irp->IoStatus = drive_map_windows_err(GetLastError());
373 Stream_Write_UINT32(irp->output, Length);
374 Stream_Write_UINT8(irp->output, 0);
376 WINPR_ASSERT(irp->Complete);
377 return irp->Complete(irp);
385static UINT drive_process_irp_query_information(DRIVE_DEVICE* drive, IRP* irp)
388 UINT32 FsInformationClass = 0;
391 return ERROR_INVALID_PARAMETER;
393 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 4))
394 return ERROR_INVALID_DATA;
396 Stream_Read_UINT32(irp->input, FsInformationClass);
397 file = drive_get_file_by_id(drive, irp->FileId);
401 irp->IoStatus = STATUS_UNSUCCESSFUL;
403 else if (!drive_file_query_information(file, FsInformationClass, irp->output))
405 irp->IoStatus = drive_map_windows_err(GetLastError());
408 WINPR_ASSERT(irp->Complete);
409 return irp->Complete(irp);
417static UINT drive_process_irp_set_information(DRIVE_DEVICE* drive, IRP* irp)
420 UINT32 FsInformationClass = 0;
423 if (!drive || !irp || !irp->input || !irp->output)
424 return ERROR_INVALID_PARAMETER;
426 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 32))
427 return ERROR_INVALID_DATA;
429 Stream_Read_UINT32(irp->input, FsInformationClass);
430 Stream_Read_UINT32(irp->input, Length);
431 Stream_Seek(irp->input, 24);
432 file = drive_get_file_by_id(drive, irp->FileId);
436 irp->IoStatus = STATUS_UNSUCCESSFUL;
438 else if (!drive_file_set_information(file, FsInformationClass, Length, irp->input))
440 irp->IoStatus = drive_map_windows_err(GetLastError());
443 Stream_Write_UINT32(irp->output, Length);
445 WINPR_ASSERT(irp->Complete);
446 return irp->Complete(irp);
454static UINT drive_process_irp_query_volume_information(DRIVE_DEVICE* drive, IRP* irp)
456 UINT32 FsInformationClass = 0;
458 DWORD lpSectorsPerCluster = 0;
459 DWORD lpBytesPerSector = 0;
460 DWORD lpNumberOfFreeClusters = 0;
461 DWORD lpTotalNumberOfClusters = 0;
463 WCHAR LabelBuffer[32] = { 0 };
466 return ERROR_INVALID_PARAMETER;
468 output = irp->output;
470 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 4))
471 return ERROR_INVALID_DATA;
473 Stream_Read_UINT32(irp->input, FsInformationClass);
474 GetDiskFreeSpaceW(drive->path, &lpSectorsPerCluster, &lpBytesPerSector, &lpNumberOfFreeClusters,
475 &lpTotalNumberOfClusters);
477 switch (FsInformationClass)
479 case FileFsVolumeInformation:
482 const WCHAR* volumeLabel =
483 InitializeConstWCharFromUtf8(
"FREERDP", LabelBuffer, ARRAYSIZE(LabelBuffer));
484 const size_t volumeLabelLen = (_wcslen(volumeLabel) + 1) *
sizeof(WCHAR);
485 const size_t length = 17ul + volumeLabelLen;
487 if ((length > UINT32_MAX) || (volumeLabelLen > UINT32_MAX))
488 return CHANNEL_RC_NO_BUFFER;
490 Stream_Write_UINT32(output, (UINT32)length);
492 if (!Stream_EnsureRemainingCapacity(output, length))
494 WLog_ERR(TAG,
"Stream_EnsureRemainingCapacity failed!");
495 return CHANNEL_RC_NO_MEMORY;
498 GetFileAttributesExW(drive->path, GetFileExInfoStandard, &wfad);
499 Stream_Write_UINT32(output, wfad.ftCreationTime.dwLowDateTime);
500 Stream_Write_UINT32(output,
501 wfad.ftCreationTime.dwHighDateTime);
502 Stream_Write_UINT32(output, lpNumberOfFreeClusters & 0xffff);
503 Stream_Write_UINT32(output, (UINT32)volumeLabelLen);
504 Stream_Write_UINT8(output, 0);
506 Stream_Write(output, volumeLabel, volumeLabelLen);
510 case FileFsSizeInformation:
512 Stream_Write_UINT32(output, 24);
514 if (!Stream_EnsureRemainingCapacity(output, 24))
516 WLog_ERR(TAG,
"Stream_EnsureRemainingCapacity failed!");
517 return CHANNEL_RC_NO_MEMORY;
520 Stream_Write_UINT64(output, lpTotalNumberOfClusters);
521 Stream_Write_UINT64(output, lpNumberOfFreeClusters);
522 Stream_Write_UINT32(output, lpSectorsPerCluster);
523 Stream_Write_UINT32(output, lpBytesPerSector);
526 case FileFsAttributeInformation:
529 const WCHAR* diskType =
530 InitializeConstWCharFromUtf8(
"FAT32", LabelBuffer, ARRAYSIZE(LabelBuffer));
531 const size_t diskTypeLen = (_wcslen(diskType) + 1) *
sizeof(WCHAR);
532 const size_t length = 12ul + diskTypeLen;
534 if ((length > UINT32_MAX) || (diskTypeLen > UINT32_MAX))
535 return CHANNEL_RC_NO_BUFFER;
537 Stream_Write_UINT32(output, (UINT32)length);
539 if (!Stream_EnsureRemainingCapacity(output, length))
541 WLog_ERR(TAG,
"Stream_EnsureRemainingCapacity failed!");
542 return CHANNEL_RC_NO_MEMORY;
545 Stream_Write_UINT32(output, FILE_CASE_SENSITIVE_SEARCH | FILE_CASE_PRESERVED_NAMES |
546 FILE_UNICODE_ON_DISK);
547 Stream_Write_UINT32(output, MAX_PATH);
548 Stream_Write_UINT32(output, (UINT32)diskTypeLen);
549 Stream_Write(output, diskType, diskTypeLen);
553 case FileFsFullSizeInformation:
555 Stream_Write_UINT32(output, 32);
557 if (!Stream_EnsureRemainingCapacity(output, 32))
559 WLog_ERR(TAG,
"Stream_EnsureRemainingCapacity failed!");
560 return CHANNEL_RC_NO_MEMORY;
563 Stream_Write_UINT64(output, lpTotalNumberOfClusters);
564 Stream_Write_UINT64(output,
565 lpNumberOfFreeClusters);
566 Stream_Write_UINT64(output, lpNumberOfFreeClusters);
567 Stream_Write_UINT32(output, lpSectorsPerCluster);
568 Stream_Write_UINT32(output, lpBytesPerSector);
571 case FileFsDeviceInformation:
573 Stream_Write_UINT32(output, 8);
575 if (!Stream_EnsureRemainingCapacity(output, 8))
577 WLog_ERR(TAG,
"Stream_EnsureRemainingCapacity failed!");
578 return CHANNEL_RC_NO_MEMORY;
581 Stream_Write_UINT32(output, FILE_DEVICE_DISK);
582 Stream_Write_UINT32(output, 0);
586 WLog_WARN(TAG,
"Unhandled FSInformationClass %s [0x%08" PRIx32
"]",
587 FSInformationClass2Tag(FsInformationClass), FsInformationClass);
588 irp->IoStatus = STATUS_UNSUCCESSFUL;
589 Stream_Write_UINT32(output, 0);
593 WINPR_ASSERT(irp->Complete);
594 return irp->Complete(irp);
604static UINT drive_process_irp_silent_ignore(DRIVE_DEVICE* drive, IRP* irp)
606 if (!drive || !irp || !irp->output)
607 return ERROR_INVALID_PARAMETER;
609 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 4))
610 return ERROR_INVALID_DATA;
612 const uint32_t FsInformationClass = Stream_Get_UINT32(irp->input);
613 WLog_VRB(TAG,
"Silently ignore FSInformationClass %s [0x%08" PRIx32
"]",
614 FSInformationClass2Tag(FsInformationClass), FsInformationClass);
615 Stream_Write_UINT32(irp->output, 0);
616 WINPR_ASSERT(irp->Complete);
617 return irp->Complete(irp);
625static UINT drive_process_irp_query_directory(DRIVE_DEVICE* drive, IRP* irp)
627 const WCHAR* path = NULL;
629 BYTE InitialQuery = 0;
630 UINT32 PathLength = 0;
631 UINT32 FsInformationClass = 0;
634 return ERROR_INVALID_PARAMETER;
636 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, 32))
637 return ERROR_INVALID_DATA;
639 Stream_Read_UINT32(irp->input, FsInformationClass);
640 Stream_Read_UINT8(irp->input, InitialQuery);
641 Stream_Read_UINT32(irp->input, PathLength);
642 Stream_Seek(irp->input, 23);
643 path = Stream_ConstPointer(irp->input);
644 if (!Stream_CheckAndLogRequiredLength(TAG, irp->input, PathLength))
645 return ERROR_INVALID_DATA;
647 file = drive_get_file_by_id(drive, irp->FileId);
651 irp->IoStatus = STATUS_UNSUCCESSFUL;
652 Stream_Write_UINT32(irp->output, 0);
654 else if (!drive_file_query_directory(file, FsInformationClass, InitialQuery, path,
655 PathLength /
sizeof(WCHAR), irp->output))
657 irp->IoStatus = drive_map_windows_err(GetLastError());
660 WINPR_ASSERT(irp->Complete);
661 return irp->Complete(irp);
669static UINT drive_process_irp_directory_control(DRIVE_DEVICE* drive, IRP* irp)
672 return ERROR_INVALID_PARAMETER;
674 switch (irp->MinorFunction)
676 case IRP_MN_QUERY_DIRECTORY:
677 return drive_process_irp_query_directory(drive, irp);
679 case IRP_MN_NOTIFY_CHANGE_DIRECTORY:
680 return irp->Discard(irp);
683 irp->IoStatus = STATUS_NOT_SUPPORTED;
684 Stream_Write_UINT32(irp->output, 0);
685 WINPR_ASSERT(irp->Complete);
686 return irp->Complete(irp);
689 return CHANNEL_RC_OK;
697static UINT drive_process_irp_device_control(DRIVE_DEVICE* drive, IRP* irp)
700 return ERROR_INVALID_PARAMETER;
702 Stream_Write_UINT32(irp->output, 0);
703 WINPR_ASSERT(irp->Complete);
704 return irp->Complete(irp);
712static UINT drive_process_irp(DRIVE_DEVICE* drive, IRP* irp)
717 return ERROR_INVALID_PARAMETER;
719 irp->IoStatus = STATUS_SUCCESS;
721 switch (irp->MajorFunction)
724 error = drive_process_irp_create(drive, irp);
728 error = drive_process_irp_close(drive, irp);
732 error = drive_process_irp_read(drive, irp);
736 error = drive_process_irp_write(drive, irp);
739 case IRP_MJ_QUERY_INFORMATION:
740 error = drive_process_irp_query_information(drive, irp);
743 case IRP_MJ_SET_INFORMATION:
744 error = drive_process_irp_set_information(drive, irp);
747 case IRP_MJ_QUERY_VOLUME_INFORMATION:
748 error = drive_process_irp_query_volume_information(drive, irp);
751 case IRP_MJ_LOCK_CONTROL:
752 error = drive_process_irp_silent_ignore(drive, irp);
755 case IRP_MJ_DIRECTORY_CONTROL:
756 error = drive_process_irp_directory_control(drive, irp);
759 case IRP_MJ_DEVICE_CONTROL:
760 error = drive_process_irp_device_control(drive, irp);
764 irp->IoStatus = STATUS_NOT_SUPPORTED;
765 WINPR_ASSERT(irp->Complete);
766 error = irp->Complete(irp);
773static BOOL drive_poll_run(DRIVE_DEVICE* drive, IRP* irp)
779 const UINT error = drive_process_irp(drive, irp);
782 WLog_ERR(TAG,
"drive_process_irp failed with error %" PRIu32
"!", error);
790static DWORD WINAPI drive_thread_func(LPVOID arg)
792 DRIVE_DEVICE* drive = (DRIVE_DEVICE*)arg;
793 UINT error = CHANNEL_RC_OK;
797 error = ERROR_INVALID_PARAMETER;
803 if (!MessageQueue_Wait(drive->IrpQueue))
805 WLog_ERR(TAG,
"MessageQueue_Wait failed!");
806 error = ERROR_INTERNAL_ERROR;
810 if (MessageQueue_Size(drive->IrpQueue) < 1)
813 wMessage message = { 0 };
814 if (!MessageQueue_Peek(drive->IrpQueue, &message, TRUE))
816 WLog_ERR(TAG,
"MessageQueue_Peek failed!");
820 if (message.id == WMQ_QUIT)
823 IRP* irp = (IRP*)message.wParam;
824 if (!drive_poll_run(drive, irp))
830 if (error && drive && drive->rdpcontext)
831 setChannelError(drive->rdpcontext, error,
"drive_thread_func reported an error");
842static UINT drive_irp_request(DEVICE* device, IRP* irp)
844 DRIVE_DEVICE* drive = (DRIVE_DEVICE*)device;
847 return ERROR_INVALID_PARAMETER;
851 if (!MessageQueue_Post(drive->IrpQueue, NULL, 0, (
void*)irp, NULL))
853 WLog_ERR(TAG,
"MessageQueue_Post failed!");
854 return ERROR_INTERNAL_ERROR;
859 if (!drive_poll_run(drive, irp))
860 return ERROR_INTERNAL_ERROR;
863 return CHANNEL_RC_OK;
866static UINT drive_free_int(DRIVE_DEVICE* drive)
868 UINT error = CHANNEL_RC_OK;
871 return ERROR_INVALID_PARAMETER;
873 (void)CloseHandle(drive->thread);
874 ListDictionary_Free(drive->files);
875 MessageQueue_Free(drive->IrpQueue);
876 Stream_Free(drive->device.data, TRUE);
887static UINT drive_free(DEVICE* device)
889 DRIVE_DEVICE* drive = (DRIVE_DEVICE*)device;
890 UINT error = CHANNEL_RC_OK;
893 return ERROR_INVALID_PARAMETER;
895 if (MessageQueue_PostQuit(drive->IrpQueue, 0) &&
896 (WaitForSingleObject(drive->thread, INFINITE) == WAIT_FAILED))
898 error = GetLastError();
899 WLog_ERR(TAG,
"WaitForSingleObject failed with error %" PRIu32
"", error);
903 return drive_free_int(drive);
909static void drive_file_objfree(
void* obj)
914static void drive_message_free(
void* obj)
922 IRP* irp = (IRP*)msg->wParam;
925 WINPR_ASSERT(irp->Discard);
935 const char* path, BOOL automount, uint32_t* pid)
940 DRIVE_DEVICE* drive = NULL;
941 UINT error = ERROR_INTERNAL_ERROR;
943 if (!pEntryPoints || !name || !path)
945 WLog_ERR(TAG,
"[%s] Invalid parameters: pEntryPoints=%p, name=%p, path=%p", pEntryPoints,
947 return ERROR_INVALID_PARAMETER;
950 if (name[0] && path[0])
952 size_t pathLength = strnlen(path, MAX_PATH);
953 drive = (DRIVE_DEVICE*)calloc(1,
sizeof(DRIVE_DEVICE));
957 WLog_ERR(TAG,
"calloc failed!");
958 return CHANNEL_RC_NO_MEMORY;
961 drive->device.type = RDPDR_DTYP_FILESYSTEM;
962 drive->device.IRPRequest = drive_irp_request;
963 drive->device.Free = drive_free;
964 drive->rdpcontext = pEntryPoints->rdpcontext;
965 drive->automount = automount;
966 length = strlen(name);
967 drive->device.data = Stream_New(NULL, length + 1);
969 if (!drive->device.data)
971 WLog_ERR(TAG,
"Stream_New failed!");
972 error = CHANNEL_RC_NO_MEMORY;
976 for (
size_t i = 0; i < length; i++)
989 Stream_Write_UINT8(drive->device.data,
'_');
992 Stream_Write_UINT8(drive->device.data, (BYTE)name[i]);
996 Stream_Write_UINT8(drive->device.data,
'\0');
998 drive->device.name = Stream_BufferAs(drive->device.data,
char);
999 if (!drive->device.name)
1002 if ((pathLength > 1) && (path[pathLength - 1] ==
'/'))
1005 drive->path = ConvertUtf8NToWCharAlloc(path, pathLength, NULL);
1008 error = CHANNEL_RC_NO_MEMORY;
1012 drive->files = ListDictionary_New(TRUE);
1016 WLog_ERR(TAG,
"ListDictionary_New failed!");
1017 error = CHANNEL_RC_NO_MEMORY;
1021 ListDictionary_ValueObject(drive->files)->fnObjectFree = drive_file_objfree;
1022 drive->IrpQueue = MessageQueue_New(NULL);
1024 if (!drive->IrpQueue)
1026 WLog_ERR(TAG,
"ListDictionary_New failed!");
1027 error = CHANNEL_RC_NO_MEMORY;
1031 wObject* obj = MessageQueue_Object(drive->IrpQueue);
1033 obj->fnObjectFree = drive_message_free;
1035 if ((error = pEntryPoints->RegisterDevice(pEntryPoints->devman, &drive->device)))
1037 WLog_ERR(TAG,
"RegisterDevice failed with error %" PRIu32
"!", error);
1040 *pid = drive->device.id;
1043 FreeRDP_SynchronousStaticChannels);
1046 if (!(drive->thread =
1047 CreateThread(NULL, 0, drive_thread_func, drive, CREATE_SUSPENDED, NULL)))
1049 WLog_ERR(TAG,
"CreateThread failed!");
1053 ResumeThread(drive->thread);
1057 return CHANNEL_RC_OK;
1059 drive_free_int(drive);
1063static BOOL drive_filtered(
const WCHAR* drive)
1065 const WCHAR a[] = {
'A',
'\0' };
1066 const WCHAR b[] = {
'B',
'\0' };
1067 const WCHAR la[] = {
'a',
'\0' };
1068 const WCHAR lb[] = {
'b',
'\0' };
1069 const WCHAR* list[] = { a, b, la, lb };
1071 for (
size_t x = 0; x < ARRAYSIZE(list); x++)
1073 const WCHAR* cur = list[x];
1074 if (_wcsncmp(drive, cur, 2) == 0)
1082 UINT error = ERROR_INTERNAL_ERROR;
1084 WINPR_ASSERT(drive);
1085 WINPR_ASSERT(pEntryPoints);
1088 const DWORD dlen = GetLogicalDriveStringsW(0, NULL);
1090 WCHAR* devlist = calloc(dlen,
sizeof(WCHAR));
1092 return ERROR_OUTOFMEMORY;
1094 const DWORD rc = GetLogicalDriveStringsW(dlen, devlist);
1099 for (
size_t offset = 0; offset < rc; offset += len + 1)
1101 len = _wcsnlen(&devlist[offset], rc - offset);
1103 const WCHAR* dev = &devlist[offset];
1104 if (!drive_filtered(dev))
1106 char* bufdup = NULL;
1107 char* devdup = ConvertWCharNToUtf8Alloc(dev, len, NULL);
1110 error = ERROR_OUTOFMEMORY;
1114 winpr_asprintf(&bufdup, &size,
"%s_%s", drive->device.Name, devdup);
1117 drive_register_drive_path(pEntryPoints, bufdup, devdup, TRUE, &drive->device.Id);
1120 if (error != CHANNEL_RC_OK)
1125 error = CHANNEL_RC_OK;
1143 WINPR_ASSERT(pEntryPoints);
1146 WINPR_ASSERT(drive);
1148 const char all[] =
"*";
1149 const char home[] =
"%";
1150 if (strncmp(drive->Path, all,
sizeof(all)) == 0)
1152 error = handle_all_drives(drive, pEntryPoints);
1154 else if (strncmp(drive->Path, home,
sizeof(home)) == 0)
1157 drive->Path = GetKnownPath(KNOWN_PATH_HOME);
1161 WLog_ERR(TAG,
"_strdup failed!");
1162 return CHANNEL_RC_NO_MEMORY;
1164 error = drive_register_drive_path(pEntryPoints, drive->device.Name, drive->Path,
1165 drive->automount, &drive->device.Id);
1169 error = drive_register_drive_path(pEntryPoints, drive->device.Name, drive->Path,
1170 drive->automount, &drive->device.Id);
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.