1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         ReactOS FS utility tool
4  * FILE:            base/applications/cmdutils/fsinfo.c
5  * PURPOSE:         FSutil file systems information
6  * PROGRAMMERS:     Pierre Schweitzer <pierre@reactos.org>
7  */
8 
9 #include "fsutil.h"
10 
11 /* Add handlers here for subcommands */
12 static HandlerProc DrivesMain;
13 static HandlerProc DriveTypeMain;
14 static HandlerProc VolumeInfoMain;
15 static HandlerProc NtfsInfoMain;
16 static HandlerProc StatisticsMain;
17 static HandlerItem HandlersList[] =
18 {
19     /* Proc, name, help */
20     { DrivesMain, _T("drives"), _T("Enumerates the drives") },
21     { DriveTypeMain, _T("drivetype"), _T("Provides the type of a drive") },
22     { VolumeInfoMain, _T("volumeinfo"), _T("Provides informations about a volume") },
23     { NtfsInfoMain, _T("ntfsinfo"), _T("Displays informations about a NTFS volume") },
24     { StatisticsMain, _T("statistics"), _T("Displays volume statistics") },
25 };
26 
27 static int
28 DrivesMain(int argc, const TCHAR *argv[])
29 {
30     UINT i;
31     DWORD Drives;
32 
33     /* Get the drives bitmap */
34     Drives = GetLogicalDrives();
35     if (Drives == 0)
36     {
37         PrintErrorMessage(GetLastError());
38         return 1;
39     }
40 
41     /* And output any found drive */
42     _ftprintf(stdout, _T("Drives:"));
43     for (i = 0; i < 26; i++)
44     {
45         if (Drives & (1 << i))
46         {
47             _ftprintf(stdout, _T(" %c:\\"), 'A' + i);
48         }
49     }
50     _ftprintf(stdout, _T("\n"));
51 
52     return 0;
53 }
54 
55 static int
56 DriveTypeMain(int argc, const TCHAR *argv[])
57 {
58     UINT Type;
59 
60     /* We need a volume (letter) */
61     if (argc < 2)
62     {
63         _ftprintf(stderr, _T("Usage: fsutil fsinfo drivetype <volume>\n"));
64         _ftprintf(stderr, _T("\tFor example: fsutil fsinfo drivetype c:\n"));
65         return 1;
66     }
67 
68     /* Get its drive type and make it readable */
69     Type = GetDriveType(argv[1]);
70     switch (Type)
71     {
72         case DRIVE_UNKNOWN:
73             _ftprintf(stdout, _T("%s - unknown drive type\n"), argv[1]);
74             break;
75 
76         case DRIVE_NO_ROOT_DIR:
77             _ftprintf(stdout, _T("%s - not a root directory\n"), argv[1]);
78             break;
79 
80         case DRIVE_REMOVABLE:
81             _ftprintf(stdout, _T("%s - removable drive\n"), argv[1]);
82             break;
83 
84         case DRIVE_FIXED:
85             _ftprintf(stdout, _T("%s - fixed drive\n"), argv[1]);
86             break;
87 
88         case DRIVE_REMOTE:
89             _ftprintf(stdout, _T("%s - remote or network drive\n"), argv[1]);
90             break;
91 
92         case DRIVE_CDROM:
93             _ftprintf(stdout, _T("%s - CD-ROM drive\n"), argv[1]);
94             break;
95 
96         case DRIVE_RAMDISK:
97             _ftprintf(stdout, _T("%s - RAM disk drive\n"), argv[1]);
98             break;
99     }
100 
101     return 0;
102 }
103 
104 static int
105 VolumeInfoMain(int argc, const TCHAR *argv[])
106 {
107     DWORD Serial, MaxComponentLen, Flags;
108     TCHAR VolumeName[MAX_PATH + 1], FileSystem[MAX_PATH + 1];
109 
110 #define HANDLE_FLAG(Flags, Flag, Desc) \
111     if (Flags & Flag)                  \
112         _ftprintf(stdout, Desc)
113 
114     /* We need a volume (path) */
115     if (argc < 2)
116     {
117         _ftprintf(stderr, _T("Usage: fsutil fsinfo volumeinfo <volume_path>\n"));
118         _ftprintf(stderr, _T("\tFor example: fsutil fsinfo volumeinfo c:\\\n"));
119         return 1;
120     }
121 
122     /* Gather information */
123     if (!GetVolumeInformation(argv[1], VolumeName,  MAX_PATH + 1, &Serial,
124                               &MaxComponentLen, &Flags, FileSystem, MAX_PATH + 1))
125     {
126         PrintErrorMessage(GetLastError());
127         return 1;
128     }
129 
130     /* Display general information */
131     _ftprintf(stdout, _T("Volume name: %s\n"), VolumeName);
132     _ftprintf(stdout, _T("Volume serial number: 0x%x\n"), Serial);
133     _ftprintf(stdout, _T("Maximum component length: %u\n"), MaxComponentLen);
134     _ftprintf(stdout, _T("File system name: %s\n"), FileSystem);
135 
136     /* Display specific flags */
137     HANDLE_FLAG(Flags, FILE_CASE_SENSITIVE_SEARCH, _T("Supports case-sensitive file names\n"));
138     HANDLE_FLAG(Flags, FILE_CASE_PRESERVED_NAMES, _T("Supports preserved case of file names\n"));
139     HANDLE_FLAG(Flags, FILE_UNICODE_ON_DISK, _T("Supports unicode file names\n"));
140     HANDLE_FLAG(Flags, FILE_PERSISTENT_ACLS, _T("Preserves and applies ACLs\n"));
141     HANDLE_FLAG(Flags, FILE_FILE_COMPRESSION, _T("Supports compression per file\n"));
142     HANDLE_FLAG(Flags, FILE_VOLUME_QUOTAS, _T("Supports disk quotas\n"));
143     HANDLE_FLAG(Flags, FILE_SUPPORTS_SPARSE_FILES, _T("Supports sparse files\n"));
144     HANDLE_FLAG(Flags, FILE_SUPPORTS_REPARSE_POINTS, _T("Supports reparse points\n"));
145     HANDLE_FLAG(Flags, FILE_VOLUME_IS_COMPRESSED, _T("Is a compressed volume\n"));
146     HANDLE_FLAG(Flags, FILE_SUPPORTS_OBJECT_IDS, _T("Supports object identifiers\n"));
147     HANDLE_FLAG(Flags, FILE_SUPPORTS_ENCRYPTION, _T("Supports the Encrypted File System (EFS)\n"));
148     HANDLE_FLAG(Flags, FILE_NAMED_STREAMS, _T("Supports named streams\n"));
149     HANDLE_FLAG(Flags, FILE_READ_ONLY_VOLUME, _T("Is a read-only volume\n"));
150     HANDLE_FLAG(Flags, FILE_SEQUENTIAL_WRITE_ONCE, _T("Supports a single sequential write\n"));
151     HANDLE_FLAG(Flags, FILE_SUPPORTS_TRANSACTIONS, _T("Supports transactions\n"));
152     HANDLE_FLAG(Flags, FILE_SUPPORTS_HARD_LINKS, _T("Supports hard links\n"));
153     HANDLE_FLAG(Flags, FILE_SUPPORTS_EXTENDED_ATTRIBUTES, _T("Supports extended attributes\n"));
154     HANDLE_FLAG(Flags, FILE_SUPPORTS_OPEN_BY_FILE_ID, _T("Supports opening files per file identifier\n"));
155     HANDLE_FLAG(Flags, FILE_SUPPORTS_USN_JOURNAL, _T("Supports Update Sequence Number (USN) journals\n"));
156     HANDLE_FLAG(Flags, FILE_DAX_VOLUME, _T("Is a direct access volume\n"));
157 
158 #undef HANDLE_FLAGS
159 
160     return 0;
161 }
162 
163 static int
164 NtfsInfoMain(int argc, const TCHAR *argv[])
165 {
166     HANDLE Volume;
167     DWORD BytesRead;
168     struct
169     {
170         NTFS_VOLUME_DATA_BUFFER;
171         NTFS_EXTENDED_VOLUME_DATA;
172     } Data;
173 
174     /* We need a volume (letter or GUID) */
175     if (argc < 2)
176     {
177         _ftprintf(stderr, _T("Usage: fsutil fsinfo ntfsinfo <volume>\n"));
178         _ftprintf(stderr, _T("\tFor example: fsutil fsinfo ntfsinfo c:\n"));
179         return 1;
180     }
181 
182     /* Get a handle for the volume */
183     Volume = OpenVolume(argv[1], FALSE, TRUE);
184     if (Volume == INVALID_HANDLE_VALUE)
185     {
186         return 1;
187     }
188 
189     /* And query the NTFS data */
190     if (DeviceIoControl(Volume, FSCTL_GET_NTFS_VOLUME_DATA, NULL, 0, &Data,
191                         sizeof(Data), &BytesRead, NULL) == FALSE)
192     {
193         PrintErrorMessage(GetLastError());
194         CloseHandle(Volume);
195         return 1;
196     }
197 
198     /* We no longer need the volume */
199     CloseHandle(Volume);
200 
201     /* Dump data */
202     _ftprintf(stdout, _T("NTFS volume serial number:\t\t0x%0.16I64x\n"), Data.VolumeSerialNumber.QuadPart);
203     /* Only print version if extended structure was returned */
204     if (BytesRead > sizeof(NTFS_VOLUME_DATA_BUFFER))
205     {
206         _ftprintf(stdout, _T("Version:\t\t\t\t%u.%u\n"), Data.MajorVersion, Data.MinorVersion);
207     }
208     _ftprintf(stdout, _T("Number of sectors:\t\t\t0x%0.16I64x\n"), Data.NumberSectors.QuadPart);
209     _ftprintf(stdout, _T("Total number of clusters:\t\t0x%0.16I64x\n"), Data.TotalClusters.QuadPart);
210     _ftprintf(stdout, _T("Free clusters:\t\t\t\t0x%0.16I64x\n"), Data.FreeClusters.QuadPart);
211     _ftprintf(stdout, _T("Total number of reserved clusters:\t0x%0.16I64x\n"), Data.TotalReserved.QuadPart);
212     _ftprintf(stdout, _T("Bytes per sector:\t\t\t%d\n"), Data.BytesPerSector);
213     _ftprintf(stdout, _T("Bytes per cluster:\t\t\t%d\n"), Data.BytesPerCluster);
214     _ftprintf(stdout, _T("Bytes per file record segment:\t\t%d\n"), Data.BytesPerFileRecordSegment);
215     _ftprintf(stdout, _T("Clusters per file record segment:\t%d\n"), Data.ClustersPerFileRecordSegment);
216     _ftprintf(stdout, _T("MFT valid data length:\t\t\t0x%0.16I64x\n"), Data.MftValidDataLength.QuadPart);
217     _ftprintf(stdout, _T("MFT start LCN:\t\t\t\t0x%0.16I64x\n"), Data.MftStartLcn.QuadPart);
218     _ftprintf(stdout, _T("MFT2 start LCN:\t\t\t\t0x%0.16I64x\n"), Data.Mft2StartLcn.QuadPart);
219     _ftprintf(stdout, _T("MFT zone start:\t\t\t\t0x%0.16I64x\n"), Data.MftZoneStart.QuadPart);
220     _ftprintf(stdout, _T("MFT zone end:\t\t\t\t0x%0.16I64x\n"), Data.MftZoneEnd.QuadPart);
221 
222     return 0;
223 }
224 
225 #define DUMP_VALUE(stats, value) fprintf(stdout, "%s: %lu\n", #value, stats->value)
226 
227 static void
228 DumpBase(PFILESYSTEM_STATISTICS Base, TCHAR * Name)
229 {
230     /* Print FS name */
231     _ftprintf(stdout, _T("File system type: %s\n\n"), Name);
232 
233     /* And then, dump any base stat */
234     DUMP_VALUE(Base, UserFileReads);
235     DUMP_VALUE(Base, UserFileReadBytes);
236     DUMP_VALUE(Base, UserDiskReads);
237     DUMP_VALUE(Base, UserFileWrites);
238     DUMP_VALUE(Base, UserFileWriteBytes);
239     DUMP_VALUE(Base, UserDiskWrites);
240     DUMP_VALUE(Base, MetaDataReads);
241     DUMP_VALUE(Base, MetaDataReadBytes);
242     DUMP_VALUE(Base, MetaDataDiskReads);
243     DUMP_VALUE(Base, MetaDataWrites);
244     DUMP_VALUE(Base, MetaDataWriteBytes);
245     DUMP_VALUE(Base, MetaDataDiskWrites);
246 
247     _ftprintf(stdout, _T("\n"));
248 }
249 
250 static void
251 DumpExFat(PVOID Statistics, PVOID Specific)
252 {
253     PEXFAT_STATISTICS ExFat;
254     PFILESYSTEM_STATISTICS Base;
255 
256     Base = Statistics;
257     ExFat = Specific;
258 
259     /* First, display the generic stats */
260     DumpBase(Base, _T("EXFAT"));
261 
262     /* Then, display the EXFAT specific ones */
263     DUMP_VALUE(ExFat, CreateHits);
264     DUMP_VALUE(ExFat, SuccessfulCreates);
265     DUMP_VALUE(ExFat, FailedCreates);
266     DUMP_VALUE(ExFat, NonCachedReads);
267     DUMP_VALUE(ExFat, NonCachedReadBytes);
268     DUMP_VALUE(ExFat, NonCachedWrites);
269     DUMP_VALUE(ExFat, NonCachedWriteBytes);
270     DUMP_VALUE(ExFat, NonCachedDiskReads);
271     DUMP_VALUE(ExFat, NonCachedDiskWrites);
272 }
273 
274 static void
275 DumpFat(PVOID Statistics, PVOID Specific)
276 {
277     PFAT_STATISTICS Fat;
278     PFILESYSTEM_STATISTICS Base;
279 
280     Base = Statistics;
281     Fat = Specific;
282 
283     /* First, display the generic stats */
284     DumpBase(Base, _T("FAT"));
285 
286     /* Then, display the FAT specific ones */
287     DUMP_VALUE(Fat, CreateHits);
288     DUMP_VALUE(Fat, SuccessfulCreates);
289     DUMP_VALUE(Fat, FailedCreates);
290     DUMP_VALUE(Fat, NonCachedReads);
291     DUMP_VALUE(Fat, NonCachedReadBytes);
292     DUMP_VALUE(Fat, NonCachedWrites);
293     DUMP_VALUE(Fat, NonCachedWriteBytes);
294     DUMP_VALUE(Fat, NonCachedDiskReads);
295     DUMP_VALUE(Fat, NonCachedDiskWrites);
296 }
297 
298 static void
299 DumpNtfs(PVOID Statistics, PVOID Specific)
300 {
301     PNTFS_STATISTICS Ntfs;
302     PFILESYSTEM_STATISTICS Base;
303 
304     Base = Statistics;
305     Ntfs = Specific;
306 
307     /* First, display the generic stats */
308     DumpBase(Base, _T("NTFS"));
309 
310     /* Then, display the NTFS specific ones */
311     DUMP_VALUE(Ntfs, MftReads);
312     DUMP_VALUE(Ntfs, MftReadBytes);
313     DUMP_VALUE(Ntfs, MftWrites);
314     DUMP_VALUE(Ntfs, MftWriteBytes);
315     DUMP_VALUE(Ntfs, Mft2Writes);
316     DUMP_VALUE(Ntfs, Mft2WriteBytes);
317     DUMP_VALUE(Ntfs, RootIndexReads);
318     DUMP_VALUE(Ntfs, RootIndexReadBytes);
319     DUMP_VALUE(Ntfs, RootIndexWrites);
320     DUMP_VALUE(Ntfs, RootIndexWriteBytes);
321     DUMP_VALUE(Ntfs, BitmapReads);
322     DUMP_VALUE(Ntfs, BitmapReadBytes);
323     DUMP_VALUE(Ntfs, BitmapWrites);
324     DUMP_VALUE(Ntfs, BitmapWriteBytes);
325     DUMP_VALUE(Ntfs, MftBitmapReads);
326     DUMP_VALUE(Ntfs, MftBitmapReadBytes);
327     DUMP_VALUE(Ntfs, MftBitmapWrites);
328     DUMP_VALUE(Ntfs, MftBitmapWriteBytes);
329     DUMP_VALUE(Ntfs, UserIndexReads);
330     DUMP_VALUE(Ntfs, UserIndexReadBytes);
331     DUMP_VALUE(Ntfs, UserIndexWrites);
332     DUMP_VALUE(Ntfs, UserIndexWriteBytes);
333     DUMP_VALUE(Ntfs, LogFileReads);
334     DUMP_VALUE(Ntfs, LogFileReadBytes);
335     DUMP_VALUE(Ntfs, LogFileWrites);
336     DUMP_VALUE(Ntfs, LogFileWriteBytes);
337 }
338 
339 #define GET_NEXT(stats, length, iter, type) (type)((ULONG_PTR)stats + (length * iter))
340 #define SUM_VALUE(stats, new, value) stats->value += new->value
341 
342 inline int
343 ValidateSizes(ULONG Length, DWORD ProcCount, DWORD BytesRead, DWORD SpecificSize)
344 {
345     /* Check whether we could read our base length for every processor */
346     if (Length * ProcCount > BytesRead)
347     {
348         _ftprintf(stderr, _T("Only performed a partial read: %lu (expected: %lu)\n"), BytesRead, Length * ProcCount);
349         return 1;
350     }
351 
352     /* Check whether this covers a specific entry and a generic entry for every processor */
353     if ((sizeof(FILESYSTEM_STATISTICS) + SpecificSize) * ProcCount > BytesRead)
354     {
355         _ftprintf(stderr, _T("Only performed a partial read: %lu (expected: %lu)\n"), BytesRead, (sizeof(FILESYSTEM_STATISTICS) + SpecificSize));
356         return 1;
357     }
358 
359     return 0;
360 }
361 
362 inline void
363 SumBase(PFILESYSTEM_STATISTICS Base, PFILESYSTEM_STATISTICS NextBase)
364 {
365     /* Sum any entry in the generic structures */
366     SUM_VALUE(Base, NextBase, UserFileReads);
367     SUM_VALUE(Base, NextBase, UserFileReadBytes);
368     SUM_VALUE(Base, NextBase, UserDiskReads);
369     SUM_VALUE(Base, NextBase, UserFileWrites);
370     SUM_VALUE(Base, NextBase, UserFileWriteBytes);
371     SUM_VALUE(Base, NextBase, UserDiskWrites);
372     SUM_VALUE(Base, NextBase, MetaDataReads);
373     SUM_VALUE(Base, NextBase, MetaDataReadBytes);
374     SUM_VALUE(Base, NextBase, MetaDataDiskReads);
375     SUM_VALUE(Base, NextBase, MetaDataWrites);
376     SUM_VALUE(Base, NextBase, MetaDataWriteBytes);
377     SUM_VALUE(Base, NextBase, MetaDataDiskWrites);
378 }
379 
380 static int
381 SumExFat(PVOID Statistics, PVOID Specific, ULONG Length, DWORD ProcCount, DWORD BytesRead)
382 {
383     DWORD i;
384     PEXFAT_STATISTICS ExFat;
385     PFILESYSTEM_STATISTICS Base;
386 
387     /* First, validate we won't read beyond allocation */
388     if (ValidateSizes(Length, ProcCount, BytesRead, sizeof(EXFAT_STATISTICS)))
389     {
390         return 1;
391     }
392 
393     Base = Statistics;
394     ExFat = Specific;
395 
396     /* And for every processor, sum every relevant value in first entry */
397     for (i = 1; i < ProcCount; ++i)
398     {
399         PEXFAT_STATISTICS NextExFat;
400         PFILESYSTEM_STATISTICS NextBase;
401 
402         NextBase = GET_NEXT(Base, Length, i, PFILESYSTEM_STATISTICS);
403         NextExFat = GET_NEXT(ExFat, Length, i, PEXFAT_STATISTICS);
404 
405         /* Generic first */
406         SumBase(Base, NextBase);
407         /* Specific then */
408         SUM_VALUE(ExFat, NextExFat, CreateHits);
409         SUM_VALUE(ExFat, NextExFat, SuccessfulCreates);
410         SUM_VALUE(ExFat, NextExFat, FailedCreates);
411         SUM_VALUE(ExFat, NextExFat, NonCachedReads);
412         SUM_VALUE(ExFat, NextExFat, NonCachedReadBytes);
413         SUM_VALUE(ExFat, NextExFat, NonCachedWrites);
414         SUM_VALUE(ExFat, NextExFat, NonCachedWriteBytes);
415         SUM_VALUE(ExFat, NextExFat, NonCachedDiskReads);
416         SUM_VALUE(ExFat, NextExFat, NonCachedDiskWrites);
417     }
418 
419     return 0;
420 }
421 
422 static int
423 SumFat(PVOID Statistics, PVOID Specific, ULONG Length, DWORD ProcCount, DWORD BytesRead)
424 {
425     DWORD i;
426     PFAT_STATISTICS Fat;
427     PFILESYSTEM_STATISTICS Base;
428 
429     /* First, validate we won't read beyond allocation */
430     if (ValidateSizes(Length, ProcCount, BytesRead, sizeof(FAT_STATISTICS)))
431     {
432         return 1;
433     }
434 
435     Base = Statistics;
436     Fat = Specific;
437 
438     /* And for every processor, sum every relevant value in first entry */
439     for (i = 1; i < ProcCount; ++i)
440     {
441         PFAT_STATISTICS NextFat;
442         PFILESYSTEM_STATISTICS NextBase;
443 
444         NextBase = GET_NEXT(Base, Length, i, PFILESYSTEM_STATISTICS);
445         NextFat = GET_NEXT(Fat, Length, i, PFAT_STATISTICS);
446 
447         /* Generic first */
448         SumBase(Base, NextBase);
449         /* Specific then */
450         SUM_VALUE(Fat, NextFat, CreateHits);
451         SUM_VALUE(Fat, NextFat, SuccessfulCreates);
452         SUM_VALUE(Fat, NextFat, FailedCreates);
453         SUM_VALUE(Fat, NextFat, NonCachedReads);
454         SUM_VALUE(Fat, NextFat, NonCachedReadBytes);
455         SUM_VALUE(Fat, NextFat, NonCachedWrites);
456         SUM_VALUE(Fat, NextFat, NonCachedWriteBytes);
457         SUM_VALUE(Fat, NextFat, NonCachedDiskReads);
458         SUM_VALUE(Fat, NextFat, NonCachedDiskWrites);
459     }
460 
461     return 0;
462 }
463 
464 static int
465 SumNtfs(PVOID Statistics, PVOID Specific, ULONG Length, DWORD ProcCount, DWORD BytesRead)
466 {
467     DWORD i;
468     PNTFS_STATISTICS Ntfs;
469     PFILESYSTEM_STATISTICS Base;
470 
471     /* First, validate we won't read beyond allocation */
472     if (ValidateSizes(Length, ProcCount, BytesRead, sizeof(NTFS_STATISTICS)))
473     {
474         return 1;
475     }
476 
477     Base = Statistics;
478     Ntfs = Specific;
479 
480     /* And for every processor, sum every relevant value in first entry */
481     for (i = 1; i < ProcCount; ++i)
482     {
483         PNTFS_STATISTICS NextNtfs;
484         PFILESYSTEM_STATISTICS NextBase;
485 
486         NextBase = GET_NEXT(Base, Length, i, PFILESYSTEM_STATISTICS);
487         NextNtfs = GET_NEXT(Ntfs, Length, i, PNTFS_STATISTICS);
488 
489         /* Generic first */
490         SumBase(Base, NextBase);
491         /* Specific then */
492         SUM_VALUE(Ntfs, NextNtfs, MftReads);
493         SUM_VALUE(Ntfs, NextNtfs, MftReadBytes);
494         SUM_VALUE(Ntfs, NextNtfs, MftWrites);
495         SUM_VALUE(Ntfs, NextNtfs, MftWriteBytes);
496         SUM_VALUE(Ntfs, NextNtfs, Mft2Writes);
497         SUM_VALUE(Ntfs, NextNtfs, Mft2WriteBytes);
498         SUM_VALUE(Ntfs, NextNtfs, RootIndexReads);
499         SUM_VALUE(Ntfs, NextNtfs, RootIndexReadBytes);
500         SUM_VALUE(Ntfs, NextNtfs, RootIndexWrites);
501         SUM_VALUE(Ntfs, NextNtfs, RootIndexWriteBytes);
502         SUM_VALUE(Ntfs, NextNtfs, BitmapReads);
503         SUM_VALUE(Ntfs, NextNtfs, BitmapReadBytes);
504         SUM_VALUE(Ntfs, NextNtfs, BitmapWrites);
505         SUM_VALUE(Ntfs, NextNtfs, BitmapWriteBytes);
506         SUM_VALUE(Ntfs, NextNtfs, MftBitmapReads);
507         SUM_VALUE(Ntfs, NextNtfs, MftBitmapReadBytes);
508         SUM_VALUE(Ntfs, NextNtfs, MftBitmapWrites);
509         SUM_VALUE(Ntfs, NextNtfs, MftBitmapWriteBytes);
510         SUM_VALUE(Ntfs, NextNtfs, UserIndexReads);
511         SUM_VALUE(Ntfs, NextNtfs, UserIndexReadBytes);
512         SUM_VALUE(Ntfs, NextNtfs, UserIndexWrites);
513         SUM_VALUE(Ntfs, NextNtfs, UserIndexWriteBytes);
514         SUM_VALUE(Ntfs, NextNtfs, LogFileReads);
515         SUM_VALUE(Ntfs, NextNtfs, LogFileReadBytes);
516         SUM_VALUE(Ntfs, NextNtfs, LogFileWrites);
517         SUM_VALUE(Ntfs, NextNtfs, LogFileWriteBytes);
518     }
519 
520     return 0;
521 }
522 
523 static int
524 StatisticsMain(int argc, const TCHAR *argv[])
525 {
526     HANDLE Volume;
527     SYSTEM_INFO SystemInfo;
528     FILESYSTEM_STATISTICS Buffer;
529     PFILESYSTEM_STATISTICS Statistics;
530     DWORD BytesRead, Length, ProcCount;
531     /* +1 because 0 isn't assigned to a filesystem */
532     void (*DumpFct[FILESYSTEM_STATISTICS_TYPE_EXFAT + 1])(PVOID, PVOID) = { NULL, DumpNtfs, DumpFat, DumpExFat };
533     int (*SumFct[FILESYSTEM_STATISTICS_TYPE_EXFAT + 1])(PVOID, PVOID, ULONG, DWORD, DWORD) = { NULL, SumNtfs, SumFat, SumExFat };
534 
535     /* We need a volume (letter or GUID) */
536     if (argc < 2)
537     {
538         _ftprintf(stderr, _T("Usage: fsutil fsinfo statistics <volume>\n"));
539         _ftprintf(stderr, _T("\tFor example: fsutil fsinfo statistics c:\n"));
540         return 1;
541     }
542 
543     /* Get a handle for the volume */
544     Volume = OpenVolume(argv[1], FALSE, FALSE);
545     if (Volume == INVALID_HANDLE_VALUE)
546     {
547         return 1;
548     }
549 
550     /* And query the statistics status - this call is expected to fail */
551     Statistics = &Buffer;
552     Length = sizeof(Buffer);
553     /* Assume we have a single proc for now */
554     ProcCount = 1;
555     if (DeviceIoControl(Volume, FSCTL_FILESYSTEM_GET_STATISTICS, NULL, 0, Statistics,
556                         Length, &BytesRead, NULL) == FALSE)
557     {
558         DWORD Error;
559 
560         /* Check we failed because we provided a too small buffer */
561         Error = GetLastError();
562         if (Error == ERROR_MORE_DATA)
563         {
564             /* Get proc count */
565             GetSystemInfo(&SystemInfo);
566             ProcCount = SystemInfo.dwNumberOfProcessors;
567             /* Get the maximum size to allocate: it's the total size (generic + specific) for every proc */
568             Length = Statistics->SizeOfCompleteStructure * ProcCount;
569 
570             Statistics = LocalAlloc(LPTR, Length);
571             if (Statistics == NULL)
572             {
573                 _ftprintf(stderr, _T("Failed to allocate memory!\n"));
574                 CloseHandle(Volume);
575                 return 1;
576             }
577 
578             /* Reissue the FSCTL, it's supposed to succeed now! */
579             if (DeviceIoControl(Volume, FSCTL_FILESYSTEM_GET_STATISTICS, NULL, 0, Statistics,
580                                 Length, &BytesRead, NULL) == FALSE)
581             {
582                 PrintErrorMessage(GetLastError());
583                 LocalFree(Statistics);
584                 CloseHandle(Volume);
585                 return 1;
586             }
587         }
588         else
589         {
590             PrintErrorMessage(Error);
591             CloseHandle(Volume);
592             return 1;
593         }
594     }
595 
596     /* No need to deal with the volume any longer */
597     CloseHandle(Volume);
598 
599     /* We only support FAT, EXFAT and NTFS for now */
600     if (Statistics->FileSystemType > FILESYSTEM_STATISTICS_TYPE_EXFAT || Statistics->FileSystemType < FILESYSTEM_STATISTICS_TYPE_NTFS)
601     {
602         _ftprintf(stderr, _T("Unrecognized file system type: %d\n"), Statistics->FileSystemType);
603         if (Statistics != &Buffer)
604         {
605             LocalFree(Statistics);
606         }
607 
608         return 1;
609     }
610 
611     /* Sum all the statistics (both generic and specific) from all the processors in the first entry */
612     if (SumFct[Statistics->FileSystemType](Statistics, (PVOID)((ULONG_PTR)Statistics + sizeof(FILESYSTEM_STATISTICS)),
613                                            Statistics->SizeOfCompleteStructure, ProcCount, BytesRead))
614     {
615         if (Statistics != &Buffer)
616         {
617             LocalFree(Statistics);
618         }
619 
620         return 1;
621     }
622 
623     /* And finally, display the statistics (both generic and specific) */
624     DumpFct[Statistics->FileSystemType](Statistics, (PVOID)((ULONG_PTR)Statistics + sizeof(FILESYSTEM_STATISTICS)));
625 
626     /* If we allocated memory, release it */
627     if (Statistics != &Buffer)
628     {
629         LocalFree(Statistics);
630     }
631 
632     return 0;
633 }
634 
635 static void
636 PrintUsage(const TCHAR * Command)
637 {
638     PrintDefaultUsage(_T(" FSINFO "), Command, (HandlerItem *)&HandlersList,
639                       (sizeof(HandlersList) / sizeof(HandlersList[0])));
640 }
641 
642 int
643 FsInfoMain(int argc, const TCHAR *argv[])
644 {
645     return FindHandler(argc, argv, (HandlerItem *)&HandlersList,
646                        (sizeof(HandlersList) / sizeof(HandlersList[0])),
647                        PrintUsage);
648 }
649