|
|
//----------------------------------------------------------------------------
//
// Copyright (c) 1999 Microsoft Corporation
// All rights reserved.
//
// File Name:
// makebt32.c
//
// Description:
// This program copies the images of the 4 Windows NT setup disks to
// floppy disk so the user can boot their system with them.
//
// All characters and strings are wide (UNICODE). This file needs to be
// compiled with UNICODE and _UNICODE defined.
//
// Assumptions:
// This program will only run on NT 3.51 or later. This is a result of
// the CreateFile function call. It is not available on DOS, Windows 3.1
// or Windows 9x.
//
// The floppy disk images are in the current dir and named CDBOOT1.IMG,
// CDBOOT2.IMG, CDBOOT3.IMG and CDBOOT4.IMG.
//
// Please note that there are a lot of places where I call exit() without
// freeing memory for strings I have allocated. This version of the
// program only runs on NT so when the process exits it frees all its
// memory so it is not a concern that I may not call free() on some memory.
//
//----------------------------------------------------------------------------
#include <windows.h>
#include <setupapi.h>
#include <winioctl.h>
#include <conio.h>
#include <stdio.h>
#include <stdlib.h>
#include "resource.h"
//
// Constants
//
#define MAKEBOOT_MAX_STRING_LEN 1024
#define BYTES_PER_SECTOR 512
#define SECTORS_PER_TRACK 18
#define TRACK_SIZE SECTORS_PER_TRACK * BYTES_PER_SECTOR
#define TRACKS_ON_DISK 80 * 2 // * 2 because it is double-sided
#define MAX_DISK_LABEL_LENGTH 256
#define MAX_INILINE_LENGTH 1023
#define ENTER_KEY 13
#define ESC_KEY 27
#define NT_IMAGE_1_NAME L"CDBOOT1.IMG"
#define NT_IMAGE_2_NAME L"CDBOOT2.IMG"
#define NT_IMAGE_3_NAME L"CDBOOT3.IMG"
#define NT_IMAGE_4_NAME L"CDBOOT4.IMG"
#define NT_IMAGE_5_NAME L"CDBOOT5.IMG"
#define NT_IMAGE_6_NAME L"CDBOOT6.IMG"
#define NUMBER_OF_ASCII_WHEEL_SYMBOLS 4
const WCHAR rgAsciiWheel[NUMBER_OF_ASCII_WHEEL_SYMBOLS] = { '|', '/', '-', '\\' };
//
// Function prototypes
//
BOOL WriteImageToFloppy( WCHAR *szFileName, WCHAR *DrivePath ); VOID PrintErrorMessage( VOID ); VOID PrintErrorWrongDriveType( UINT iDriveType ); BOOL IsDriveLargeEnough( WCHAR *DrivePath ); VOID FreeStrings( VOID ); VOID LoadStrings( VOID ); INT DoImageFilesExist( VOID ) ; VOID CleanUp( HANDLE *hFloppyDrive, HANDLE *hFloppyImage ); BOOL DoesUserWantToTryCopyAgain( VOID ); LPWSTR MyLoadString( UINT StringId ); VOID print( WCHAR *szFirstString, ... ); BOOL DoesFileExist( LPWSTR lpFileName ); VOID PressAnyKeyToContinue( VOID ); void SetFarEastThread(); void ConsolePrint( WCHAR *szFirstString, ... ); //
// Global Strings
//
WCHAR *StrOutOfMemory = NULL; WCHAR *StrComplete = NULL; WCHAR *StrNtVersionName = NULL; WCHAR *StrCanNotFindFile = NULL; WCHAR *StrDiskLabel1 = NULL; WCHAR *StrDiskLabel2 = NULL; WCHAR *StrDiskLabel3 = NULL; WCHAR *StrDiskLabel4 = NULL; WCHAR *StrDiskLabel5 = NULL; WCHAR *StrDiskLabel6 = NULL;
WCHAR *StrStars = NULL; WCHAR *StrExplanationLine1 = NULL; WCHAR *StrExplanationLine2 = NULL; WCHAR *StrExplanationLine3 = NULL; WCHAR *StrExplanationLine4 = NULL; WCHAR *StrInsertFirstDiskLine1 = NULL; WCHAR *StrInsertFirstDiskLine2 = NULL; WCHAR *StrInsertAnotherDiskLine1 = NULL; WCHAR *StrInsertAnotherDiskLine2 = NULL; WCHAR *StrPressAnyKeyWhenReady = NULL; WCHAR *StrCompletedSuccessfully = NULL;
//----------------------------------------------------------------------------
//
// Function: wmain
//
// Purpose: Instructs user to insert floppy disks to be copied and performs
// the copy.
//
// Arguments: int argc - standard program argument, count of the command line args
// char *argv[] - standard program argument, the 2nd argument is the
// floppy drive to copy the images to.
//
// Returns: INT - zero on successful program completion
// - non-zero on unsuccessful program completion, program
// terminated because of an error
//
//----------------------------------------------------------------------------
INT __cdecl wmain( INT argc, WCHAR *argv[] ) {
WCHAR *szOsName; WCHAR DriveLetter; WCHAR Drive[10]; WCHAR DrivePath[10]; UINT iDriveType; BOOL bTryAgain;
szOsName = _wgetenv( L"OS" );
//
// Make sure we are on NT.
//
if( ( szOsName == NULL ) || ( _wcsicmp( szOsName, L"Windows_NT" ) != 0 ) ) {
// ******
// This string cannot be localized because if we are not on NT then
// we don't have wide chars.
//
printf( "This program only runs on Windows NT, Windows 2000 and Windows XP.\n" );
exit( 1 );
}
SetFarEastThread();
//
// Load all of the strings from the resource file
//
LoadStrings();
//
// Don't allow the system to do any pop-ups. We will handle all
// error messages
//
SetErrorMode( SEM_FAILCRITICALERRORS );
print( L"" );
print( StrStars );
print( StrExplanationLine1 );
print( StrExplanationLine2, StrNtVersionName );
print( StrExplanationLine3 ); print( StrExplanationLine4 );
print( L"" );
//
// If they didn't specified the floppy drive on the command line then
// prompt them for it.
//
if( argc == 1 ) {
WCHAR *StrSpecifyFloppyDrive = MyLoadString( IDS_SPECIFY_FLOPPY_DRIVE );
ConsolePrint( L"%s", StrSpecifyFloppyDrive );
DriveLetter = (WCHAR)_getche();
ConsolePrint( L"\n\n" );
free( StrSpecifyFloppyDrive );
} else {
DriveLetter = argv[1][0];
}
//
// Make sure the character they entered is a possible drive letter
//
if( ! isalpha( DriveLetter ) ) {
WCHAR *StrInvalidDriveLetter = MyLoadString( IDS_INVALID_DRIVE_LETTER );
ConsolePrint( L"%s\n", StrInvalidDriveLetter );
free( StrInvalidDriveLetter );
exit( 1 );
}
//
// Make sure all the image files are in the current directory.
//
if( ! DoImageFilesExist() ) {
exit( 1 );
}
//
// Make the char DriveLetter into a string
//
Drive[0] = DriveLetter; Drive[1] = L'\0';
//
// Build the drive path. For example the a: drive looks like \\.\a:
//
swprintf( DrivePath, L"\\\\.\\%c:", DriveLetter );
//
// Make sure the drive is a floppy drive
//
iDriveType = GetDriveType( wcscat( Drive, L":\\" ) );
if( iDriveType != DRIVE_REMOVABLE ) {
PrintErrorWrongDriveType( iDriveType );
exit( 1 );
}
//
// Make sure the drive can hold at least 1.44 MB
//
if( ! IsDriveLargeEnough( DrivePath ) ) {
WCHAR *Str144NotSupported = MyLoadString( IDS_144_NOT_SUPPORTED );
ConsolePrint( L"%s\n", Str144NotSupported );
free( Str144NotSupported );
exit( 1 );
}
print( StrInsertFirstDiskLine1, DriveLetter ); print( StrInsertFirstDiskLine2, StrDiskLabel1 );
PressAnyKeyToContinue();
while( ! WriteImageToFloppy( NT_IMAGE_1_NAME, DrivePath ) ) {
bTryAgain = DoesUserWantToTryCopyAgain();
if( ! bTryAgain ) { exit( 1 ); }
}
print( L"" );
print( StrInsertAnotherDiskLine1, DriveLetter ); print( StrInsertAnotherDiskLine2, StrDiskLabel2 );
PressAnyKeyToContinue();
while( ! WriteImageToFloppy( NT_IMAGE_2_NAME, DrivePath ) ) {
bTryAgain = DoesUserWantToTryCopyAgain();
if( ! bTryAgain ) { exit( 1 ); }
}
print( L"" );
print( StrInsertAnotherDiskLine1, DriveLetter ); print( StrInsertAnotherDiskLine2, StrDiskLabel3 );
PressAnyKeyToContinue();
while( ! WriteImageToFloppy( NT_IMAGE_3_NAME, DrivePath ) ) {
bTryAgain = DoesUserWantToTryCopyAgain();
if( ! bTryAgain ) { exit( 1 ); }
}
print( L"" );
print( StrInsertAnotherDiskLine1, DriveLetter ); print( StrInsertAnotherDiskLine2, StrDiskLabel4 );
PressAnyKeyToContinue();
while( ! WriteImageToFloppy( NT_IMAGE_4_NAME, DrivePath ) ) {
bTryAgain = DoesUserWantToTryCopyAgain();
if( ! bTryAgain ) { exit( 1 ); }
}
print( L"" );
print( StrInsertAnotherDiskLine1, DriveLetter ); print( StrInsertAnotherDiskLine2, StrDiskLabel5 );
PressAnyKeyToContinue();
while( ! WriteImageToFloppy( NT_IMAGE_5_NAME, DrivePath ) ) {
bTryAgain = DoesUserWantToTryCopyAgain();
if( ! bTryAgain ) { exit( 1 ); }
}
print( L"" );
print( StrInsertAnotherDiskLine1, DriveLetter ); print( StrInsertAnotherDiskLine2, StrDiskLabel6 );
PressAnyKeyToContinue();
while( ! WriteImageToFloppy( NT_IMAGE_6_NAME, DrivePath ) ) {
bTryAgain = DoesUserWantToTryCopyAgain();
if( ! bTryAgain ) { exit( 1 ); }
}
print( L"" );
print( StrCompletedSuccessfully );
print( StrStars );
FreeStrings();
return( 0 );
}
//----------------------------------------------------------------------------
//
// Function: WriteImageToFloppy
//
// Purpose: Writes an image file to a floppy disk. Handles all error
// reporting to the user.
//
// Arguments: char *szFileName - filename to write to the floppy
// char *DrivePath - drive path of the floppy to write to, it is of
// the form \\.\x where x is the drive letter
//
// Returns: BOOL - TRUE if image written to floppy properly
// FALSE if there were errors
//
//----------------------------------------------------------------------------
BOOL WriteImageToFloppy( WCHAR *szFileName, WCHAR *DrivePath ) {
INT iCurrentTrack; INT cBytesRead = 0; INT cBytesWritten = 0; INT iPercentComplete = 0; INT iWheelPosition = 0; HANDLE hFloppyImage = NULL; HANDLE hFloppyDrive = NULL; char TrackBuffer[TRACK_SIZE];
hFloppyImage = CreateFile( szFileName, GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL );
if( hFloppyImage == INVALID_HANDLE_VALUE ) {
PrintErrorMessage();
return( FALSE );
}
hFloppyDrive = CreateFile( DrivePath, GENERIC_WRITE, 0, NULL, OPEN_EXISTING, FILE_FLAG_SEQUENTIAL_SCAN, NULL );
if( hFloppyDrive == INVALID_HANDLE_VALUE ) {
PrintErrorMessage();
CleanUp( &hFloppyDrive, &hFloppyImage );
return( FALSE );
}
for( iCurrentTrack = 0; iCurrentTrack < TRACKS_ON_DISK; iCurrentTrack++ ) { if( ! ReadFile( hFloppyImage, TrackBuffer, TRACK_SIZE, &cBytesRead, NULL ) ) {
PrintErrorMessage();
CleanUp( &hFloppyDrive, &hFloppyImage );
return( FALSE );
}
if( ! WriteFile( hFloppyDrive, TrackBuffer, TRACK_SIZE, &cBytesWritten, NULL ) ) {
PrintErrorMessage();
CleanUp( &hFloppyDrive, &hFloppyImage );
return( FALSE );
}
iPercentComplete = (int) ( ( (double) (iCurrentTrack) / (double) (TRACKS_ON_DISK) ) * 100.0 );
ConsolePrint( L"%c %3d%% %s.\r", rgAsciiWheel[iWheelPosition], iPercentComplete, StrComplete );
//
// Advance the ASCII wheel
//
iWheelPosition++;
if( iWheelPosition >= NUMBER_OF_ASCII_WHEEL_SYMBOLS ) { iWheelPosition = 0; }
}
//
// We are done copying the disk so force it to read 100% and get rid of
// the ascii wheel symbol.
//
ConsolePrint( L" 100%% %s. \n", StrComplete );
//
// Free allocated resources
//
CleanUp( &hFloppyDrive, &hFloppyImage );
return TRUE;
}
//----------------------------------------------------------------------------
//
// Function: PrintErrorMessage
//
// Purpose: To get the last system error, look up what it is and print it out
// to the user.
//
// Arguments: VOID
//
// Returns: VOID
//
//----------------------------------------------------------------------------
VOID PrintErrorMessage( VOID ) {
LPVOID lpMsgBuf = NULL;
if(!FormatMessage( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPWSTR) &lpMsgBuf, 0, NULL )) { // Great. Not enough memory to format an error message.
ConsolePrint( L"\nNot enough memory to format error message.\n" ); if( lpMsgBuf ) LocalFree( lpMsgBuf ); } else {
ConsolePrint( L"\n%s\n", (LPCWSTR)lpMsgBuf );
LocalFree( lpMsgBuf ); }
}
//----------------------------------------------------------------------------
//
// Function: PrintErrorWrongDriveType
//
// Purpose: To translate a drive type error code into a message and print it
//
// Arguments: UINT iDriveType - drive type error code to look-up
//
// Returns: VOID
//
//----------------------------------------------------------------------------
VOID PrintErrorWrongDriveType( UINT iDriveType ) {
if( iDriveType == DRIVE_NO_ROOT_DIR ) { WCHAR *StrNoRootDir = MyLoadString( IDS_NO_ROOT_DIR );
ConsolePrint( L"\n%s\n", StrNoRootDir );
free( StrNoRootDir ); } else { WCHAR *StrDriveNotFloppy = MyLoadString( IDS_DRIVE_NOT_FL0PPY );
ConsolePrint( L"\n%s\n", StrDriveNotFloppy );
free( StrDriveNotFloppy ); }
}
//----------------------------------------------------------------------------
//
// Function: IsDriveLargeEnough
//
// Purpose: To determine if the floppy drive supports 1.44 MB or larger disks
//
// Arguments: char* DrivePath - drive path of the floppy to write to, it is of
// the form \\.\x where x is the drive letter
//
// Returns: BOOL - TRUE if the drive supports 1.44 MB or greater, FALSE if not
//
//----------------------------------------------------------------------------
BOOL IsDriveLargeEnough( WCHAR *DrivePath ) {
UINT i; HANDLE hFloppyDrive; DISK_GEOMETRY SupportedGeometry[20]; DWORD SupportedGeometryCount; DWORD ReturnedByteCount;
hFloppyDrive = CreateFile( DrivePath, 0, FILE_SHARE_READ, NULL, OPEN_ALWAYS, 0, NULL );
if( hFloppyDrive == INVALID_HANDLE_VALUE ) {
PrintErrorMessage();
exit( 1 );
}
if( DeviceIoControl( hFloppyDrive, IOCTL_DISK_GET_MEDIA_TYPES, NULL, 0, SupportedGeometry, sizeof( SupportedGeometry ), &ReturnedByteCount, NULL ) ) {
SupportedGeometryCount = ( ReturnedByteCount / sizeof( DISK_GEOMETRY ) );
} else { SupportedGeometryCount = 0; }
CloseHandle( hFloppyDrive );
for( i = 0; i < SupportedGeometryCount; i++ ) {
if( SupportedGeometry[i].MediaType == F3_1Pt44_512 ) {
//
// This drive supports 3.5, 1.44MB, 512 bytes/sector.
//
return( TRUE );
}
}
return( FALSE );
}
//----------------------------------------------------------------------------
//
// Function: DoImageFilesExist
//
// Purpose: Determines if all the image files are in the current directory or
// not. If an image file is missing, an error message is printed
// to the user.
//
// Note: it detemines if a file exists by seeing if it can open it
// for reading.
//
// Arguments: VOID
//
// Returns: INT -- non-zero on success, all images files exist in current dir
// zero on failure, 1 or more image files do not exist
//
//----------------------------------------------------------------------------
INT DoImageFilesExist( VOID ) {
BOOL bAllFilesExist = TRUE;
if( ! DoesFileExist( NT_IMAGE_1_NAME ) ) { print( StrCanNotFindFile, NT_IMAGE_1_NAME ); bAllFilesExist = FALSE; } if( ! DoesFileExist( NT_IMAGE_2_NAME ) ) { print( StrCanNotFindFile, NT_IMAGE_2_NAME ); bAllFilesExist = FALSE; } if( ! DoesFileExist( NT_IMAGE_3_NAME ) ) { print( StrCanNotFindFile, NT_IMAGE_3_NAME ); bAllFilesExist = FALSE; } if( ! DoesFileExist( NT_IMAGE_4_NAME ) ) { print( StrCanNotFindFile, NT_IMAGE_4_NAME ); bAllFilesExist = FALSE; } if( ! DoesFileExist( NT_IMAGE_5_NAME ) ) { print( StrCanNotFindFile, NT_IMAGE_5_NAME ); bAllFilesExist = FALSE; } if( ! DoesFileExist( NT_IMAGE_6_NAME ) ) { print( StrCanNotFindFile, NT_IMAGE_6_NAME ); bAllFilesExist = FALSE; }
if( bAllFilesExist ) { return( 1 ); } else { return( 0 ); }
}
//----------------------------------------------------------------------------
//
// Function: CleanUp
//
// Purpose: Closes open handles. This function should be called just before
// exiting the program.
//
// Arguments: HANDLE *hFloppyDrive - the floppy disk handle to be closed
// HANDLE *hFloppyImage - the floppy image file handle to be closed
//
// Returns: VOID
//
//----------------------------------------------------------------------------
VOID CleanUp( HANDLE *hFloppyDrive, HANDLE *hFloppyImage ) {
if( *hFloppyDrive ) { CloseHandle( *hFloppyDrive ); *hFloppyDrive = NULL; }
if( *hFloppyImage ) { CloseHandle( *hFloppyImage ); *hFloppyImage = NULL; }
}
//----------------------------------------------------------------------------
//
// Function: FreeStrings
//
// Purpose: Deallocate the memory for all the strings.
//
// Arguments: VOID
//
// Returns: VOID
//
//----------------------------------------------------------------------------
VOID FreeStrings( VOID ) {
free( StrNtVersionName ); free( StrDiskLabel1 ); free( StrDiskLabel2 ); free( StrDiskLabel3 ); free( StrDiskLabel4 ); free( StrDiskLabel5 ); free( StrDiskLabel6 );
free( StrStars ); free( StrExplanationLine1 ); free( StrExplanationLine2 ); free( StrExplanationLine3 ); free( StrExplanationLine4 ); free( StrInsertFirstDiskLine1 ); free( StrInsertFirstDiskLine2 ); free( StrInsertAnotherDiskLine1 ); free( StrInsertAnotherDiskLine2 ); free( StrPressAnyKeyWhenReady ); free( StrCompletedSuccessfully ); free( StrComplete );
free( StrCanNotFindFile );
free( StrOutOfMemory );
}
//----------------------------------------------------------------------------
//
// Function: LoadStrings
//
// Purpose: Load the string constants from the string table.
//
// Arguments: VOID
//
// Returns: VOID
//
//----------------------------------------------------------------------------
VOID LoadStrings( VOID ) {
INT Length;
//
// Loading the Out of Memory string is a little tricky because of the
// error that can occur while loading it.
//
// For the other strings, just call the MyLoadString function to do
// the work
//
StrOutOfMemory = (WCHAR *) malloc( MAKEBOOT_MAX_STRING_LEN * sizeof(WCHAR) + 1 );
if( StrOutOfMemory == NULL ) { //
// Can't localize this string
//
ConsolePrint( L"\nOut of memory. Please free more memory and run this program again.\n" );
exit( 1 ); }
Length = LoadString( NULL, IDS_OUT_OF_MEMORY, StrOutOfMemory, MAKEBOOT_MAX_STRING_LEN );
if( ! Length ) { //
// Can't localize this string
//
ConsolePrint( L"Unable to load resources.\n" );
exit( 1 ) ; }
StrStars = MyLoadString( IDS_STARS ); StrExplanationLine1 = MyLoadString( IDS_PROGRAM_EXPLANATION_LINE_1 ); StrExplanationLine2 = MyLoadString( IDS_PROGRAM_EXPLANATION_LINE_2 ); StrExplanationLine3 = MyLoadString( IDS_PROGRAM_EXPLANATION_LINE_3 ); StrExplanationLine4 = MyLoadString( IDS_PROGRAM_EXPLANATION_LINE_4 );
StrInsertFirstDiskLine1 = MyLoadString( IDS_INSERT_FIRST_DISK_LINE_1 ); StrInsertFirstDiskLine2 = MyLoadString( IDS_INSERT_FIRST_DISK_LINE_2 );
StrInsertAnotherDiskLine1 = MyLoadString( IDS_INSERT_ANOTHER_DISK_LINE_1 ); StrInsertAnotherDiskLine2 = MyLoadString( IDS_INSERT_ANOTHER_DISK_LINE_2 );
StrPressAnyKeyWhenReady = MyLoadString( IDS_PRESS_ANY_KEY_WHEN_READY );
StrCompletedSuccessfully = MyLoadString( IDS_COMPLETED_SUCCESSFULLY ); StrComplete = MyLoadString( IDS_COMPLETE );
StrCanNotFindFile = MyLoadString( IDS_CANNOT_FIND_FILE );
StrNtVersionName = MyLoadString( IDS_NT_VERSION_NAME_DEFAULT ); StrDiskLabel1 = MyLoadString( IDS_DISK_LABEL_1_DEFAULT ); StrDiskLabel2 = MyLoadString( IDS_DISK_LABEL_2_DEFAULT ); StrDiskLabel3 = MyLoadString( IDS_DISK_LABEL_3_DEFAULT ); StrDiskLabel4 = MyLoadString( IDS_DISK_LABEL_4_DEFAULT ); StrDiskLabel5 = MyLoadString( IDS_DISK_LABEL_5_DEFAULT ); StrDiskLabel6 = MyLoadString( IDS_DISK_LABEL_6_DEFAULT );
}
//----------------------------------------------------------------------------
//
// Function: DoesUserWantToTryCopyAgain
//
// Purpose: Ask the user if they want to retry to copy the image to floppy.
// Get the user input and return whether to copy again or not.
//
// Arguments: VOID
//
// Returns: BOOL - TRUE if user wants to attempt to copy again
// - FALSE if user does not want to attempt to copy again
//
//----------------------------------------------------------------------------
BOOL DoesUserWantToTryCopyAgain( VOID ) {
INT ch;
WCHAR *StrAttemptToCreateFloppyAgain = MyLoadString( IDS_ATTEMPT_TO_CREATE_FLOPPY_AGAIN ); WCHAR *StrPressEnterOrEsc = MyLoadString( IDS_PRESS_ENTER_OR_ESC );
//
// Clear the input stream by eating all the chars until there are none
// left. Print the message and then wait for a key press.
//
while( _kbhit() ) { _getch(); }
do { ConsolePrint( L"%s\n", StrAttemptToCreateFloppyAgain ); ConsolePrint( L"%s\n", StrPressEnterOrEsc );
ch = _getch();
} while( ch != ENTER_KEY && ch != ESC_KEY );
if( ch == ENTER_KEY ) { return( TRUE ); } else { return( FALSE ); }
free( StrAttemptToCreateFloppyAgain ); free( StrPressEnterOrEsc );
}
//----------------------------------------------------------------------------
//
// Function: PressAnyKeyToContinue
//
// Purpose: Print the "Press any key when ready" message and wait until the
// user presses a key.
//
// Arguments: VOID
//
// Returns: VOID
//
//----------------------------------------------------------------------------
VOID PressAnyKeyToContinue( VOID ) {
//
// Clear the input stream by eating all the chars until there are none
// left. Print the message and then wait for a key press.
//
while( _kbhit() ) { _getch(); }
print( StrPressAnyKeyWhenReady );
//
// Spin until the keyboard is pressed
//
while( ! _kbhit() ) { ; }
}
//----------------------------------------------------------------------------
//
// Function: MyLoadString
//
// Purpose: Loads a string resource given it's IDS_* and returns
// a malloc'ed buffer with its contents.
//
// The malloc()'ed buffer must be freed with free()
//
// This function will either return the string or exit. It will
// never return NULL or a bad pointer.
//
// Arguments: UINT StringId - the string ID to load
//
// Returns:
// Pointer to buffer. An empty string is returned if the StringId
// does not exist. Null is returned if out of memory.
//
//----------------------------------------------------------------------------
LPWSTR MyLoadString( UINT StringId ) {
WCHAR Buffer[ MAKEBOOT_MAX_STRING_LEN ]; WCHAR *String = NULL; UINT Length;
Length = LoadString( NULL, StringId, Buffer, MAKEBOOT_MAX_STRING_LEN );
if( Length ) {
String = (WCHAR *) malloc( Length * sizeof(WCHAR) + 1 );
if( String == NULL ) {
ConsolePrint( L"%s\n", StrOutOfMemory );
exit( 1 );
} else {
wcscpy( String, Buffer ); String[Length] = L'\0';
return( String );
}
} else {
//
// Can't load the string so exit
// NOTE: this string will not be localized
//
ConsolePrint( L"Unable to load resources.\n" );
exit( 1 );
}
}
//----------------------------------------------------------------------------
//
// Function: print
//
// Purpose: To print out strings to the user. Useful when there is
// embedded formatting characters in a string that was loaded from
// a string table.
//
// Arguments: szFirstString - the string that contains the embedded formatting
// characters (such as %s, %c, etc.)
// ... - variable number of arguments that correspond to each
// formatting character
//
// Returns: VOID
//
//----------------------------------------------------------------------------
VOID print( WCHAR *szFirstString, ... ) {
WCHAR OutputBuffer[MAKEBOOT_MAX_STRING_LEN]; va_list arglist;
va_start( arglist, szFirstString );
wvsprintf( OutputBuffer, szFirstString, arglist );
ConsolePrint( L"%s\n", OutputBuffer );
va_end( arglist );
}
//----------------------------------------------------------------------------
//
// Function: DoesFileExist
//
// Purpose: To determine whether a file exists or not
//
// Arguments: LPWSTR lpFileName - filename to see if it exists or not
//
// Returns: BOOL - TRUE if the file exists
// - FALSE if the file does not exist
//
//----------------------------------------------------------------------------
BOOL DoesFileExist( LPWSTR lpFileName ) { DWORD dwAttribs = GetFileAttributes( lpFileName );
if( dwAttribs == (DWORD) -1 ) { return( FALSE ); }
if( dwAttribs & FILE_ATTRIBUTE_DIRECTORY ) { return( FALSE ); }
return( TRUE ); }
//----------------------------------------------------------------------------
//
// Function: IsDBCSConsole
//
// Purpose: To determine whether a DBC console or not
//
// Arguments: None
//
// Returns: BOOL - TRUE if FE console codepage
// - FALSE if not FE console codepage
//
//----------------------------------------------------------------------------
BOOL IsDBCSCodePage(UINT CodePage) { switch(CodePage) { case 932: case 936: case 949: case 950: return TRUE; } return FALSE; }
//----------------------------------------------------------------------------
//
// Function: SetFarEastThread
//
// Purpose: FarEast version wants to display bi-lingual string according console OCP
//
// Arguments: None
//
// Returns: None
//----------------------------------------------------------------------------
void SetFarEastThread() { LANGID LangId = 0;
switch(GetConsoleOutputCP()) { case 932: LangId = MAKELANGID( LANG_JAPANESE, SUBLANG_DEFAULT ); break; case 949: LangId = MAKELANGID( LANG_KOREAN, SUBLANG_KOREAN ); break; case 936: LangId = MAKELANGID( LANG_CHINESE, SUBLANG_CHINESE_SIMPLIFIED ); break; case 950: LangId = MAKELANGID( LANG_CHINESE, SUBLANG_CHINESE_TRADITIONAL ); break; default: { LANGID TmpLangId = PRIMARYLANGID(LANGIDFROMLCID( GetUserDefaultLCID() ));
if (TmpLangId == LANG_JAPANESE || TmpLangId == LANG_KOREAN || TmpLangId == LANG_CHINESE ) { LangId = MAKELANGID( LANG_ENGLISH, SUBLANG_ENGLISH_US ); } } break; }
if (LangId) { SetThreadLocale( MAKELCID(LangId, SORT_DEFAULT) ); } }
//----------------------------------------------------------------------------
//
// Function: ConsolePrint
//
// Purpose: There is a bug in CRT library that unicode FE characters can't
// convert correctly, so we output characters directly.
//
// Arguments: None
//
// Returns: None
//----------------------------------------------------------------------------
void ConsolePrint( WCHAR *szFirstString, ... ) { HANDLE StdOut; DWORD WrittenCount; WCHAR OutputBuffer[MAKEBOOT_MAX_STRING_LEN]; va_list arglist;
if((StdOut = GetStdHandle(STD_OUTPUT_HANDLE)) == INVALID_HANDLE_VALUE) { return; }
va_start( arglist, szFirstString );
wvsprintf( OutputBuffer, szFirstString, arglist );
WriteConsoleW( StdOut, OutputBuffer, lstrlenW(OutputBuffer), &WrittenCount, NULL );
va_end( arglist ); }
|