xref: /reactos/dll/win32/kernel32/client/file/disk.c (revision cc3672cb)
1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         ReactOS system libraries
4  * FILE:            dll/win32/kernel32/client/file/disk.c
5  * PURPOSE:         Disk and Drive functions
6  * PROGRAMMER:      Ariadne ( ariadne@xs4all.nl)
7  *                  Erik Bos, Alexandre Julliard :
8  *                      GetLogicalDriveStringsA,
9  *                      GetLogicalDriveStringsW, GetLogicalDrives
10  * UPDATE HISTORY:
11  *                  Created 01/11/98
12  */
13 //WINE copyright notice:
14 /*
15  * DOS drives handling functions
16  *
17  * Copyright 1993 Erik Bos
18  * Copyright 1996 Alexandre Julliard
19  */
20 
21 #include <k32.h>
22 
23 #define NDEBUG
24 #include <debug.h>
25 
26 #define MAX_DOS_DRIVES 26
27 
28 /*
29  * @implemented
30  */
31 /* Synced to Wine-2008/12/28 */
32 DWORD
33 WINAPI
34 GetLogicalDriveStringsA(IN DWORD nBufferLength,
35                         IN LPSTR lpBuffer)
36 {
37     DWORD drive, count;
38     DWORD dwDriveMap;
39     LPSTR p;
40 
41     dwDriveMap = GetLogicalDrives();
42 
43     for (drive = count = 0; drive < MAX_DOS_DRIVES; drive++)
44     {
45         if (dwDriveMap & (1<<drive))
46             count++;
47     }
48 
49 
50     if ((count * 4) + 1 > nBufferLength) return ((count * 4) + 1);
51 
52     p = lpBuffer;
53 
54     for (drive = 0; drive < MAX_DOS_DRIVES; drive++)
55         if (dwDriveMap & (1<<drive))
56         {
57             *p++ = 'A' + (UCHAR)drive;
58             *p++ = ':';
59             *p++ = '\\';
60             *p++ = '\0';
61         }
62     *p = '\0';
63 
64     return (count * 4);
65 }
66 
67 /*
68  * @implemented
69  */
70 /* Synced to Wine-2008/12/28 */
71 DWORD
72 WINAPI
73 GetLogicalDriveStringsW(IN DWORD nBufferLength,
74                         IN LPWSTR lpBuffer)
75 {
76     DWORD drive, count;
77     DWORD dwDriveMap;
78     LPWSTR p;
79 
80     dwDriveMap = GetLogicalDrives();
81 
82     for (drive = count = 0; drive < MAX_DOS_DRIVES; drive++)
83     {
84         if (dwDriveMap & (1<<drive))
85             count++;
86     }
87 
88     if ((count * 4) + 1 > nBufferLength) return ((count * 4) + 1);
89 
90     p = lpBuffer;
91     for (drive = 0; drive < MAX_DOS_DRIVES; drive++)
92         if (dwDriveMap & (1<<drive))
93         {
94             *p++ = (WCHAR)('A' + drive);
95             *p++ = (WCHAR)':';
96             *p++ = (WCHAR)'\\';
97             *p++ = (WCHAR)'\0';
98         }
99     *p = (WCHAR)'\0';
100 
101     return (count * 4);
102 }
103 
104 /*
105  * @implemented
106  */
107 /* Synced to Wine-? */
108 DWORD
109 WINAPI
110 GetLogicalDrives(VOID)
111 {
112     NTSTATUS Status;
113     PROCESS_DEVICEMAP_INFORMATION ProcessDeviceMapInfo;
114 
115     /* Get the Device Map for this Process */
116     Status = NtQueryInformationProcess(NtCurrentProcess(),
117                                        ProcessDeviceMap,
118                                        &ProcessDeviceMapInfo.Query,
119                                        sizeof(ProcessDeviceMapInfo.Query),
120                                        NULL);
121 
122     /* Return the Drive Map */
123     if (!NT_SUCCESS(Status))
124     {
125         BaseSetLastNTError(Status);
126         return 0;
127     }
128 
129     if (ProcessDeviceMapInfo.Query.DriveMap == 0)
130     {
131         SetLastError(ERROR_SUCCESS);
132     }
133 
134     return ProcessDeviceMapInfo.Query.DriveMap;
135 }
136 
137 /*
138  * @implemented
139  */
140 BOOL
141 WINAPI
142 GetDiskFreeSpaceA(IN LPCSTR lpRootPathName,
143                   OUT LPDWORD lpSectorsPerCluster,
144                   OUT LPDWORD lpBytesPerSector,
145                   OUT LPDWORD lpNumberOfFreeClusters,
146                   OUT LPDWORD lpTotalNumberOfClusters)
147 {
148     PCSTR RootPath;
149     PUNICODE_STRING RootPathU;
150 
151     RootPath = lpRootPathName;
152     if (RootPath == NULL)
153     {
154         RootPath = "\\";
155     }
156 
157     RootPathU = Basep8BitStringToStaticUnicodeString(RootPath);
158     if (RootPathU == NULL)
159     {
160         return FALSE;
161     }
162 
163     return GetDiskFreeSpaceW(RootPathU->Buffer, lpSectorsPerCluster,
164                              lpBytesPerSector, lpNumberOfFreeClusters,
165                              lpTotalNumberOfClusters);
166 }
167 
168 /*
169  * @implemented
170  */
171 BOOL
172 WINAPI
173 GetDiskFreeSpaceW(IN LPCWSTR lpRootPathName,
174                   OUT LPDWORD lpSectorsPerCluster,
175                   OUT LPDWORD lpBytesPerSector,
176                   OUT LPDWORD lpNumberOfFreeClusters,
177                   OUT LPDWORD lpTotalNumberOfClusters)
178 {
179     BOOL Below2GB;
180     PCWSTR RootPath;
181     NTSTATUS Status;
182     HANDLE RootHandle;
183     UNICODE_STRING FileName;
184     IO_STATUS_BLOCK IoStatusBlock;
185     OBJECT_ATTRIBUTES ObjectAttributes;
186     FILE_FS_SIZE_INFORMATION FileFsSize;
187 
188     /* If no path provided, get root path */
189     RootPath = lpRootPathName;
190     if (lpRootPathName == NULL)
191     {
192         RootPath = L"\\";
193     }
194 
195     /* Convert the path to NT path */
196     if (!RtlDosPathNameToNtPathName_U(RootPath, &FileName, NULL, NULL))
197     {
198         SetLastError(ERROR_PATH_NOT_FOUND);
199         return FALSE;
200     }
201 
202     /* Open it for disk space query! */
203     InitializeObjectAttributes(&ObjectAttributes, &FileName,
204                                OBJ_CASE_INSENSITIVE, NULL, NULL);
205     Status = NtOpenFile(&RootHandle, SYNCHRONIZE, &ObjectAttributes, &IoStatusBlock,
206                         FILE_SHARE_READ | FILE_SHARE_WRITE,
207                         FILE_DIRECTORY_FILE | FILE_SYNCHRONOUS_IO_NONALERT | FILE_OPEN_FOR_FREE_SPACE_QUERY);
208     if (!NT_SUCCESS(Status))
209     {
210         BaseSetLastNTError(Status);
211         RtlFreeHeap(RtlGetProcessHeap(), 0, FileName.Buffer);
212         if (lpBytesPerSector != NULL)
213         {
214             *lpBytesPerSector = 0;
215         }
216 
217         return FALSE;
218     }
219 
220     /* We don't need the name any longer */
221     RtlFreeHeap(RtlGetProcessHeap(), 0, FileName.Buffer);
222 
223     /* Query disk space! */
224     Status = NtQueryVolumeInformationFile(RootHandle, &IoStatusBlock, &FileFsSize,
225                                           sizeof(FILE_FS_SIZE_INFORMATION),
226                                           FileFsSizeInformation);
227     NtClose(RootHandle);
228     if (!NT_SUCCESS(Status))
229     {
230         BaseSetLastNTError(Status);
231         return FALSE;
232     }
233 
234     /* Are we in some compatibility mode where size must be below 2GB? */
235     Below2GB = ((NtCurrentPeb()->AppCompatFlags.LowPart & GetDiskFreeSpace2GB) == GetDiskFreeSpace2GB);
236 
237     /* If we're to overflow output, make sure we return the maximum */
238     if (FileFsSize.TotalAllocationUnits.HighPart != 0)
239     {
240         FileFsSize.TotalAllocationUnits.LowPart = -1;
241     }
242 
243     if (FileFsSize.AvailableAllocationUnits.HighPart != 0)
244     {
245         FileFsSize.AvailableAllocationUnits.LowPart = -1;
246     }
247 
248     /* Return what user asked for */
249     if (lpSectorsPerCluster != NULL)
250     {
251         *lpSectorsPerCluster = FileFsSize.SectorsPerAllocationUnit;
252     }
253 
254     if (lpBytesPerSector != NULL)
255     {
256         *lpBytesPerSector = FileFsSize.BytesPerSector;
257     }
258 
259     if (lpNumberOfFreeClusters != NULL)
260     {
261         if (!Below2GB)
262         {
263             *lpNumberOfFreeClusters = FileFsSize.AvailableAllocationUnits.LowPart;
264         }
265         /* If we have to remain below 2GB... */
266         else
267         {
268             DWORD FreeClusters;
269 
270             /* Compute how many clusters there are in less than 2GB: 2 * 1024 * 1024 * 1024- 1 */
271             FreeClusters = 0x7FFFFFFF / (FileFsSize.SectorsPerAllocationUnit * FileFsSize.BytesPerSector);
272             /* If that's higher than what was queried, then return the queried value, it's OK! */
273             if (FreeClusters > FileFsSize.AvailableAllocationUnits.LowPart)
274             {
275                 FreeClusters = FileFsSize.AvailableAllocationUnits.LowPart;
276             }
277 
278             *lpNumberOfFreeClusters = FreeClusters;
279         }
280     }
281 
282     if (lpTotalNumberOfClusters != NULL)
283     {
284         if (!Below2GB)
285         {
286             *lpTotalNumberOfClusters = FileFsSize.TotalAllocationUnits.LowPart;
287         }
288         /* If we have to remain below 2GB... */
289         else
290         {
291             DWORD TotalClusters;
292 
293             /* Compute how many clusters there are in less than 2GB: 2 * 1024 * 1024 * 1024- 1 */
294             TotalClusters = 0x7FFFFFFF / (FileFsSize.SectorsPerAllocationUnit * FileFsSize.BytesPerSector);
295             /* If that's higher than what was queried, then return the queried value, it's OK! */
296             if (TotalClusters > FileFsSize.TotalAllocationUnits.LowPart)
297             {
298                 TotalClusters = FileFsSize.TotalAllocationUnits.LowPart;
299             }
300 
301             *lpTotalNumberOfClusters = TotalClusters;
302         }
303     }
304 
305     return TRUE;
306 }
307 
308 /*
309  * @implemented
310  */
311 BOOL
312 WINAPI
313 GetDiskFreeSpaceExA(IN LPCSTR lpDirectoryName OPTIONAL,
314                     OUT PULARGE_INTEGER lpFreeBytesAvailableToCaller,
315                     OUT PULARGE_INTEGER lpTotalNumberOfBytes,
316                     OUT PULARGE_INTEGER lpTotalNumberOfFreeBytes)
317 {
318     PCSTR RootPath;
319     PUNICODE_STRING RootPathU;
320 
321     RootPath = lpDirectoryName;
322     if (RootPath == NULL)
323     {
324         RootPath = "\\";
325     }
326 
327     RootPathU = Basep8BitStringToStaticUnicodeString(RootPath);
328     if (RootPathU == NULL)
329     {
330         return FALSE;
331     }
332 
333     return GetDiskFreeSpaceExW(RootPathU->Buffer, lpFreeBytesAvailableToCaller,
334                               lpTotalNumberOfBytes, lpTotalNumberOfFreeBytes);
335 }
336 
337 /*
338  * @implemented
339  */
340 BOOL
341 WINAPI
342 GetDiskFreeSpaceExW(IN LPCWSTR lpDirectoryName OPTIONAL,
343                     OUT PULARGE_INTEGER lpFreeBytesAvailableToCaller,
344                     OUT PULARGE_INTEGER lpTotalNumberOfBytes,
345                     OUT PULARGE_INTEGER lpTotalNumberOfFreeBytes)
346 {
347     PCWSTR RootPath;
348     NTSTATUS Status;
349     HANDLE RootHandle;
350     UNICODE_STRING FileName;
351     DWORD BytesPerAllocationUnit;
352     IO_STATUS_BLOCK IoStatusBlock;
353     OBJECT_ATTRIBUTES ObjectAttributes;
354     FILE_FS_SIZE_INFORMATION FileFsSize;
355 
356     /* If no path provided, get root path */
357     RootPath = lpDirectoryName;
358     if (lpDirectoryName == NULL)
359     {
360         RootPath = L"\\";
361     }
362 
363     /* Convert the path to NT path */
364     if (!RtlDosPathNameToNtPathName_U(RootPath, &FileName, NULL, NULL))
365     {
366         SetLastError(ERROR_PATH_NOT_FOUND);
367         return FALSE;
368     }
369 
370     /* Open it for disk space query! */
371     InitializeObjectAttributes(&ObjectAttributes, &FileName,
372                                OBJ_CASE_INSENSITIVE, NULL, NULL);
373     Status = NtOpenFile(&RootHandle, SYNCHRONIZE, &ObjectAttributes, &IoStatusBlock,
374                         FILE_SHARE_READ | FILE_SHARE_WRITE,
375                         FILE_DIRECTORY_FILE | FILE_SYNCHRONOUS_IO_NONALERT | FILE_OPEN_FOR_FREE_SPACE_QUERY);
376     if (!NT_SUCCESS(Status))
377     {
378         BaseSetLastNTError(Status);
379         /* If error conversion lead to file not found, override to use path not found
380          * which is more accurate
381          */
382         if (GetLastError() == ERROR_FILE_NOT_FOUND)
383         {
384             SetLastError(ERROR_PATH_NOT_FOUND);
385         }
386 
387         RtlFreeHeap(RtlGetProcessHeap(), 0, FileName.Buffer);
388 
389         return FALSE;
390     }
391 
392     RtlFreeHeap(RtlGetProcessHeap(), 0, FileName.Buffer);
393 
394     /* If user asks for lpTotalNumberOfFreeBytes, try to use full size information */
395     if (lpTotalNumberOfFreeBytes != NULL)
396     {
397         FILE_FS_FULL_SIZE_INFORMATION FileFsFullSize;
398 
399         /* Issue the full fs size request */
400         Status = NtQueryVolumeInformationFile(RootHandle, &IoStatusBlock, &FileFsFullSize,
401                                               sizeof(FILE_FS_FULL_SIZE_INFORMATION),
402                                               FileFsFullSizeInformation);
403         /* If it succeed, complete out buffers */
404         if (NT_SUCCESS(Status))
405         {
406             /* We can close here, we'll return */
407             NtClose(RootHandle);
408 
409             /* Compute the size of an AU */
410             BytesPerAllocationUnit = FileFsFullSize.SectorsPerAllocationUnit * FileFsFullSize.BytesPerSector;
411 
412             /* And then return what was asked */
413             if (lpFreeBytesAvailableToCaller != NULL)
414             {
415                 lpFreeBytesAvailableToCaller->QuadPart = FileFsFullSize.CallerAvailableAllocationUnits.QuadPart * BytesPerAllocationUnit;
416             }
417 
418             if (lpTotalNumberOfBytes != NULL)
419             {
420                 lpTotalNumberOfBytes->QuadPart = FileFsFullSize.TotalAllocationUnits.QuadPart * BytesPerAllocationUnit;
421             }
422 
423             /* No need to check for nullness ;-) */
424             lpTotalNumberOfFreeBytes->QuadPart = FileFsFullSize.ActualAvailableAllocationUnits.QuadPart * BytesPerAllocationUnit;
425 
426             return TRUE;
427         }
428     }
429 
430     /* Otherwise, fallback to normal size information */
431     Status = NtQueryVolumeInformationFile(RootHandle, &IoStatusBlock,
432                                           &FileFsSize, sizeof(FILE_FS_SIZE_INFORMATION),
433                                           FileFsSizeInformation);
434     NtClose(RootHandle);
435     if (!NT_SUCCESS(Status))
436     {
437         BaseSetLastNTError(Status);
438         return FALSE;
439     }
440 
441     /* Compute the size of an AU */
442     BytesPerAllocationUnit = FileFsSize.SectorsPerAllocationUnit * FileFsSize.BytesPerSector;
443 
444     /* And then return what was asked, available is free, the same! */
445     if (lpFreeBytesAvailableToCaller != NULL)
446     {
447         lpFreeBytesAvailableToCaller->QuadPart = FileFsSize.AvailableAllocationUnits.QuadPart * BytesPerAllocationUnit;
448     }
449 
450     if (lpTotalNumberOfBytes != NULL)
451     {
452         lpTotalNumberOfBytes->QuadPart = FileFsSize.TotalAllocationUnits.QuadPart * BytesPerAllocationUnit;
453     }
454 
455     if (lpTotalNumberOfFreeBytes != NULL)
456     {
457         lpTotalNumberOfFreeBytes->QuadPart = FileFsSize.AvailableAllocationUnits.QuadPart * BytesPerAllocationUnit;
458     }
459 
460     return TRUE;
461 }
462 
463 /*
464  * @implemented
465  */
466 UINT
467 WINAPI
468 GetDriveTypeA(IN LPCSTR lpRootPathName)
469 {
470     PWSTR RootPathU;
471 
472     if (lpRootPathName != NULL)
473     {
474         PUNICODE_STRING RootPathUStr;
475 
476         RootPathUStr = Basep8BitStringToStaticUnicodeString(lpRootPathName);
477         if (RootPathUStr == NULL)
478         {
479             return DRIVE_NO_ROOT_DIR;
480         }
481 
482         RootPathU = RootPathUStr->Buffer;
483     }
484     else
485     {
486         RootPathU = NULL;
487     }
488 
489     return GetDriveTypeW(RootPathU);
490 }
491 
492 /*
493  * @implemented
494  */
495 UINT
496 WINAPI
497 GetDriveTypeW(IN LPCWSTR lpRootPathName)
498 {
499     BOOL RetryOpen;
500     PCWSTR RootPath;
501     NTSTATUS Status;
502     WCHAR DriveLetter;
503     HANDLE RootHandle;
504     IO_STATUS_BLOCK IoStatusBlock;
505     OBJECT_ATTRIBUTES ObjectAttributes;
506     UNICODE_STRING PathName, VolumeString;
507     FILE_FS_DEVICE_INFORMATION FileFsDevice;
508     WCHAR Buffer[MAX_PATH], VolumeName[MAX_PATH];
509 
510     /* If no path, get one */
511     if (lpRootPathName == NULL)
512     {
513         RootPath = Buffer;
514         /* This will be current drive (<letter>:\ - drop the rest)*/
515         if (RtlGetCurrentDirectory_U(sizeof(Buffer), Buffer) > 3 * sizeof(WCHAR))
516         {
517             Buffer[3] = UNICODE_NULL;
518         }
519     }
520     else
521     {
522         /* Handle broken value */
523         if (lpRootPathName == (PVOID)-1)
524         {
525             return DRIVE_UNKNOWN;
526         }
527 
528         RootPath = lpRootPathName;
529         /* If provided path is 2-len, it might be a drive letter... */
530         if (wcslen(lpRootPathName) == 2)
531         {
532             /* Check it! */
533             DriveLetter = RtlUpcaseUnicodeChar(lpRootPathName[0]);
534             /* That's a drive letter! */
535             if (DriveLetter >= L'A' && DriveLetter <= L'Z' && lpRootPathName[1] == L':')
536             {
537                 /* Make it a volume */
538                 Buffer[0] = DriveLetter;
539                 Buffer[1] = L':';
540                 Buffer[2] = L'\\';
541                 Buffer[3] = UNICODE_NULL;
542                 RootPath = Buffer;
543             }
544         }
545     }
546 
547     /* If the provided looks like a DOS device... Like <letter>:\<0> */
548     DriveLetter = RtlUpcaseUnicodeChar(RootPath[0]);
549     /* We'll take the quick path!
550      * We'll find the device type looking at the device map (and types ;-))
551      * associated with the current process
552      */
553     if (DriveLetter >= L'A' && DriveLetter <= L'Z' && RootPath[1] == L':' &&
554         RootPath[2] == L'\\' && RootPath[3] == UNICODE_NULL)
555     {
556         USHORT Index;
557         PROCESS_DEVICEMAP_INFORMATION DeviceMap;
558 
559         /* Query the device map */
560         Status = NtQueryInformationProcess(NtCurrentProcess(),
561                                            ProcessDeviceMap,
562                                            &DeviceMap.Query,
563                                            sizeof(DeviceMap.Query),
564                                            NULL);
565         /* Zero output if we failed */
566         if (!NT_SUCCESS(Status))
567         {
568             RtlZeroMemory(&DeviceMap, sizeof(PROCESS_DEVICEMAP_INFORMATION));
569         }
570 
571         /* Get our index in the device map */
572         Index = DriveLetter - L'A';
573         /* Check we're in the device map (bit set) */
574         if (((1 << Index) & DeviceMap.Query.DriveMap) != 0)
575         {
576             /* Validate device type and return it */
577             if (DeviceMap.Query.DriveType[Index] >= DRIVE_REMOVABLE &&
578                 DeviceMap.Query.DriveType[Index] <= DRIVE_RAMDISK)
579             {
580                 return DeviceMap.Query.DriveType[Index];
581             }
582             /* Otherwise, return we don't know the type */
583             else
584             {
585                 return DRIVE_UNKNOWN;
586             }
587         }
588 
589         /* We couldn't find ourselves, do it the slow way */
590     }
591 
592     /* No path provided, use root */
593     if (lpRootPathName == NULL)
594     {
595         RootPath = L"\\";
596     }
597 
598     /* Convert to NT path */
599     if (!RtlDosPathNameToNtPathName_U(RootPath, &PathName, NULL, NULL))
600     {
601         return DRIVE_NO_ROOT_DIR;
602     }
603 
604     /* If not a directory, fail, we need a volume */
605     if (PathName.Buffer[(PathName.Length / sizeof(WCHAR)) - 1] != L'\\')
606     {
607         RtlFreeHeap(RtlGetProcessHeap(), 0, PathName.Buffer);
608         return DRIVE_NO_ROOT_DIR;
609     }
610 
611     /* We will work with a device object, so trim the trailing backslash now */
612     PathName.Length -= sizeof(WCHAR);
613 
614     /* Let's probe for it, by forcing open failure! */
615     RetryOpen = TRUE;
616     InitializeObjectAttributes(&ObjectAttributes, &PathName,
617                                OBJ_CASE_INSENSITIVE, NULL, NULL);
618     Status = NtOpenFile(&RootHandle, SYNCHRONIZE | FILE_READ_ATTRIBUTES,
619                         &ObjectAttributes, &IoStatusBlock,
620                         FILE_SHARE_READ | FILE_SHARE_WRITE,
621                         FILE_SYNCHRONOUS_IO_NONALERT | FILE_NON_DIRECTORY_FILE);
622     /* It properly failed! */
623     if (Status == STATUS_FILE_IS_A_DIRECTORY)
624     {
625         /* It might be a mount point, then, query for target */
626         if (BasepGetVolumeNameFromReparsePoint(lpRootPathName, VolumeName, MAX_PATH, NULL))
627         {
628             /* We'll reopen the target */
629             RtlInitUnicodeString(&VolumeString, VolumeName);
630             VolumeName[1] = L'?';
631             VolumeString.Length -= sizeof(WCHAR);
632             InitializeObjectAttributes(&ObjectAttributes, &VolumeString,
633                                        OBJ_CASE_INSENSITIVE, NULL, NULL);
634         }
635     }
636     else
637     {
638         /* heh. It worked? Or failed for whatever other reason?
639          * Check we have a directory if we get farther in path
640          */
641         PathName.Length += sizeof(WCHAR);
642         if (IsThisARootDirectory(0, &PathName))
643         {
644             /* Yes? Heh, then it's fine, keep our current handle */
645             RetryOpen = FALSE;
646         }
647         else
648         {
649             /* Then, retry to open without forcing non directory type */
650             PathName.Length -= sizeof(WCHAR);
651             if (NT_SUCCESS(Status))
652             {
653                 NtClose(RootHandle);
654             }
655         }
656     }
657 
658     /* Now, we retry without forcing file type - should work now */
659     if (RetryOpen)
660     {
661         Status = NtOpenFile(&RootHandle, SYNCHRONIZE | FILE_READ_ATTRIBUTES,
662                             &ObjectAttributes, &IoStatusBlock,
663                             FILE_SHARE_READ | FILE_SHARE_WRITE,
664                             FILE_SYNCHRONOUS_IO_NONALERT);
665     }
666 
667     /* We don't need path any longer */
668     RtlFreeHeap(RtlGetProcessHeap(), 0, PathName.Buffer);
669     if (!NT_SUCCESS(Status))
670     {
671         return DRIVE_NO_ROOT_DIR;
672     }
673 
674     /* Query the device for its type */
675     Status = NtQueryVolumeInformationFile(RootHandle,
676                                           &IoStatusBlock,
677                                           &FileFsDevice,
678                                           sizeof(FILE_FS_DEVICE_INFORMATION),
679                                           FileFsDeviceInformation);
680     /* No longer required */
681     NtClose(RootHandle);
682     if (!NT_SUCCESS(Status))
683     {
684         return DRIVE_UNKNOWN;
685     }
686 
687     /* Do we have a remote device? Return so! */
688     if ((FileFsDevice.Characteristics & FILE_REMOTE_DEVICE) == FILE_REMOTE_DEVICE)
689     {
690         return DRIVE_REMOTE;
691     }
692 
693     /* Check the device type */
694     switch (FileFsDevice.DeviceType)
695     {
696         /* CDROM, easy */
697         case FILE_DEVICE_CD_ROM:
698         case FILE_DEVICE_CD_ROM_FILE_SYSTEM:
699             return DRIVE_CDROM;
700 
701         /* Disk... */
702         case FILE_DEVICE_DISK:
703         case FILE_DEVICE_DISK_FILE_SYSTEM:
704             /* Removable media? Floppy is one */
705             if ((FileFsDevice.Characteristics & FILE_REMOVABLE_MEDIA) == FILE_REMOVABLE_MEDIA ||
706                 (FileFsDevice.Characteristics & FILE_FLOPPY_DISKETTE) == FILE_FLOPPY_DISKETTE)
707             {
708                 return DRIVE_REMOVABLE;
709             }
710             else
711             {
712                 return DRIVE_FIXED;
713             }
714 
715         /* Easy cases */
716         case FILE_DEVICE_NETWORK:
717         case FILE_DEVICE_NETWORK_FILE_SYSTEM:
718             return DRIVE_REMOTE;
719 
720         case FILE_DEVICE_VIRTUAL_DISK:
721             return DRIVE_RAMDISK;
722     }
723 
724     /* Nothing matching, just fail */
725     return DRIVE_UNKNOWN;
726 }
727 
728 /* EOF */
729