|
|
/*++
Copyright (c) 1989 Microsoft Corporation
Module Name:
tfat.c
Abstract:
Test program for the Fat File system
Author:
Gary Kimura [GaryKi] 24-May-1989
Revision History:
--*/
#include <stdio.h>
#include <string.h>
//#include <ctype.h>
#define toupper(C) ((C) >= 'a' && (C) <= 'z' ? (C) - ('a' - 'A') : (C))
#define isdigit(C) ((C) >= '0' && (C) <= '9')
#include <nt.h>
#include <ntrtl.h>
#ifndef SIMULATOR
ULONG IoInitIncludeDevices; #endif // SIMULATOR
#define simprintf(X,Y) {if (!Silent) {DbgPrint(X,Y);} }
BOOLEAN Silent;
//
// The buffer size must be a multiple of 512
//
#define BUFFERSIZE 1024
UCHAR Buffer[BUFFERSIZE];
HANDLE CurrentProcessHandle; CHAR Prefix[32];
ULONG WriteThrough = 0;
VOID PrintTime ( IN PLARGE_INTEGER Time );
VOID WaitForSingleObjectError( IN NTSTATUS Status );
VOID CreateFileError( IN NTSTATUS Status, PCHAR File );
VOID OpenFileError( IN NTSTATUS Status, PCHAR File );
VOID ReadFileError( IN NTSTATUS Status );
VOID WriteFileError( IN NTSTATUS Status );
VOID CheckIoStatus( IN PIO_STATUS_BLOCK IoStatus, IN ULONG Length, IN BOOLEAN Read );
VOID SetInformationFileError( IN NTSTATUS Status );
VOID QueryInformationFileError( IN NTSTATUS Status );
VOID SetVolumeInformationFileError( IN NTSTATUS Status );
VOID QueryVolumeInformationFileError( IN NTSTATUS Status );
VOID CloseError( IN NTSTATUS Status );
VOID IoStatusError( IN NTSTATUS Status );
VOID main( int argc, char *argv[], char *envp[] ) { NTSTATUS Status; ULONG i; ULONG Count; VOID FatMain(); CHAR Device[8];
CurrentProcessHandle = NtCurrentProcess(); Status = STATUS_SUCCESS;
// DbgPrint( "Entering User Mode Test Program\n" );
// DbgPrint( "argc: %ld\n", argc );
// if (argv != NULL) {
// for (i=0; i<argc; i++) {
// DbgPrint( "argv[ %ld ]: %s\n", i, argv[ i ] );
// }
// }
// if (envp != NULL) {
// i = 0;
// while (*envp) {
// DbgPrint( "envp[ %02ld ]: %s\n", i++, *envp++ );
// }
// }
if (argc > 1) { Count = 0; for (i = 0; isdigit(argv[1][i]); i += 1) { Count = Count * 10 + argv[1][i] - '0'; } } else { Count = 1; }
if (argc > 2) { strcpy( Device, argv[2]); } else { strcpy( Device, "\\A:\\"); }
if (argc > 3) { WriteThrough = FILE_WRITE_THROUGH; }
if (toupper(Device[1]) != Device[1]) { Device[1] = (CHAR)toupper(Device[1]); Silent = TRUE; } else { Silent = FALSE; }
FatMain(Count, Device);
// DbgPrint( "Exiting User Mode Test Program with Status = %X\n", Status );
NtTerminateProcess( CurrentProcessHandle, Status ); }
VOID FatMain( IN ULONG LoopCount, IN CHAR Device[] ) { VOID Upcase(); VOID Append(),Chmode(),Copy(),Create(),DebugLevel(),Delete(); VOID Directory(),Mkdir(),Query(),QVolume(),Rename(); VOID SVolume(),Type(),Quit();
CHAR Str[64]; CHAR Str2[64]; CHAR LoopStr[64]; ULONG i; LARGE_INTEGER Time;
DbgPrint("FatMain %d\n", LoopCount);
NtQuerySystemTime(&Time); strcpy( Prefix, Device); // "\\A:\\";
Prefix[12] = 0; RtlIntegerToChar((ULONG)NtCurrentTeb()->ClientId.UniqueProcess, 16, -8, &Prefix[4]);
Mkdir( Prefix ); Directory( Device ); Directory( Prefix );
for (i = 0; i < LoopCount; i += 1) { NtQuerySystemTime(&Time); strcpy(LoopStr, "Start loop xxxxxxxx "); RtlIntegerToChar(i, 16, -8, &LoopStr[11]); strcat( LoopStr, Prefix ); DbgPrint(LoopStr); DbgPrint("\n");
//
// Create new log
//
strcpy( Str2, "\\" ); RtlIntegerToChar(i, 16, -8, &Str2[1]); Str2[9] = 0; strcpy( Str, Prefix ); Create( strcat( Str, Str2 ), i);
//
// Delete last log file if i > 0
//
if (i > 0) { strcpy( Str2, "\\" ); RtlIntegerToChar(i-1, 16, -8, &Str2[1]); Str2[9] = 0; strcpy( Str, Prefix ); Delete( strcat( Str, Str2 ) ); }
//
// Create some new files
//
strcpy( Str, Prefix ); Create( strcat( Str, "\\0.tmp" ), 0 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\1.tmp" ), 1 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\511.tmp" ), 511 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\512.tmp" ), 512 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\513.tmp" ), 513 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\1023.tmp" ), 1023 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\1024.tmp" ), 1024 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\1025.tmp" ), 1025 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\4095.tmp" ), 4095 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\4096.tmp" ), 4096 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\4097.tmp" ), 4097 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\8191.tmp" ), 8191 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\8192.tmp" ), 8192 ); strcpy( Str, Prefix ); Create( strcat( Str, "\\8193.tmp" ), 8193 );
Directory( Device ); Directory( Prefix );
strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\0.tmp" ), strcat( Str2, "\\0.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\1.tmp" ), strcat( Str2, "\\1.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\511.tmp" ), strcat( Str2, "\\511.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\512.tmp" ), strcat( Str2, "\\512.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\513.tmp" ), strcat( Str2, "\\513.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\1023.tmp" ), strcat( Str2, "\\1023.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\1024.tmp" ), strcat( Str2, "\\1024.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\1025.tmp" ), strcat( Str2, "\\1025.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\4095.tmp" ), strcat( Str2, "\\4095.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\4096.tmp" ), strcat( Str2, "\\4096.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\4097.tmp" ), strcat( Str2, "\\4097.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\8191.tmp" ), strcat( Str2, "\\8191.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\8192.tmp" ), strcat( Str2, "\\8192.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Copy( strcat( Str, "\\8193.tmp" ), strcat( Str2, "\\8193.tmq" ) );
Directory( Device ); Directory( Prefix );
strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\0.tmp" ), strcat( Str2, "\\0.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\1.tmp" ), strcat( Str2, "\\1.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\511.tmp" ), strcat( Str2, "\\511.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\512.tmp" ), strcat( Str2, "\\512.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\513.tmp" ), strcat( Str2, "\\513.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\1023.tmp" ), strcat( Str2, "\\1023.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\1024.tmp" ), strcat( Str2, "\\1024.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\1025.tmp" ), strcat( Str2, "\\1025.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\4095.tmp" ), strcat( Str2, "\\4095.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\4096.tmp" ), strcat( Str2, "\\4096.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\4097.tmp" ), strcat( Str2, "\\4097.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\8191.tmp" ), strcat( Str2, "\\8191.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\8192.tmp" ), strcat( Str2, "\\8192.tmq" ) ); strcpy( Str, Prefix ); strcpy( Str2, Prefix ); Append( strcat( Str, "\\8193.tmp" ), strcat( Str2, "\\8193.tmq" ) );
Directory( Device ); Directory( Prefix );
strcpy( Str, Prefix ); Delete( strcat( Str, "\\0.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\1.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\511.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\512.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\513.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\1023.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\1024.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\1025.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\4095.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\4096.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\4097.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\8191.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\8192.tmp" ) ); strcpy( Str, Prefix ); Delete( strcat( Str, "\\8193.tmp" ) );
Directory( Device ); Directory( Prefix );
}
DbgPrint( "Done\n" ); NtQuerySystemTime(&Time);
Quit();
return; // TRUE;
}
VOID Upcase ( IN OUT PUCHAR String ) { while (*String != '\0') { *String = (UCHAR)toupper(*String); String += 1; } }
VOID Append( IN PCHAR FromName, IN PCHAR ToName ) { NTSTATUS Status;
HANDLE FromFileHandle; HANDLE ToFileHandle;
OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus;
LARGE_INTEGER ByteOffset; LARGE_INTEGER EofOffset; ULONG LogLsn;
simprintf("Append ", 0); simprintf(FromName, 0); simprintf(" ", 0); simprintf(ToName, 0); simprintf("\n", 0);
//
// Open the From file for read access
//
RtlInitString( &NameString, FromName ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &FromFileHandle, FILE_READ_DATA | SYNCHRONIZE, &ObjectAttributes, &IoStatus, 0L, WriteThrough ))) { OpenFileError( Status , FromName ); return; }
//
// Open the To file for write access
//
RtlInitString( &NameString, ToName ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &ToFileHandle, FILE_WRITE_DATA | SYNCHRONIZE, &ObjectAttributes, &IoStatus, 0L, WriteThrough ))) { OpenFileError( Status , ToName ); return; }
//
// Now append the files
//
ByteOffset = LiFromLong( 0 ); EofOffset = LiFromLong( FILE_WRITE_TO_END_OF_FILE );
for (LogLsn = 0; TRUE; LogLsn += BUFFERSIZE/512) {
ULONG ReadLength;
//
// Read the next logical sectors in
//
ByteOffset.LowPart = LogLsn * 512;
if (!NT_SUCCESS(Status = NtReadFile( FromFileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, BUFFERSIZE, &ByteOffset, (PULONG) NULL ))) { if (Status == STATUS_END_OF_FILE) { break; } ReadFileError( Status ); break; } if (!NT_SUCCESS( Status = NtWaitForSingleObject( FromFileHandle, TRUE, NULL))) { // NtPartyByNumber(50);
WaitForSingleObjectError( Status ); return; }
//
// Check how the read turned out
//
if (IoStatus.Status == STATUS_END_OF_FILE) { break; } CheckIoStatus( &IoStatus, BUFFERSIZE, TRUE ); if (!NT_SUCCESS(IoStatus.Status)) { break; }
//
// Append the sectors to the To file
//
ReadLength = IoStatus.Information; if (!NT_SUCCESS(Status = NtWriteFile( ToFileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, IoStatus.Information, &EofOffset, (PULONG) NULL ))) { WriteFileError( Status ); return; } if (!NT_SUCCESS(Status = NtWaitForSingleObject( ToFileHandle, TRUE, NULL))) { // NtPartyByNumber(50);
WaitForSingleObjectError( Status ); return; }
//
// Check how the write turned out
//
CheckIoStatus( &IoStatus, ReadLength, FALSE ); if (!NT_SUCCESS(IoStatus.Status)) { IoStatusError( IoStatus.Status ); break; }
//
// If we didn't read or write a full buffer then the copy is done
//
if (IoStatus.Information < BUFFERSIZE) { break; }
}
if (!NT_SUCCESS(IoStatus.Status) && (IoStatus.Status != STATUS_END_OF_FILE)) {
IoStatusError( IoStatus.Status );
}
//
// Close both files
//
if (!NT_SUCCESS(Status = NtClose( FromFileHandle ))) { CloseError( Status ); }
if (!NT_SUCCESS(Status = NtClose( ToFileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Chmode( IN PCHAR Attrib, IN PCHAR String ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus;
ULONG FileAttribute;
//
// Get the attribute
//
Upcase( (PUCHAR)Attrib );
//
// Get the filename
//
simprintf("Chmode", 0); simprintf(String, 0); simprintf(" ", 0); simprintf(Attrib, 0); simprintf("\n", 0);
//
// Decode the attributes
//
FileAttribute = 0; if (strpbrk(Attrib,"N") != NULL) {FileAttribute |= FILE_ATTRIBUTE_NORMAL;} if (strpbrk(Attrib,"R") != NULL) {FileAttribute |= FILE_ATTRIBUTE_READONLY;} if (strpbrk(Attrib,"H") != NULL) {FileAttribute |= FILE_ATTRIBUTE_HIDDEN;} if (strpbrk(Attrib,"S") != NULL) {FileAttribute |= FILE_ATTRIBUTE_SYSTEM;} if (strpbrk(Attrib,"A") != NULL) {FileAttribute |= FILE_ATTRIBUTE_ARCHIVE;}
//
// Open the file for write attributes access
//
RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &FileHandle, FILE_WRITE_ATTRIBUTES | SYNCHRONIZE, &ObjectAttributes, &IoStatus, 0L, WriteThrough ))) { OpenFileError( Status , String ); return; }
//
// Change the file attributes
//
((PFILE_BASIC_INFORMATION)&Buffer[0])->CreationTime.HighPart = 0; ((PFILE_BASIC_INFORMATION)&Buffer[0])->CreationTime.LowPart = 0;
((PFILE_BASIC_INFORMATION)&Buffer[0])->LastAccessTime.HighPart = 0; ((PFILE_BASIC_INFORMATION)&Buffer[0])->LastAccessTime.LowPart = 0;
((PFILE_BASIC_INFORMATION)&Buffer[0])->LastWriteTime.HighPart = 0; ((PFILE_BASIC_INFORMATION)&Buffer[0])->LastWriteTime.LowPart = 0;
((PFILE_BASIC_INFORMATION)&Buffer[0])->FileAttributes = FileAttribute;
if (!NT_SUCCESS(Status = NtSetInformationFile( FileHandle, &IoStatus, Buffer, sizeof(FILE_BASIC_INFORMATION), FileBasicInformation))) { SetInformationFileError( Status ); }
//
// Now close the file
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Copy( IN PCHAR FromName, IN PCHAR ToName ) { NTSTATUS Status;
HANDLE FromFileHandle; HANDLE ToFileHandle;
OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus;
LARGE_INTEGER FromFileAllocation;
LARGE_INTEGER ByteOffset; ULONG LogLsn;
//
// Get both file names
//
simprintf("Copy ", 0); simprintf(FromName, 0); simprintf(" ", 0); simprintf(ToName, 0); simprintf("\n", 0);
//
// Open the From file for read access
//
RtlInitString( &NameString, FromName ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &FromFileHandle, FILE_READ_DATA | FILE_READ_ATTRIBUTES | SYNCHRONIZE, &ObjectAttributes, &IoStatus, 0L, WriteThrough ))) { OpenFileError( Status , FromName ); return; }
//
// Get the size of the from file
//
if (!NT_SUCCESS(Status = NtQueryInformationFile( FromFileHandle, &IoStatus, Buffer, BUFFERSIZE, FileStandardInformation))) { QueryInformationFileError( Status ); return; } FromFileAllocation = ((PFILE_STANDARD_INFORMATION)Buffer)->AllocationSize;
//
// Create the To file
//
RtlInitString( &NameString, ToName ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtCreateFile( &ToFileHandle, FILE_WRITE_DATA | SYNCHRONIZE, &ObjectAttributes, &IoStatus, &FromFileAllocation, FILE_ATTRIBUTE_NORMAL, 0L, FILE_SUPERSEDE, WriteThrough, (PVOID)NULL, 0L ))) { CreateFileError( Status , ToName ); return; }
//
// Now copy the files
//
ByteOffset = LiFromLong( 0 );
for (LogLsn = 0; TRUE; LogLsn += BUFFERSIZE/512) {
ULONG ReadLength;
//
// Read the next logical sectors in
//
ByteOffset.LowPart = LogLsn * 512;
if (!NT_SUCCESS(Status = NtReadFile( FromFileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, BUFFERSIZE, &ByteOffset, (PULONG) NULL ))) { if (Status == STATUS_END_OF_FILE) { break; } ReadFileError( Status ); break; } if (!NT_SUCCESS(Status = NtWaitForSingleObject( FromFileHandle, TRUE, NULL))) { // NtPartyByNumber(50);
WaitForSingleObjectError( Status ); return; }
//
// Check how the read turned out
//
if (IoStatus.Status == STATUS_END_OF_FILE) { break; } CheckIoStatus( &IoStatus, BUFFERSIZE, TRUE ); if (!NT_SUCCESS(IoStatus.Status)) { break; }
//
// Write the sectors out
//
ReadLength = IoStatus.Information; if (!NT_SUCCESS(Status = NtWriteFile( ToFileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, IoStatus.Information, &ByteOffset, (PULONG) NULL ))) { WriteFileError( Status ); return; } if (!NT_SUCCESS(Status = NtWaitForSingleObject( ToFileHandle, TRUE, NULL))) { // NtPartyByNumber(50);
WaitForSingleObjectError( Status ); return; }
//
// Check how the write turned out
//
CheckIoStatus( &IoStatus, ReadLength, FALSE ); if (!NT_SUCCESS(IoStatus.Status)) { IoStatusError( IoStatus.Status ); break; }
//
// If we didn't read or write a full buffer then the copy is done
//
if (IoStatus.Information < BUFFERSIZE) { break; }
}
if (!NT_SUCCESS(IoStatus.Status) && (IoStatus.Status != STATUS_END_OF_FILE)) {
IoStatusError( IoStatus.Status );
}
//
// Close both files
//
if (!NT_SUCCESS(Status = NtClose( FromFileHandle ))) { CloseError( Status ); }
if (!NT_SUCCESS(Status = NtClose( ToFileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Create( IN PCHAR String, IN ULONG Size ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus; LARGE_INTEGER AllocationSize;
LARGE_INTEGER ByteOffset; ULONG BufferLength; ULONG i;
static CHAR FoxString[] = "The quick brown fox jumped over the lazy dog.\r\n"; ULONG FoxLength;
//
// Get the filename
//
simprintf("Create ", 0); simprintf(String, 0); simprintf("\n", 0);
//
// Create the new file
//
AllocationSize = LiFromUlong( Size ); RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtCreateFile( &FileHandle, FILE_WRITE_DATA | SYNCHRONIZE, &ObjectAttributes, &IoStatus, &AllocationSize, FILE_ATTRIBUTE_NORMAL, 0L, FILE_SUPERSEDE, WriteThrough, (PVOID)NULL, 0L ))) { CreateFileError( Status , String ); return; }
//
// create a test buffer to output
//
FoxLength = strlen(FoxString); for (i = FoxLength; i < BUFFERSIZE; i += FoxLength) { strcpy((PCHAR)&Buffer[i-FoxLength], FoxString); } BufferLength = i - FoxLength;
//
// The main loop writes out the test buffer a BufferLength amount
// at a time
//
ByteOffset = LiFromLong( 0 );
for (i = BufferLength; i < Size; i += BufferLength) {
//
// Write the next buffer
//
ByteOffset.LowPart = i-BufferLength;
if (!NT_SUCCESS(Status = NtWriteFile( FileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, BufferLength, &ByteOffset, (PULONG) NULL ))) { WriteFileError( Status ); return; }
if (!NT_SUCCESS(Status = NtWaitForSingleObject( FileHandle, TRUE, NULL))) { // NtPartyByNumber(50);
WaitForSingleObjectError( Status ); return; }
//
// check how the write turned out
//
CheckIoStatus( &IoStatus, BufferLength, FALSE ); if (!NT_SUCCESS(IoStatus.Status)) { IoStatusError( IoStatus.Status ); break; }
}
//
// Check for a residual to output
//
if (i - BufferLength < Size) {
//
// Write out the residual buffer
//
ByteOffset.LowPart = i-BufferLength;
if (!NT_SUCCESS(Status = NtWriteFile( FileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, Size - (i - BufferLength), &ByteOffset, (PULONG) NULL ))) { WriteFileError( Status ); return; }
if (!NT_SUCCESS(Status = NtWaitForSingleObject( FileHandle, TRUE, NULL))) { // NtPartyByNumber(50);
WaitForSingleObjectError( Status ); return; }
//
// check how the write turned out
//
CheckIoStatus( &IoStatus, Size - (i - BufferLength), FALSE ); if (!NT_SUCCESS(IoStatus.Status)) { IoStatusError( IoStatus.Status ); }
}
//
// Now close the file
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID DebugLevel() {
#ifdef FATDBG
//simprintf("Debug Trace Level %x\n", FatDebugTraceLevel);
#else
//simprintf("System not compiled for debug tracing\n", 0);
#endif // FATDBG
return; }
VOID Delete( IN PCHAR String ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus;
//
// Get the filename
//
simprintf("Delete ", 0); simprintf(String, 0); simprintf("\n", 0);
//
// Open the file for delete access
//
RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtCreateFile( &FileHandle, DELETE | SYNCHRONIZE, &ObjectAttributes, &IoStatus, (PLARGE_INTEGER)NULL, 0L, 0L, FILE_OPEN, WriteThrough, (PVOID)NULL, 0L ))) { CreateFileError( Status , String); return; }
//
// Mark the file for delete
//
((PFILE_DISPOSITION_INFORMATION)&Buffer[0])->DeleteFile = TRUE;
if (!NT_SUCCESS(Status = NtSetInformationFile( FileHandle, &IoStatus, Buffer, sizeof(FILE_DISPOSITION_INFORMATION), FileDispositionInformation))) { SetInformationFileError( Status ); return; }
//
// Now close the file
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Directory( IN PCHAR String ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus;
NTSTATUS NtStatus;
PFILE_ADIRECTORY_INFORMATION FileInfo; ULONG i;
//
// Get the filename
//
simprintf("Directory ", 0); simprintf(String, 0); simprintf("\n", 0);
//
// Open the file for list directory access
//
RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &FileHandle, FILE_LIST_DIRECTORY | SYNCHRONIZE, &ObjectAttributes, &IoStatus, FILE_SHARE_READ, WriteThrough | FILE_DIRECTORY_FILE ))) { OpenFileError( Status , String ); return; }
//
// zero out the buffer so next time we'll recognize the end of data
//
for (i = 0; i < BUFFERSIZE; i += 1) { Buffer[i] = 0; }
//
// Do the directory loop
//
for (NtStatus = NtQueryDirectoryFile( FileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, BUFFERSIZE, FileADirectoryInformation, FALSE, (PSTRING)NULL, TRUE); NT_SUCCESS(NtStatus); NtStatus = NtQueryDirectoryFile( FileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, BUFFERSIZE, FileADirectoryInformation, FALSE, (PSTRING)NULL, FALSE) ) {
if (!NT_SUCCESS(Status = NtWaitForSingleObject(FileHandle, TRUE, NULL))) { // NtPartyByNumber(50);
WaitForSingleObjectError( Status ); return; }
//
// Check the Irp for success
//
if (!NT_SUCCESS(IoStatus.Status)) {
break;
}
//
// For every record in the buffer type out the directory information
//
//
// Point to the first record in the buffer, we are guaranteed to have
// one otherwise IoStatus would have been No More Files
//
FileInfo = (PFILE_ADIRECTORY_INFORMATION)&Buffer[0];
while (TRUE) {
//
// Print out information about the file
//
simprintf("%8lx ", FileInfo->FileAttributes); simprintf("%8lx/", FileInfo->EndOfFile.LowPart); simprintf("%8lx ", FileInfo->AllocationSize.LowPart);
{ CHAR Saved; Saved = FileInfo->FileName[FileInfo->FileNameLength]; FileInfo->FileName[FileInfo->FileNameLength] = 0; simprintf(FileInfo->FileName, 0); FileInfo->FileName[FileInfo->FileNameLength] = Saved; }
simprintf("\n", 0);
//
// Check if there is another record, if there isn't then we
// simply get out of this loop
//
if (FileInfo->NextEntryOffset == 0) { break; }
//
// There is another record so advance FileInfo to the next
// record
//
FileInfo = (PFILE_ADIRECTORY_INFORMATION)(((PUCHAR)FileInfo) + FileInfo->NextEntryOffset);
}
//
// zero out the buffer so next time we'll recognize the end of data
//
for (i = 0; i < BUFFERSIZE; i += 1) { Buffer[i] = 0; }
}
//
// Now close the file
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Mkdir( IN PCHAR String ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus; LARGE_INTEGER AllocationSize;
//
// Get the filename
//
simprintf("Mkdir ", 0); simprintf(String, 0); simprintf("\n", 0);
//
// Create the new directory
//
AllocationSize = LiFromLong( 4 ); RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtCreateFile( &FileHandle, SYNCHRONIZE, &ObjectAttributes, &IoStatus, &AllocationSize, 0L, 0L, FILE_CREATE, WriteThrough | FILE_DIRECTORY_FILE, (PVOID)NULL, 0L ))) { CreateFileError( Status , String ); return; }
//
// Now close the directory
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Query( IN PCHAR String ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus; ULONG i;
PFILE_AALL_INFORMATION AllInfo; PFILE_BASIC_INFORMATION BasicInfo; PFILE_STANDARD_INFORMATION StandardInfo; PFILE_INTERNAL_INFORMATION InternalInfo; PFILE_EA_INFORMATION EaInfo; PFILE_ACCESS_INFORMATION AccessInfo; PFILE_ANAME_INFORMATION NameInfo;
//
// zero out the buffer so next time we'll recognize the end of data
//
for (i = 0; i < BUFFERSIZE; i += 1) { Buffer[i] = 0; }
//
// Set up some local pointers
//
AllInfo = (PFILE_AALL_INFORMATION)Buffer; BasicInfo = &AllInfo->BasicInformation; StandardInfo = &AllInfo->StandardInformation; InternalInfo = &AllInfo->InternalInformation; EaInfo = &AllInfo->EaInformation; AccessInfo = &AllInfo->AccessInformation; NameInfo = &AllInfo->NameInformation;
//
// Get the filename
//
simprintf("Query ", 0); simprintf(String, 0); simprintf("\n", 0);
//
// Open the file for read attributes, read ea, and read control access
//
RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &FileHandle, FILE_READ_ATTRIBUTES | FILE_READ_EA | READ_CONTROL | SYNCHRONIZE, &ObjectAttributes, &IoStatus, 0L, WriteThrough ))) { OpenFileError( Status , String ); return; }
//
// Query the file
//
if (!NT_SUCCESS(Status = NtQueryInformationFile( FileHandle, &IoStatus, Buffer, BUFFERSIZE, FileAAllInformation))) { QueryInformationFileError( Status ); return; }
//
// Output file name information
//
simprintf("\"", 0); simprintf(NameInfo->FileName, 0); simprintf("\"\n", 0);
//
// Output the times
//
simprintf(" Create = ", 0); PrintTime( &BasicInfo->CreationTime ); simprintf("\n", 0); simprintf(" Access = ", 0); PrintTime( &BasicInfo->LastAccessTime ); simprintf("\n", 0); simprintf(" Write = ", 0); PrintTime( &BasicInfo->LastWriteTime ); simprintf("\n", 0);
//
// Output File size, and allocation size
//
simprintf(" Size = %8lx\n", StandardInfo->EndOfFile.LowPart); simprintf(" Alloc = %8lx\n", StandardInfo->AllocationSize.LowPart);
//
// Output File attributes, Device type, link count, and flags
//
simprintf(" Attrib = %8lx\n", BasicInfo->FileAttributes); // simprintf(" DevType = %8lx\n", StandardInfo->DeviceType);
simprintf(" Links = %8lx\n", StandardInfo->NumberOfLinks); simprintf(" Dir = %8lx\n", StandardInfo->Directory); simprintf(" Delete = %8lx\n", StandardInfo->DeletePending);
//
// Output the index number and ea size
//
simprintf(" Index = %8lx\n", InternalInfo->IndexNumber.LowPart); simprintf(" EaSize = %8lx\n", EaInfo->EaSize);
//
// Output the file access flags
//
simprintf(" Flags = %8lx\n", AccessInfo->AccessFlags);
//
// Now close the file
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID QVolume( IN PCHAR String ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus; ULONG i;
PFILE_FS_AVOLUME_INFORMATION VolumeInfo;
//
// zero out the buffer so next time we'll recognize the end of data
//
for (i = 0; i < BUFFERSIZE; i += 1) { Buffer[i] = 0; }
//
// Set up some local pointers
//
VolumeInfo = (PFILE_FS_AVOLUME_INFORMATION)Buffer;
//
// Get the volume name
//
simprintf("QVolume ", 0); simprintf(String, 0); simprintf("\n", 0);
//
// Open the Volume for no access
//
RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &FileHandle, FILE_READ_ATTRIBUTES | SYNCHRONIZE, &ObjectAttributes, &IoStatus, 0L, WriteThrough ))) { OpenFileError( Status , String ); return; }
//
// zero out the buffer so next time we'll recognize the end of data
//
for (i = 0; i < BUFFERSIZE; i += 1) { Buffer[i] = 0; }
//
// Query the volume info
//
if (!NT_SUCCESS(Status = NtQueryVolumeInformationFile( FileHandle, &IoStatus, Buffer, BUFFERSIZE, FileAFsVolumeInformation))) { QueryVolumeInformationFileError( Status ); return; }
//
// Output Volume name information
//
simprintf("\"", 0); simprintf(VolumeInfo->VolumeLabel, 0); simprintf("\"\n", 0);
//
// Output the volume serial number
//
simprintf(" SerialNum = %8lx\n", VolumeInfo->VolumeSerialNumber);
//
// Now close the Volume
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Rename() { //simprintf("Rename not implemented\n", 0);
}
VOID SVolume( IN PCHAR String, IN PCHAR Label ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus; ULONG i;
PFILE_FS_ALABEL_INFORMATION LabelInfo;
//
// zero out the buffer so next time we'll recognize the end of data
//
for (i = 0; i < BUFFERSIZE; i += 1) { Buffer[i] = 0; }
//
// Set up some local pointers
//
LabelInfo = (PFILE_FS_ALABEL_INFORMATION)Buffer;
//
// Get the volume name, and new label name
//
strcpy( LabelInfo->VolumeLabel, Label );
LabelInfo->VolumeLabelLength = strlen(LabelInfo->VolumeLabel);
if ((LabelInfo->VolumeLabelLength == 1) && (LabelInfo->VolumeLabel[0] == '.')) {
LabelInfo->VolumeLabelLength = 0;
}
simprintf("SVolume ", 0); simprintf(String, 0); simprintf(" ", 0); simprintf(LabelInfo->VolumeLabel, 0); simprintf("\n", 0);
//
// Open the Volume for no access
//
RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &FileHandle, FILE_WRITE_ATTRIBUTES | SYNCHRONIZE, &ObjectAttributes, &IoStatus, 0L, WriteThrough ))) { OpenFileError( Status, String ); return; }
//
// Set the volume info
//
if (!NT_SUCCESS(Status = NtSetVolumeInformationFile( FileHandle, &IoStatus, LabelInfo, BUFFERSIZE, FileAFsLabelInformation))) { SetVolumeInformationFileError( Status ); return; }
//
// Now close the Volume
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Type( IN PCHAR String ) { NTSTATUS Status;
HANDLE FileHandle; OBJECT_ATTRIBUTES ObjectAttributes; STRING NameString; IO_STATUS_BLOCK IoStatus;
LARGE_INTEGER ByteOffset; ULONG LogLsn; ULONG i;
//
// Get the filename
//
simprintf("Type ", 0); simprintf(String, 0); simprintf("\n", 0);
//
// Open the file for read
//
RtlInitString( &NameString, String ); InitializeObjectAttributes( &ObjectAttributes, &NameString, 0, NULL, NULL ); if (!NT_SUCCESS(Status = NtOpenFile( &FileHandle, FILE_READ_DATA | SYNCHRONIZE, &ObjectAttributes, &IoStatus, 0L, WriteThrough ))) { OpenFileError( Status, String ); return; }
//
// While there is data to be read we'll read a buffer and write it out
//
ByteOffset = LiFromLong( 0 );
for (LogLsn = 0; TRUE; LogLsn += BUFFERSIZE/512) {
//
// Read the next logical sector
//
ByteOffset.LowPart = LogLsn * 512;
if (!NT_SUCCESS(Status = NtReadFile( FileHandle, (HANDLE)NULL, (PIO_APC_ROUTINE)NULL, (PVOID)NULL, &IoStatus, Buffer, BUFFERSIZE, &ByteOffset, (PULONG) NULL ))) { if (Status == STATUS_END_OF_FILE) { break; } ReadFileError( Status ); break; }
if (!NT_SUCCESS(Status = NtWaitForSingleObject( FileHandle, TRUE, NULL))) { // NtPartyByNumber(50);
WaitForSingleObjectError( Status ); return; }
//
// check how the read turned out
//
if (IoStatus.Status == STATUS_END_OF_FILE) { break; } CheckIoStatus( &IoStatus, BUFFERSIZE, TRUE ); if (!NT_SUCCESS(IoStatus.Status)) { IoStatusError( IoStatus.Status ); break; }
//
// Write out the buffer
//
for (i = 0; i < IoStatus.Information; i += 1) { simprintf("%c", Buffer[i]); }
//
// If we didn't read in a complete buffer then we're all done reading
// and can get out of here
//
if (IoStatus.Information < BUFFERSIZE) { break; }
}
//
// Now close the file
//
if (!NT_SUCCESS(Status = NtClose( FileHandle ))) { CloseError( Status ); }
//
// And return to our caller
//
return;
}
VOID Quit() { simprintf("FatTest Exiting.\n", 0); return; }
VOID PrintTime ( IN PLARGE_INTEGER Time ) { TIME_FIELDS TimeFields;
static PCHAR Months[] = {"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
static PCHAR Days[] = {"Sun", "Mon", "Tue", "Wed", "Thu", "Fri", "Sat"};
RtlTimeToTimeFields( Time, &TimeFields );
simprintf(" %4d-", TimeFields.Year); simprintf(Months[TimeFields.Month-1], 0); simprintf("-%2d", TimeFields.Day);
simprintf(" %2d", TimeFields.Hour); simprintf(":%2d", TimeFields.Minute); simprintf(":%2d", TimeFields.Second); simprintf(".%3d (", TimeFields.Milliseconds);
simprintf(Days[TimeFields.Weekday], 0); simprintf(")", 0);
return; }
VOID WaitForSingleObjectError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" WaitForSingleObject Error %X\n", Status); }
VOID CreateFileError( IN NTSTATUS Status, PCHAR File ) { DbgPrint(Prefix); DbgPrint(" CreateFile of %s Error %X\n", File, Status); }
VOID OpenFileError( IN NTSTATUS Status, PCHAR File ) { DbgPrint(Prefix); DbgPrint(" OpenFile of %s Error %X\n", File, Status); }
VOID ReadFileError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" ReadFile Error %X\n", Status); }
VOID WriteFileError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" WriteFile Error %X\n", Status); }
VOID CheckIoStatus( IN PIO_STATUS_BLOCK IoStatus, IN ULONG Length, IN BOOLEAN Read ) { if (!NT_SUCCESS(IoStatus->Status)) { DbgPrint(" IoStatus->Status Error %08lx\n", IoStatus->Status); } if ((!Read && (IoStatus->Information != Length))
||
(Read && (IoStatus->Information > Length))) {
DbgPrint(" IoStatus->Information Error %08lx\n", IoStatus->Information); } }
VOID SetInformationFileError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" SetInfoFile Error %X\n", Status); }
VOID QueryInformationFileError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" QueryInfoFile Error %X\n", Status); }
VOID SetVolumeInformationFileError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" SetVolumeInfoFile Error %X\n", Status); }
VOID QueryVolumeInformationFileError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" QueryVolumeInfoFile Error %X\n", Status); }
VOID CloseError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" Close Error %X\n", Status); }
VOID IoStatusError( IN NTSTATUS Status ) { DbgPrint(Prefix); DbgPrint(" IoStatus Error %X\n", Status); }
|