/** * FreeRDP: A Remote Desktop Protocol client. * File System Virtual Channel * * Copyright 2010-2011 Marc-Andre Moreau * Copyright 2010-2011 Vic Lee * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif #ifndef _WIN32 #define __USE_LARGEFILE64 #define _LARGEFILE_SOURCE #define _LARGEFILE64_SOURCE #include #endif #include #include #include #include #include #include #include #include #include #include #include #include "rdpdr_constants.h" #include "rdpdr_types.h" #include "disk_file.h" typedef struct _DISK_DEVICE DISK_DEVICE; struct _DISK_DEVICE { DEVICE device; char* path; LIST* files; HANDLE mutex; HANDLE thread; LIST* irp_list; HANDLE irpEvent; HANDLE stopEvent; DEVMAN* devman; pcRegisterDevice UnregisterDevice; }; static uint32 disk_map_posix_err(int fs_errno) { uint32 rc; /* try to return NTSTATUS version of error code */ switch (fs_errno) { case EPERM: case EACCES: rc = STATUS_ACCESS_DENIED; break; case ENOENT: rc = STATUS_NO_SUCH_FILE; break; case EBUSY: rc = STATUS_DEVICE_BUSY; break; case EEXIST: rc = STATUS_OBJECT_NAME_COLLISION; break; case EISDIR: rc = STATUS_FILE_IS_A_DIRECTORY; break; default: rc = STATUS_UNSUCCESSFUL; break; } DEBUG_SVC("errno 0x%x mapped to 0x%x", fs_errno, rc); return rc; } static DISK_FILE* disk_get_file_by_id(DISK_DEVICE* disk, uint32 id) { LIST_ITEM* item; DISK_FILE* file; for (item = disk->files->head; item; item = item->next) { file = (DISK_FILE*) item->data; if (file->id == id) return file; } return NULL; } static void disk_process_irp_create(DISK_DEVICE* disk, IRP* irp) { DISK_FILE* file; uint32 DesiredAccess; uint32 CreateDisposition; uint32 CreateOptions; uint32 PathLength; char* path; uint32 FileId; uint8 Information; stream_read_uint32(irp->input, DesiredAccess); stream_seek(irp->input, 16); /* AllocationSize(8), FileAttributes(4), SharedAccess(4) */ stream_read_uint32(irp->input, CreateDisposition); stream_read_uint32(irp->input, CreateOptions); stream_read_uint32(irp->input, PathLength); freerdp_UnicodeToAsciiAlloc((WCHAR*) stream_get_tail(irp->input), &path, PathLength / 2); FileId = irp->devman->id_sequence++; file = disk_file_new(disk->path, path, FileId, DesiredAccess, CreateDisposition, CreateOptions); if (file == NULL) { irp->IoStatus = STATUS_UNSUCCESSFUL; FileId = 0; Information = 0; DEBUG_WARN("failed to create %s.", path); } else if (file->err) { FileId = 0; Information = 0; /* map errno to windows result*/ irp->IoStatus = disk_map_posix_err(file->err); disk_file_free(file); } else { list_enqueue(disk->files, file); switch (CreateDisposition) { case FILE_SUPERSEDE: case FILE_OPEN: case FILE_CREATE: case FILE_OVERWRITE: Information = FILE_SUPERSEDED; break; case FILE_OPEN_IF: Information = FILE_OPENED; break; case FILE_OVERWRITE_IF: Information = FILE_OVERWRITTEN; break; default: Information = 0; break; } DEBUG_SVC("%s(%d) created.", file->fullpath, file->id); } stream_write_uint32(irp->output, FileId); stream_write_uint8(irp->output, Information); xfree(path); irp->Complete(irp); } static void disk_process_irp_close(DISK_DEVICE* disk, IRP* irp) { DISK_FILE* file; file = disk_get_file_by_id(disk, irp->FileId); if (file == NULL) { irp->IoStatus = STATUS_UNSUCCESSFUL; DEBUG_WARN("FileId %d not valid.", irp->FileId); } else { DEBUG_SVC("%s(%d) closed.", file->fullpath, file->id); list_remove(disk->files, file); disk_file_free(file); } stream_write_zero(irp->output, 5); /* Padding(5) */ irp->Complete(irp); } static void disk_process_irp_read(DISK_DEVICE* disk, IRP* irp) { DISK_FILE* file; uint32 Length; uint64 Offset; uint8* buffer = NULL; stream_read_uint32(irp->input, Length); stream_read_uint64(irp->input, Offset); file = disk_get_file_by_id(disk, irp->FileId); if (file == NULL) { irp->IoStatus = STATUS_UNSUCCESSFUL; Length = 0; DEBUG_WARN("FileId %d not valid.", irp->FileId); } else if (!disk_file_seek(file, Offset)) { irp->IoStatus = STATUS_UNSUCCESSFUL; Length = 0; DEBUG_WARN("seek %s(%d) failed.", file->fullpath, file->id); } else { buffer = (uint8*) xmalloc(Length); if (!disk_file_read(file, buffer, &Length)) { irp->IoStatus = STATUS_UNSUCCESSFUL; xfree(buffer); buffer = NULL; Length = 0; DEBUG_WARN("read %s(%d) failed.", file->fullpath, file->id); } else { DEBUG_SVC("read %llu-%llu from %s(%d).", Offset, Offset + Length, file->fullpath, file->id); } } stream_write_uint32(irp->output, Length); if (Length > 0) { stream_check_size(irp->output, (int)Length); stream_write(irp->output, buffer, Length); } xfree(buffer); irp->Complete(irp); } static void disk_process_irp_write(DISK_DEVICE* disk, IRP* irp) { DISK_FILE* file; uint32 Length; uint64 Offset; stream_read_uint32(irp->input, Length); stream_read_uint64(irp->input, Offset); stream_seek(irp->input, 20); /* Padding */ file = disk_get_file_by_id(disk, irp->FileId); if (file == NULL) { irp->IoStatus = STATUS_UNSUCCESSFUL; Length = 0; DEBUG_WARN("FileId %d not valid.", irp->FileId); } else if (!disk_file_seek(file, Offset)) { irp->IoStatus = STATUS_UNSUCCESSFUL; Length = 0; DEBUG_WARN("seek %s(%d) failed.", file->fullpath, file->id); } else if (!disk_file_write(file, stream_get_tail(irp->input), Length)) { irp->IoStatus = STATUS_UNSUCCESSFUL; Length = 0; DEBUG_WARN("write %s(%d) failed.", file->fullpath, file->id); } else { DEBUG_SVC("write %llu-%llu to %s(%d).", Offset, Offset + Length, file->fullpath, file->id); } stream_write_uint32(irp->output, Length); stream_write_uint8(irp->output, 0); /* Padding */ irp->Complete(irp); } static void disk_process_irp_query_information(DISK_DEVICE* disk, IRP* irp) { DISK_FILE* file; uint32 FsInformationClass; stream_read_uint32(irp->input, FsInformationClass); file = disk_get_file_by_id(disk, irp->FileId); if (file == NULL) { irp->IoStatus = STATUS_UNSUCCESSFUL; DEBUG_WARN("FileId %d not valid.", irp->FileId); } else if (!disk_file_query_information(file, FsInformationClass, irp->output)) { irp->IoStatus = STATUS_UNSUCCESSFUL; DEBUG_WARN("FsInformationClass %d on %s(%d) failed.", FsInformationClass, file->fullpath, file->id); } else { DEBUG_SVC("FsInformationClass %d on %s(%d).", FsInformationClass, file->fullpath, file->id); } irp->Complete(irp); } static void disk_process_irp_set_information(DISK_DEVICE* disk, IRP* irp) { DISK_FILE* file; uint32 FsInformationClass; uint32 Length; stream_read_uint32(irp->input, FsInformationClass); stream_read_uint32(irp->input, Length); stream_seek(irp->input, 24); /* Padding */ file = disk_get_file_by_id(disk, irp->FileId); if (file == NULL) { irp->IoStatus = STATUS_UNSUCCESSFUL; DEBUG_WARN("FileId %d not valid.", irp->FileId); } else if (!disk_file_set_information(file, FsInformationClass, Length, irp->input)) { irp->IoStatus = STATUS_UNSUCCESSFUL; DEBUG_WARN("FsInformationClass %d on %s(%d) failed.", FsInformationClass, file->fullpath, file->id); } else { DEBUG_SVC("FsInformationClass %d on %s(%d) ok.", FsInformationClass, file->fullpath, file->id); } stream_write_uint32(irp->output, Length); irp->Complete(irp); } static void disk_process_irp_query_volume_information(DISK_DEVICE* disk, IRP* irp) { uint32 FsInformationClass; STREAM* output = irp->output; struct STATVFS svfst; struct STAT st; char* volumeLabel = {"FREERDP"}; char* diskType = {"FAT32"}; WCHAR* outStr; int length; stream_read_uint32(irp->input, FsInformationClass); STATVFS(disk->path, &svfst); STAT(disk->path, &st); switch (FsInformationClass) { case FileFsVolumeInformation: /* http://msdn.microsoft.com/en-us/library/cc232108.aspx */ length = freerdp_AsciiToUnicodeAlloc(volumeLabel, &outStr, 0) * 2; stream_write_uint32(output, 17 + length); /* Length */ stream_check_size(output, 17 + length); stream_write_uint64(output, FILE_TIME_SYSTEM_TO_RDP(st.st_ctime)); /* VolumeCreationTime */ stream_write_uint32(output, svfst.f_fsid); /* VolumeSerialNumber */ stream_write_uint32(output, length); /* VolumeLabelLength */ stream_write_uint8(output, 0); /* SupportsObjects */ /* Reserved(1), MUST NOT be added! */ stream_write(output, outStr, length); /* VolumeLabel (Unicode) */ xfree(outStr); break; case FileFsSizeInformation: /* http://msdn.microsoft.com/en-us/library/cc232107.aspx */ stream_write_uint32(output, 24); /* Length */ stream_check_size(output, 24); stream_write_uint64(output, svfst.f_blocks); /* TotalAllocationUnits */ stream_write_uint64(output, svfst.f_bavail); /* AvailableAllocationUnits */ stream_write_uint32(output, 1); /* SectorsPerAllocationUnit */ stream_write_uint32(output, svfst.f_bsize); /* BytesPerSector */ break; case FileFsAttributeInformation: /* http://msdn.microsoft.com/en-us/library/cc232101.aspx */ length = freerdp_AsciiToUnicodeAlloc(diskType, &outStr, 0) * 2; stream_write_uint32(output, 12 + length); /* Length */ stream_check_size(output, 12 + length); stream_write_uint32(output, FILE_CASE_SENSITIVE_SEARCH | FILE_CASE_PRESERVED_NAMES | FILE_UNICODE_ON_DISK); /* FileSystemAttributes */ stream_write_uint32(output, svfst.f_namemax/*510*/); /* MaximumComponentNameLength */ stream_write_uint32(output, length); /* FileSystemNameLength */ stream_write(output, outStr, length); /* FileSystemName (Unicode) */ xfree(outStr); break; case FileFsFullSizeInformation: /* http://msdn.microsoft.com/en-us/library/cc232104.aspx */ stream_write_uint32(output, 32); /* Length */ stream_check_size(output, 32); stream_write_uint64(output, svfst.f_blocks); /* TotalAllocationUnits */ stream_write_uint64(output, svfst.f_bavail); /* CallerAvailableAllocationUnits */ stream_write_uint64(output, svfst.f_bfree); /* AvailableAllocationUnits */ stream_write_uint32(output, 1); /* SectorsPerAllocationUnit */ stream_write_uint32(output, svfst.f_bsize); /* BytesPerSector */ break; case FileFsDeviceInformation: /* http://msdn.microsoft.com/en-us/library/cc232109.aspx */ stream_write_uint32(output, 8); /* Length */ stream_check_size(output, 8); stream_write_uint32(output, FILE_DEVICE_DISK); /* DeviceType */ stream_write_uint32(output, 0); /* Characteristics */ break; default: irp->IoStatus = STATUS_UNSUCCESSFUL; stream_write_uint32(output, 0); /* Length */ DEBUG_WARN("invalid FsInformationClass %d", FsInformationClass); break; } irp->Complete(irp); } static void disk_process_irp_query_directory(DISK_DEVICE* disk, IRP* irp) { DISK_FILE* file; uint32 FsInformationClass; uint8 InitialQuery; uint32 PathLength; char* path; stream_read_uint32(irp->input, FsInformationClass); stream_read_uint8(irp->input, InitialQuery); stream_read_uint32(irp->input, PathLength); stream_seek(irp->input, 23); /* Padding */ freerdp_UnicodeToAsciiAlloc((WCHAR*) stream_get_tail(irp->input), &path, PathLength / 2); file = disk_get_file_by_id(disk, irp->FileId); if (file == NULL) { irp->IoStatus = STATUS_UNSUCCESSFUL; stream_write_uint32(irp->output, 0); /* Length */ DEBUG_WARN("FileId %d not valid.", irp->FileId); } else if (!disk_file_query_directory(file, FsInformationClass, InitialQuery, path, irp->output)) { irp->IoStatus = STATUS_NO_MORE_FILES; } xfree(path); irp->Complete(irp); } static void disk_process_irp_directory_control(DISK_DEVICE* disk, IRP* irp) { switch (irp->MinorFunction) { case IRP_MN_QUERY_DIRECTORY: disk_process_irp_query_directory(disk, irp); break; case IRP_MN_NOTIFY_CHANGE_DIRECTORY: /* TODO */ irp->Discard(irp); break; default: DEBUG_WARN("MinorFunction 0x%X not supported", irp->MinorFunction); irp->IoStatus = STATUS_NOT_SUPPORTED; stream_write_uint32(irp->output, 0); /* Length */ irp->Complete(irp); break; } } static void disk_process_irp_device_control(DISK_DEVICE* disk, IRP* irp) { stream_write_uint32(irp->output, 0); /* OutputBufferLength */ irp->Complete(irp); } static void disk_process_irp(DISK_DEVICE* disk, IRP* irp) { switch (irp->MajorFunction) { case IRP_MJ_CREATE: disk_process_irp_create(disk, irp); break; case IRP_MJ_CLOSE: disk_process_irp_close(disk, irp); break; case IRP_MJ_READ: disk_process_irp_read(disk, irp); break; case IRP_MJ_WRITE: disk_process_irp_write(disk, irp); break; case IRP_MJ_QUERY_INFORMATION: disk_process_irp_query_information(disk, irp); break; case IRP_MJ_SET_INFORMATION: disk_process_irp_set_information(disk, irp); break; case IRP_MJ_QUERY_VOLUME_INFORMATION: disk_process_irp_query_volume_information(disk, irp); break; case IRP_MJ_DIRECTORY_CONTROL: disk_process_irp_directory_control(disk, irp); break; case IRP_MJ_DEVICE_CONTROL: disk_process_irp_device_control(disk, irp); break; default: DEBUG_WARN("MajorFunction 0x%X not supported", irp->MajorFunction); irp->IoStatus = STATUS_NOT_SUPPORTED; irp->Complete(irp); break; } } static void disk_process_irp_list(DISK_DEVICE* disk) { IRP* irp; while (1) { if (WaitForSingleObject(disk->stopEvent, 0) == WAIT_OBJECT_0) break; WaitForSingleObject(disk->mutex, INFINITE); irp = (IRP*) list_dequeue(disk->irp_list); ReleaseMutex(disk->mutex); if (irp == NULL) break; disk_process_irp(disk, irp); } } static void* disk_thread_func(void* arg) { DISK_DEVICE* disk = (DISK_DEVICE*) arg; while (1) { WaitForSingleObject(disk->irpEvent, INFINITE); if (WaitForSingleObject(disk->stopEvent, 0) == WAIT_OBJECT_0) break; ResetEvent(disk->irpEvent); disk_process_irp_list(disk); } return NULL; } static void disk_irp_request(DEVICE* device, IRP* irp) { DISK_DEVICE* disk = (DISK_DEVICE*) device; WaitForSingleObject(disk->mutex, INFINITE); list_enqueue(disk->irp_list, irp); ReleaseMutex(disk->mutex); SetEvent(disk->irpEvent); } static void disk_free(DEVICE* device) { IRP* irp; DISK_FILE* file; DISK_DEVICE* disk = (DISK_DEVICE*) device; SetEvent(disk->stopEvent); CloseHandle(disk->thread); CloseHandle(disk->irpEvent); CloseHandle(disk->mutex); while ((irp = (IRP*) list_dequeue(disk->irp_list)) != NULL) irp->Discard(irp); list_free(disk->irp_list); while ((file = (DISK_FILE*) list_dequeue(disk->files)) != NULL) disk_file_free(file); list_free(disk->files); xfree(disk); } void disk_register_disk_path(PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints, char *name, char *path) { DISK_DEVICE* disk; int i, length ; #ifdef WIN32 /* * We cannot enter paths like c:\ because : is an arg separator * thus, paths are entered as c+\ and the + is substituted here */ if ( path[1] == '+' ) { if ( (path[0]>='a' && path[0]<='z') || (path[0]>='A' && path[0]<='Z') ) { path[1] = ':'; } } #endif if (name[0] && path[0]) { disk = xnew(DISK_DEVICE); disk->device.type = RDPDR_DTYP_FILESYSTEM; disk->device.name = name; disk->device.IRPRequest = disk_irp_request; disk->device.Free = disk_free; length = strlen(name); disk->device.data = stream_new(length + 1); for (i = 0; i <= length; i++) stream_write_uint8(disk->device.data, name[i] < 0 ? '_' : name[i]); disk->path = path; disk->files = list_new(); disk->irp_list = list_new(); disk->mutex = CreateMutex(NULL, FALSE, NULL); disk->irpEvent = CreateEvent(NULL, TRUE, FALSE, NULL); disk->stopEvent = CreateEvent(NULL, TRUE, FALSE, NULL); disk->thread = CreateThread(NULL, 0, (LPTHREAD_START_ROUTINE) disk_thread_func, disk, CREATE_SUSPENDED, NULL); pEntryPoints->RegisterDevice(pEntryPoints->devman, (DEVICE*) disk); ResumeThread(disk->thread); } } #ifdef WITH_STATIC_PLUGINS int disk_entry(PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints) #else int DeviceServiceEntry(PDEVICE_SERVICE_ENTRY_POINTS pEntryPoints) #endif { char* name; char* path; #ifdef WIN32 char devlist[512], buf[512]; char *dev; int len ; #endif name = (char*) pEntryPoints->plugin_data->data[1]; path = (char*) pEntryPoints->plugin_data->data[2]; #ifndef WIN32 disk_register_disk_path(pEntryPoints, name, path); #else /* Special case: path[0] == '*' -> export all drives */ /* Special case: path[0] == '%' -> user home dir */ if( path[0] == '%' ) { _snprintf(buf, sizeof(buf), "%s\\", getenv("USERPROFILE")); disk_register_disk_path(pEntryPoints, name, xstrdup(buf)); } else if( path[0] == '*' ) { int i; /* Enumerate all devices: */ GetLogicalDriveStringsA(sizeof(devlist) - 1, devlist); for (dev = devlist, i = 0; *dev; dev += 4, i++) { if (*dev > 'B') { /* Suppress disk drives A and B to avoid pesty messages */ _snprintf(buf, sizeof(buf) - 4, "%s", name); len = strlen(buf); buf[len] = '_'; buf[len + 1] = dev[0]; buf[len + 2] = 0; buf[len + 3] = 0; disk_register_disk_path(pEntryPoints, xstrdup(buf), xstrdup(dev)); } } } else { disk_register_disk_path(pEntryPoints, name, path); } #endif return 0; }