694 lines
19 KiB
C
694 lines
19 KiB
C
|
/*++
|
||
|
|
||
|
Copyright (c) 2000 Microsoft Corporation
|
||
|
|
||
|
Module Name:
|
||
|
|
||
|
fsinfo.c
|
||
|
|
||
|
Abstract:
|
||
|
|
||
|
This file contains code for commands that affect
|
||
|
information specific to the file system.
|
||
|
|
||
|
Author:
|
||
|
|
||
|
Wesley Witt [wesw] 1-March-2000
|
||
|
|
||
|
Revision History:
|
||
|
|
||
|
--*/
|
||
|
|
||
|
#include <precomp.h>
|
||
|
|
||
|
|
||
|
INT
|
||
|
FsInfoHelp(
|
||
|
IN INT argc,
|
||
|
IN PWSTR argv[]
|
||
|
)
|
||
|
{
|
||
|
DisplayMsg( MSG_USAGE_FSINFO );
|
||
|
return EXIT_CODE_SUCCESS;
|
||
|
}
|
||
|
|
||
|
typedef struct _NTFS_FILE_SYSTEM_STATISTICS {
|
||
|
FILESYSTEM_STATISTICS Common;
|
||
|
NTFS_STATISTICS Ntfs;
|
||
|
UCHAR Pad[64-(sizeof(FILESYSTEM_STATISTICS)+sizeof(NTFS_STATISTICS))%64];
|
||
|
} NTFS_FILE_SYSTEM_STATISTICS, *PNTFS_FILE_SYSTEM_STATISTICS;
|
||
|
|
||
|
typedef struct _FAT_FILE_SYSTEM_STATISTICS {
|
||
|
FILESYSTEM_STATISTICS Common;
|
||
|
FAT_STATISTICS Fat;
|
||
|
UCHAR Pad[64-(sizeof(FILESYSTEM_STATISTICS)+sizeof(NTFS_STATISTICS))%64];
|
||
|
} FAT_FILE_SYSTEM_STATISTICS, *PFAT_FILE_SYSTEM_STATISTICS;
|
||
|
|
||
|
|
||
|
|
||
|
ULONGLONG
|
||
|
FsStat(
|
||
|
IN PVOID FsStats,
|
||
|
IN ULONG FsSize,
|
||
|
IN ULONG Offset,
|
||
|
IN ULONG NumProcs
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine iterates through the file system statistics structure
|
||
|
and accumulates a total amount for a given statistic field.
|
||
|
|
||
|
Note: This function assumes that the width of the field being accumulated
|
||
|
is a ULONG. If this assumption is invalidated then a width argument
|
||
|
should be added to this function.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
FsStats - Pointer to an array of file system statistics structures
|
||
|
FsSize - Size of the individual file system statistics structures in the array
|
||
|
Offset - Offset of the desired field from the beginning of the file system statistics structure
|
||
|
NumProcs - Number of processors on the machine where the stats were gathered
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
Accumulated total.
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
ULONG i;
|
||
|
ULONGLONG Total = 0;
|
||
|
|
||
|
for (i=0; i<NumProcs; i++) {
|
||
|
Total += *(PULONG)((PUCHAR)FsStats + Offset);
|
||
|
FsStats = (PVOID)((PUCHAR)FsStats + FsSize);
|
||
|
}
|
||
|
|
||
|
return Total;
|
||
|
}
|
||
|
|
||
|
|
||
|
INT
|
||
|
ListDrives(
|
||
|
IN INT argc,
|
||
|
IN PWSTR argv[]
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine lists all the drive names in the system.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
argc - The argument count
|
||
|
argv - Array of Strings of the form : 'fscutl lsdrv'.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
DWORD nBufferLen = MAX_PATH;
|
||
|
DWORD Length;
|
||
|
LPWSTR lpBuffer = NULL;
|
||
|
WORD Index;
|
||
|
INT ExitCode = EXIT_CODE_SUCCESS;
|
||
|
|
||
|
try {
|
||
|
|
||
|
if (argc != 0) {
|
||
|
DisplayMsg( MSG_LISTDRIVES_USAGE );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
lpBuffer = (LPWSTR) malloc( nBufferLen * sizeof(WCHAR) );
|
||
|
if (!lpBuffer) {
|
||
|
DisplayErrorMsg( ERROR_NOT_ENOUGH_MEMORY );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
Length = GetLogicalDriveStrings( nBufferLen, lpBuffer );
|
||
|
if (!Length) {
|
||
|
DisplayError();
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
wprintf( L"\n" );
|
||
|
DisplayMsg( MSG_DRIVES );
|
||
|
for ( Index = 0 ; Index < Length; Index ++ ) {
|
||
|
wprintf( L"%c", lpBuffer[Index] );
|
||
|
}
|
||
|
wprintf( L"\n" );
|
||
|
|
||
|
} finally {
|
||
|
|
||
|
if (lpBuffer) {
|
||
|
free( lpBuffer );
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return ExitCode;
|
||
|
}
|
||
|
|
||
|
|
||
|
INT
|
||
|
DriveType(
|
||
|
IN INT argc,
|
||
|
IN PWSTR argv[]
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine describes the drive type of the drive passed in.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
argc - The argument count
|
||
|
argv - Array of Strings of the form :
|
||
|
'fscutl dt <drive name>' or
|
||
|
'fscutl drivetype <drive name>'
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
if (argc != 1) {
|
||
|
DisplayMsg( MSG_USAGE_DRIVETYPE );
|
||
|
if (argc != 0) {
|
||
|
return EXIT_CODE_FAILURE;
|
||
|
} else {
|
||
|
return EXIT_CODE_SUCCESS;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
DisplayMsg( GetDriveType( argv[0] ) + MSG_DRIVE_UNKNOWN, argv[0] );
|
||
|
|
||
|
return EXIT_CODE_SUCCESS;
|
||
|
}
|
||
|
|
||
|
|
||
|
INT
|
||
|
VolumeInfo(
|
||
|
IN INT argc,
|
||
|
IN PWSTR argv[]
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine provides the information about the Volume.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
argc - The argument count
|
||
|
argv - Array of Strings of the form :
|
||
|
'fscutl infov <root pathname>'.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
LPWSTR lpVolumeNameBuffer = NULL; // address of name of the volume
|
||
|
DWORD nVolumeNameSize; // length of lpVolumeNameBuffer
|
||
|
LPDWORD lpVolumeSerialNumber = NULL; // address of volume serial number
|
||
|
LPDWORD lpMaximumComponentLength = NULL; // address of system's maximum
|
||
|
// filename length
|
||
|
LPDWORD lpFileSystemFlags = NULL; // address of file system flags
|
||
|
LPWSTR lpFileSystemNameBuffer = NULL; // address of name of file system
|
||
|
DWORD nFileSystemNameSize; // length of lpFileSystemNameBuffer
|
||
|
BOOL Status; // return status
|
||
|
DWORD dwMask; // FileSystem Flag Mask
|
||
|
DWORD Index;
|
||
|
DWORD FsFlag;
|
||
|
INT ExitCode = EXIT_CODE_SUCCESS;
|
||
|
|
||
|
|
||
|
try {
|
||
|
|
||
|
if (argc != 1) {
|
||
|
DisplayMsg( MSG_USAGE_INFOV );
|
||
|
if (argc != 0) {
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
}
|
||
|
leave ;
|
||
|
}
|
||
|
|
||
|
nVolumeNameSize = MAX_PATH;
|
||
|
|
||
|
lpVolumeNameBuffer = (LPWSTR) malloc ( MAX_PATH * sizeof(WCHAR) );
|
||
|
if (lpVolumeNameBuffer == NULL) {
|
||
|
DisplayErrorMsg( ERROR_NOT_ENOUGH_MEMORY );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
lpVolumeSerialNumber = (LPDWORD) malloc( sizeof(DWORD) );
|
||
|
if (lpVolumeSerialNumber == NULL) {
|
||
|
DisplayErrorMsg( ERROR_NOT_ENOUGH_MEMORY );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
lpFileSystemFlags = (LPDWORD) malloc ( sizeof(DWORD) );
|
||
|
if (lpFileSystemFlags == NULL) {
|
||
|
DisplayErrorMsg( ERROR_NOT_ENOUGH_MEMORY );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
lpMaximumComponentLength = (LPDWORD) malloc ( sizeof(DWORD) );
|
||
|
if (lpMaximumComponentLength == NULL) {
|
||
|
DisplayErrorMsg( ERROR_NOT_ENOUGH_MEMORY );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
nFileSystemNameSize = MAX_PATH;
|
||
|
lpFileSystemNameBuffer = (LPWSTR) malloc ( MAX_PATH * sizeof(WCHAR) );
|
||
|
if (lpFileSystemNameBuffer == NULL) {
|
||
|
DisplayErrorMsg( ERROR_NOT_ENOUGH_MEMORY );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
Status = GetVolumeInformation (
|
||
|
argv[0],
|
||
|
lpVolumeNameBuffer,
|
||
|
nVolumeNameSize,
|
||
|
lpVolumeSerialNumber,
|
||
|
lpMaximumComponentLength,
|
||
|
lpFileSystemFlags,
|
||
|
lpFileSystemNameBuffer,
|
||
|
nFileSystemNameSize
|
||
|
);
|
||
|
if (!Status) {
|
||
|
DisplayError();
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
DisplayMsg( MSG_VOLNAME, lpVolumeNameBuffer );
|
||
|
DisplayMsg( MSG_SERIALNO, *lpVolumeSerialNumber );
|
||
|
DisplayMsg( MSG_MAX_COMP_LEN, *lpMaximumComponentLength );
|
||
|
DisplayMsg( MSG_FS_NAME, lpFileSystemNameBuffer );
|
||
|
|
||
|
dwMask = 1 ;
|
||
|
FsFlag = *lpFileSystemFlags;
|
||
|
|
||
|
for ( Index=0 ; Index<32 ; Index++ ) {
|
||
|
switch (FsFlag & dwMask) {
|
||
|
case FILE_CASE_SENSITIVE_SEARCH:
|
||
|
DisplayMsg( MSG_FILE_CASE_SENSITIVE_SEARCH );
|
||
|
break;
|
||
|
|
||
|
case FILE_CASE_PRESERVED_NAMES:
|
||
|
DisplayMsg( MSG_FILE_CASE_PRESERVED_NAMES );
|
||
|
break;
|
||
|
|
||
|
case FILE_UNICODE_ON_DISK:
|
||
|
DisplayMsg( MSG_FILE_UNICODE_ON_DISK );
|
||
|
break;
|
||
|
|
||
|
case FILE_PERSISTENT_ACLS:
|
||
|
DisplayMsg( MSG_FILE_PERSISTENT_ACLS );
|
||
|
break;
|
||
|
|
||
|
case FILE_FILE_COMPRESSION:
|
||
|
DisplayMsg( MSG_FILE_FILE_COMPRESSION );
|
||
|
break;
|
||
|
|
||
|
case FILE_VOLUME_QUOTAS:
|
||
|
DisplayMsg( MSG_FILE_VOLUME_QUOTAS );
|
||
|
break;
|
||
|
|
||
|
case FILE_SUPPORTS_SPARSE_FILES:
|
||
|
DisplayMsg( MSG_FILE_SUPPORTS_SPARSE_FILES );
|
||
|
break;
|
||
|
|
||
|
case FILE_SUPPORTS_REPARSE_POINTS:
|
||
|
DisplayMsg( MSG_FILE_SUPPORTS_REPARSE_POINTS );
|
||
|
break;
|
||
|
|
||
|
case FILE_SUPPORTS_REMOTE_STORAGE:
|
||
|
DisplayMsg( MSG_FILE_SUPPORTS_REMOTE_STORAGE );
|
||
|
break;
|
||
|
|
||
|
case FILE_VOLUME_IS_COMPRESSED:
|
||
|
DisplayMsg( MSG_FILE_VOLUME_IS_COMPRESSED );
|
||
|
break;
|
||
|
|
||
|
case FILE_SUPPORTS_OBJECT_IDS:
|
||
|
DisplayMsg( MSG_FILE_SUPPORTS_OBJECT_IDS );
|
||
|
break;
|
||
|
|
||
|
case FILE_SUPPORTS_ENCRYPTION:
|
||
|
DisplayMsg( MSG_FILE_SUPPORTS_ENCRYPTION );
|
||
|
break;
|
||
|
|
||
|
case FILE_NAMED_STREAMS:
|
||
|
DisplayMsg( MSG_FILE_NAMED_STREAMS );
|
||
|
break;
|
||
|
}
|
||
|
dwMask <<= 1;
|
||
|
}
|
||
|
|
||
|
} finally {
|
||
|
|
||
|
free( lpVolumeNameBuffer );
|
||
|
free( lpVolumeSerialNumber );
|
||
|
free( lpFileSystemFlags );
|
||
|
free( lpMaximumComponentLength );
|
||
|
free( lpFileSystemNameBuffer );
|
||
|
}
|
||
|
|
||
|
return ExitCode;
|
||
|
}
|
||
|
|
||
|
|
||
|
INT
|
||
|
GetNtfsVolumeData(
|
||
|
IN INT argc,
|
||
|
IN PWSTR argv[]
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine gets the NTFS volume data for the volume
|
||
|
specified.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
argc - The argument count.
|
||
|
argv - Array of Strings of the form :
|
||
|
' fscutl getntfsdv <volume pathname>'.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
HANDLE FileHandle = INVALID_HANDLE_VALUE;
|
||
|
WCHAR FileName[MAX_PATH];
|
||
|
BOOL Status;
|
||
|
BYTE Buffer[sizeof( NTFS_VOLUME_DATA_BUFFER ) + sizeof( NTFS_EXTENDED_VOLUME_DATA )];
|
||
|
PNTFS_VOLUME_DATA_BUFFER pvdb = (PNTFS_VOLUME_DATA_BUFFER)Buffer;
|
||
|
PNTFS_EXTENDED_VOLUME_DATA pevd = (PNTFS_EXTENDED_VOLUME_DATA)(pvdb + 1);
|
||
|
DWORD BytesReturned;
|
||
|
INT ExitCode = EXIT_CODE_SUCCESS;
|
||
|
|
||
|
try {
|
||
|
|
||
|
if (argc != 1) {
|
||
|
DisplayMsg( MSG_USAGE_NTFSINFO );
|
||
|
if (argc != 0) {
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
}
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
if (!IsVolumeLocalNTFS( argv[0][0] )) {
|
||
|
DisplayMsg( MSG_NTFS_REQUIRED );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
wcscpy( FileName, L"\\\\.\\" );
|
||
|
wcscat( FileName, argv[0] );
|
||
|
|
||
|
FileHandle = CreateFile(
|
||
|
FileName,
|
||
|
GENERIC_READ,
|
||
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
||
|
NULL,
|
||
|
OPEN_EXISTING,
|
||
|
FILE_ATTRIBUTE_NORMAL,
|
||
|
NULL
|
||
|
);
|
||
|
if (FileHandle == INVALID_HANDLE_VALUE) {
|
||
|
DisplayError();
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
Status = DeviceIoControl(
|
||
|
FileHandle,
|
||
|
FSCTL_GET_NTFS_VOLUME_DATA,
|
||
|
NULL,
|
||
|
0,
|
||
|
pvdb,
|
||
|
sizeof( Buffer ),
|
||
|
&BytesReturned,
|
||
|
(LPOVERLAPPED)NULL
|
||
|
);
|
||
|
if (!Status) {
|
||
|
DisplayError();
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
DisplayMsg(
|
||
|
MSG_NTFSINFO_STATS,
|
||
|
QuadToPaddedHexText( pvdb->VolumeSerialNumber.QuadPart ), // Serial number in hex
|
||
|
pevd->MajorVersion,
|
||
|
pevd->MinorVersion,
|
||
|
QuadToPaddedHexText( pvdb->NumberSectors.QuadPart ),
|
||
|
QuadToPaddedHexText( pvdb->TotalClusters.QuadPart ),
|
||
|
QuadToPaddedHexText( pvdb->FreeClusters.QuadPart ),
|
||
|
QuadToPaddedHexText( pvdb->TotalReserved.QuadPart ),
|
||
|
pvdb->BytesPerSector,
|
||
|
pvdb->BytesPerCluster,
|
||
|
pvdb->BytesPerFileRecordSegment,
|
||
|
pvdb->ClustersPerFileRecordSegment,
|
||
|
QuadToPaddedHexText( pvdb->MftValidDataLength.QuadPart ),
|
||
|
QuadToPaddedHexText( pvdb->MftStartLcn.QuadPart ),
|
||
|
QuadToPaddedHexText( pvdb->Mft2StartLcn.QuadPart ),
|
||
|
QuadToPaddedHexText( pvdb->MftZoneStart.QuadPart ),
|
||
|
QuadToPaddedHexText( pvdb->MftZoneEnd.QuadPart )
|
||
|
);
|
||
|
|
||
|
} finally {
|
||
|
|
||
|
if (FileHandle != INVALID_HANDLE_VALUE) {
|
||
|
CloseHandle( FileHandle );
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return ExitCode;
|
||
|
}
|
||
|
|
||
|
|
||
|
INT
|
||
|
GetFileSystemStatistics(
|
||
|
IN INT argc,
|
||
|
IN PWSTR argv[]
|
||
|
)
|
||
|
/*++
|
||
|
|
||
|
Routine Description:
|
||
|
|
||
|
This routine gets the file system statistics for the volume
|
||
|
specified.
|
||
|
|
||
|
Arguments:
|
||
|
|
||
|
argc - The argument count.
|
||
|
argv - Array of Strings of the form :
|
||
|
' fscutl getfss <volume pathname>'.
|
||
|
|
||
|
Return Value:
|
||
|
|
||
|
None
|
||
|
|
||
|
--*/
|
||
|
{
|
||
|
#define FS_STAT(_f) FsStat( FsStats, StrucSize, offsetof(FILESYSTEM_STATISTICS,_f), SysInfo.dwNumberOfProcessors )
|
||
|
#define FAT_STAT(_f) FsStat( FatFsStats, StrucSize, offsetof(FAT_FILE_SYSTEM_STATISTICS,_f), SysInfo.dwNumberOfProcessors )
|
||
|
#define NTFS_STAT(_f) FsStat( NtfsFsStats, StrucSize, offsetof(NTFS_FILE_SYSTEM_STATISTICS,_f), SysInfo.dwNumberOfProcessors )
|
||
|
|
||
|
BOOL Status;
|
||
|
HANDLE FileHandle = INVALID_HANDLE_VALUE;
|
||
|
WCHAR FileName[MAX_PATH];
|
||
|
PFILESYSTEM_STATISTICS FsStats = NULL;
|
||
|
DWORD OutBufferSize;
|
||
|
DWORD BytesReturned;
|
||
|
SYSTEM_INFO SysInfo;
|
||
|
PNTFS_FILE_SYSTEM_STATISTICS NtfsFsStats = NULL;
|
||
|
PFAT_FILE_SYSTEM_STATISTICS FatFsStats = NULL;
|
||
|
ULONG StrucSize;
|
||
|
INT ExitCode = EXIT_CODE_SUCCESS;
|
||
|
|
||
|
try {
|
||
|
|
||
|
if (argc != 1) {
|
||
|
DisplayMsg( MSG_USAGE_GETFSS );
|
||
|
if (argc != 0) {
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
}
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
if (!IsVolumeLocalNTFS( argv[0][0] )) {
|
||
|
DisplayMsg( MSG_NTFS_REQUIRED );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
wcscpy( FileName, L"\\\\.\\" );
|
||
|
wcsncat( FileName, argv[0], (sizeof(FileName)/sizeof(WCHAR))-wcslen(FileName) );
|
||
|
|
||
|
FileHandle = CreateFile(
|
||
|
FileName,
|
||
|
GENERIC_READ,
|
||
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
||
|
NULL,
|
||
|
OPEN_EXISTING,
|
||
|
FILE_ATTRIBUTE_NORMAL,
|
||
|
NULL
|
||
|
);
|
||
|
if (FileHandle == INVALID_HANDLE_VALUE) {
|
||
|
DisplayError();
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
GetSystemInfo( &SysInfo );
|
||
|
|
||
|
OutBufferSize = max(sizeof(NTFS_FILE_SYSTEM_STATISTICS),sizeof(FAT_FILE_SYSTEM_STATISTICS)) * SysInfo.dwNumberOfProcessors;
|
||
|
|
||
|
FsStats = (PFILESYSTEM_STATISTICS) malloc ( OutBufferSize );
|
||
|
if (FsStats == NULL) {
|
||
|
DisplayErrorMsg( ERROR_NOT_ENOUGH_MEMORY );
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
Status = DeviceIoControl(
|
||
|
FileHandle,
|
||
|
FSCTL_FILESYSTEM_GET_STATISTICS,
|
||
|
NULL,
|
||
|
0,
|
||
|
FsStats,
|
||
|
OutBufferSize,
|
||
|
&BytesReturned,
|
||
|
(LPOVERLAPPED)NULL
|
||
|
);
|
||
|
if (!Status) {
|
||
|
DisplayError();
|
||
|
ExitCode = EXIT_CODE_FAILURE;
|
||
|
leave;
|
||
|
}
|
||
|
|
||
|
switch (FsStats->FileSystemType) {
|
||
|
case FILESYSTEM_STATISTICS_TYPE_NTFS:
|
||
|
DisplayMsg( MSG_FSTYPE_NTFS );
|
||
|
NtfsFsStats = (PNTFS_FILE_SYSTEM_STATISTICS) FsStats;
|
||
|
StrucSize = sizeof(NTFS_FILE_SYSTEM_STATISTICS);
|
||
|
break;
|
||
|
|
||
|
case FILESYSTEM_STATISTICS_TYPE_FAT:
|
||
|
DisplayMsg( MSG_FSTYPE_FAT );
|
||
|
FatFsStats = (PFAT_FILE_SYSTEM_STATISTICS) FsStats;
|
||
|
StrucSize = sizeof(FAT_FILE_SYSTEM_STATISTICS);
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
DisplayMsg(
|
||
|
MSG_GENERAL_FSSTAT,
|
||
|
QuadToDecimalText( FS_STAT(UserFileReads)),
|
||
|
QuadToDecimalText( FS_STAT(UserFileReadBytes)),
|
||
|
QuadToDecimalText( FS_STAT(UserDiskReads)),
|
||
|
QuadToDecimalText( FS_STAT(UserFileWrites)),
|
||
|
QuadToDecimalText( FS_STAT(UserFileWriteBytes)),
|
||
|
QuadToDecimalText( FS_STAT(UserDiskWrites)),
|
||
|
QuadToDecimalText( FS_STAT(MetaDataReads)),
|
||
|
QuadToDecimalText( FS_STAT(MetaDataReadBytes)),
|
||
|
QuadToDecimalText( FS_STAT(MetaDataDiskReads)),
|
||
|
QuadToDecimalText( FS_STAT(MetaDataWrites)),
|
||
|
QuadToDecimalText( FS_STAT(MetaDataWriteBytes)),
|
||
|
QuadToDecimalText( FS_STAT(MetaDataDiskWrites))
|
||
|
);
|
||
|
|
||
|
//
|
||
|
// Print FileSystem specific data
|
||
|
//
|
||
|
|
||
|
switch (FsStats->FileSystemType) {
|
||
|
case FILESYSTEM_STATISTICS_TYPE_FAT:
|
||
|
DisplayMsg(
|
||
|
MSG_FAT_FSSTA,
|
||
|
QuadToDecimalText( FAT_STAT(Fat.CreateHits)),
|
||
|
QuadToDecimalText( FAT_STAT(Fat.SuccessfulCreates)),
|
||
|
QuadToDecimalText( FAT_STAT(Fat.FailedCreates)),
|
||
|
QuadToDecimalText( FAT_STAT(Fat.NonCachedReads)),
|
||
|
QuadToDecimalText( FAT_STAT(Fat.NonCachedReadBytes)),
|
||
|
QuadToDecimalText( FAT_STAT(Fat.NonCachedWrites)),
|
||
|
QuadToDecimalText( FAT_STAT(Fat.NonCachedWriteBytes)),
|
||
|
QuadToDecimalText( FAT_STAT(Fat.NonCachedDiskReads)),
|
||
|
QuadToDecimalText( FAT_STAT(Fat.NonCachedDiskWrites))
|
||
|
);
|
||
|
break;
|
||
|
|
||
|
case FILESYSTEM_STATISTICS_TYPE_NTFS:
|
||
|
DisplayMsg(
|
||
|
MSG_NTFS_FSSTA,
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.MftReads)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.MftReadBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.MftWrites)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.MftWriteBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.Mft2Writes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.Mft2WriteBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.RootIndexReads)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.RootIndexReadBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.RootIndexWrites)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.RootIndexWriteBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.BitmapReads)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.BitmapReadBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.BitmapWrites)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.BitmapWriteBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.MftBitmapReads)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.MftBitmapReadBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.MftBitmapWrites)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.MftBitmapWriteBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.UserIndexReads)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.UserIndexReadBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.UserIndexWrites)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.UserIndexWriteBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.LogFileReads)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.LogFileReadBytes)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.LogFileWrites)),
|
||
|
QuadToDecimalText( NTFS_STAT(Ntfs.LogFileWriteBytes))
|
||
|
);
|
||
|
|
||
|
//
|
||
|
// Still some more fields left
|
||
|
//
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
} finally {
|
||
|
|
||
|
if (FileHandle != INVALID_HANDLE_VALUE) {
|
||
|
CloseHandle( FileHandle );
|
||
|
}
|
||
|
if (FsStats) {
|
||
|
free( FsStats );
|
||
|
}
|
||
|
|
||
|
}
|
||
|
return ExitCode;
|
||
|
}
|
||
|
|